diff --git a/core/.phpstan-baseline.php b/core/.phpstan-baseline.php
index 29807a4d23194f2da99c58d6747bc24fb95e42d8..12213a5e2a3dfad6bd93c758b34d9d8311523877 100644
--- a/core/.phpstan-baseline.php
+++ b/core/.phpstan-baseline.php
@@ -22224,21 +22224,9 @@
 ];
 $ignoreErrors[] = [
 	// identifier: missingType.return
-	'message' => '#^Method Drupal\\\\Tests\\\\jsonapi\\\\Kernel\\\\Normalizer\\\\JsonApiDocumentTopLevelNormalizerTest\\:\\:createImageField\\(\\) has no return type specified\\.$#',
+	'message' => '#^Method Drupal\\\\Tests\\\\jsonapi\\\\Kernel\\\\Normalizer\\\\JsonApiTopLevelResourceNormalizerTest\\:\\:createImageField\\(\\) has no return type specified\\.$#',
 	'count' => 1,
-	'path' => __DIR__ . '/modules/jsonapi/tests/src/Kernel/Normalizer/JsonApiDocumentTopLevelNormalizerTest.php',
-];
-$ignoreErrors[] = [
-	// identifier: missingType.return
-	'message' => '#^Method Drupal\\\\Tests\\\\jsonapi\\\\Kernel\\\\Normalizer\\\\JsonApiDocumentTopLevelNormalizerTest\\:\\:getNormalizer\\(\\) has no return type specified\\.$#',
-	'count' => 1,
-	'path' => __DIR__ . '/modules/jsonapi/tests/src/Kernel/Normalizer/JsonApiDocumentTopLevelNormalizerTest.php',
-];
-$ignoreErrors[] = [
-	// identifier: missingType.return
-	'message' => '#^Method Drupal\\\\Tests\\\\jsonapi\\\\Kernel\\\\Normalizer\\\\JsonApiDocumentTopLevelNormalizerTest\\:\\:testCacheableMetadataProvider\\(\\) has no return type specified\\.$#',
-	'count' => 1,
-	'path' => __DIR__ . '/modules/jsonapi/tests/src/Kernel/Normalizer/JsonApiDocumentTopLevelNormalizerTest.php',
+	'path' => __DIR__ . '/modules/jsonapi/tests/src/Kernel/Normalizer/JsonApiTopLevelResourceNormalizerTest.php',
 ];
 $ignoreErrors[] = [
 	// identifier: missingType.return
diff --git a/core/lib/Drupal/Core/Serialization/Attribute/JsonSchema.php b/core/lib/Drupal/Core/Serialization/Attribute/JsonSchema.php
new file mode 100644
index 0000000000000000000000000000000000000000..8a2ef5fa2d0a803c930b30b89231ef23db9c0eef
--- /dev/null
+++ b/core/lib/Drupal/Core/Serialization/Attribute/JsonSchema.php
@@ -0,0 +1,36 @@
+<?php
+
+declare(strict_types=1);
+
+namespace Drupal\Core\Serialization\Attribute;
+
+/**
+ * Attribute for methods to express the JSON Schema of its return value.
+ *
+ * This attribute may be repeated to define multiple potential types.
+ */
+#[\Attribute(\Attribute::TARGET_METHOD | \Attribute::IS_REPEATABLE)]
+class JsonSchema {
+
+  /**
+   * Constructor.
+   *
+   * @param array $schema
+   *   Schema.
+   */
+  public function __construct(
+    public readonly array $schema = [],
+  ) {
+  }
+
+  /**
+   * Get a JSON Schema type definition array.
+   *
+   * @return array
+   *   Type definition.
+   */
+  public function getJsonSchema(): array {
+    return $this->schema;
+  }
+
+}
diff --git a/core/lib/Drupal/Core/Template/Attribute.php b/core/lib/Drupal/Core/Template/Attribute.php
index 52485e6a538a4071d44641f34f9c2fb385b2831b..77a56689219df28092b462afd8e950b68188df4f 100644
--- a/core/lib/Drupal/Core/Template/Attribute.php
+++ b/core/lib/Drupal/Core/Template/Attribute.php
@@ -5,6 +5,7 @@
 use Drupal\Component\Render\PlainTextOutput;
 use Drupal\Component\Render\MarkupInterface;
 use Drupal\Component\Utility\NestedArray;
+use Drupal\Core\Serialization\Attribute\JsonSchema;
 
 /**
  * Collects, sanitizes, and renders HTML attributes.
@@ -320,6 +321,7 @@ public function hasClass($class) {
   /**
    * Implements the magic __toString() method.
    */
+  #[JsonSchema(['type' => 'string', 'description' => 'Rendered HTML element attributes'])]
   public function __toString() {
     $return = '';
     /** @var \Drupal\Core\Template\AttributeValueBase $value */
diff --git a/core/lib/Drupal/Core/TypedData/Plugin/DataType/BooleanData.php b/core/lib/Drupal/Core/TypedData/Plugin/DataType/BooleanData.php
index 124129ad9f40d1a0d828fcf47ee03e8c67a0290c..8dbab870de445a351e2d4aa0c6b94e23b92084af 100644
--- a/core/lib/Drupal/Core/TypedData/Plugin/DataType/BooleanData.php
+++ b/core/lib/Drupal/Core/TypedData/Plugin/DataType/BooleanData.php
@@ -2,6 +2,7 @@
 
 namespace Drupal\Core\TypedData\Plugin\DataType;
 
+use Drupal\Core\Serialization\Attribute\JsonSchema;
 use Drupal\Core\StringTranslation\TranslatableMarkup;
 use Drupal\Core\TypedData\Attribute\DataType;
 use Drupal\Core\TypedData\PrimitiveBase;
@@ -22,6 +23,7 @@ class BooleanData extends PrimitiveBase implements BooleanInterface {
   /**
    * {@inheritdoc}
    */
+  #[JsonSchema(['type' => 'boolean'])]
   public function getCastedValue() {
     return (bool) $this->value;
   }
diff --git a/core/lib/Drupal/Core/TypedData/Plugin/DataType/DecimalData.php b/core/lib/Drupal/Core/TypedData/Plugin/DataType/DecimalData.php
index e03459704f0a12c5ef8ce61b7efa402bedc64e1e..03efb21504c10eb5b1a75db382127981ee409a63 100644
--- a/core/lib/Drupal/Core/TypedData/Plugin/DataType/DecimalData.php
+++ b/core/lib/Drupal/Core/TypedData/Plugin/DataType/DecimalData.php
@@ -2,6 +2,7 @@
 
 namespace Drupal\Core\TypedData\Plugin\DataType;
 
+use Drupal\Core\Serialization\Attribute\JsonSchema;
 use Drupal\Core\StringTranslation\TranslatableMarkup;
 use Drupal\Core\TypedData\Attribute\DataType;
 use Drupal\Core\TypedData\Type\DecimalInterface;
@@ -22,6 +23,7 @@ class DecimalData extends StringData implements DecimalInterface {
   /**
    * {@inheritdoc}
    */
+  #[JsonSchema(['type' => 'string', 'format' => 'number'])]
   public function getCastedValue() {
     return $this->getString() ?: '0.0';
   }
diff --git a/core/lib/Drupal/Core/TypedData/Plugin/DataType/DurationIso8601.php b/core/lib/Drupal/Core/TypedData/Plugin/DataType/DurationIso8601.php
index 50a5201a6aa2619ee6dbb32160cfcc7853649590..d72c1c7083bba4113404b099530ca5ed469acda3 100644
--- a/core/lib/Drupal/Core/TypedData/Plugin/DataType/DurationIso8601.php
+++ b/core/lib/Drupal/Core/TypedData/Plugin/DataType/DurationIso8601.php
@@ -2,6 +2,7 @@
 
 namespace Drupal\Core\TypedData\Plugin\DataType;
 
+use Drupal\Core\Serialization\Attribute\JsonSchema;
 use Drupal\Core\StringTranslation\TranslatableMarkup;
 use Drupal\Core\TypedData\Attribute\DataType;
 use Drupal\Core\TypedData\Type\DurationInterface;
@@ -28,6 +29,22 @@ public function getDuration() {
     }
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  #[JsonSchema(['type' => 'string', 'format' => 'duration'])]
+  public function getDurationAsIso8601Abnf(): string {
+    return $this->value;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  #[JsonSchema(['type' => 'string', 'format' => 'duration'])]
+  public function getCastedValue() {
+    return parent::getCastedValue();
+  }
+
   /**
    * {@inheritdoc}
    */
diff --git a/core/lib/Drupal/Core/TypedData/Plugin/DataType/Email.php b/core/lib/Drupal/Core/TypedData/Plugin/DataType/Email.php
index 6a74ce755cdfa4fd4a24d042a35f1c802c3704e2..91407d1e3151480d7f7ac10f170e5240511cf12a 100644
--- a/core/lib/Drupal/Core/TypedData/Plugin/DataType/Email.php
+++ b/core/lib/Drupal/Core/TypedData/Plugin/DataType/Email.php
@@ -2,6 +2,7 @@
 
 namespace Drupal\Core\TypedData\Plugin\DataType;
 
+use Drupal\Core\Serialization\Attribute\JsonSchema;
 use Drupal\Core\StringTranslation\TranslatableMarkup;
 use Drupal\Core\TypedData\Attribute\DataType;
 use Drupal\Core\TypedData\Type\StringInterface;
@@ -18,4 +19,12 @@
 )]
 class Email extends StringData implements StringInterface {
 
+  /**
+   * {@inheritdoc}
+   */
+  #[JsonSchema(['type' => 'string', 'format' => 'email'])]
+  public function getCastedValue() {
+    return parent::getCastedValue();
+  }
+
 }
diff --git a/core/lib/Drupal/Core/TypedData/Plugin/DataType/FloatData.php b/core/lib/Drupal/Core/TypedData/Plugin/DataType/FloatData.php
index 3bec2e46f20bd6b90c69219f4ddc00d0d4daa3d1..060cad9e186acca387e6c67be12a846b09510b53 100644
--- a/core/lib/Drupal/Core/TypedData/Plugin/DataType/FloatData.php
+++ b/core/lib/Drupal/Core/TypedData/Plugin/DataType/FloatData.php
@@ -2,6 +2,7 @@
 
 namespace Drupal\Core\TypedData\Plugin\DataType;
 
+use Drupal\Core\Serialization\Attribute\JsonSchema;
 use Drupal\Core\StringTranslation\TranslatableMarkup;
 use Drupal\Core\TypedData\Attribute\DataType;
 use Drupal\Core\TypedData\PrimitiveBase;
@@ -22,6 +23,7 @@ class FloatData extends PrimitiveBase implements FloatInterface {
   /**
    * {@inheritdoc}
    */
+  #[JsonSchema(['type' => 'number'])]
   public function getCastedValue() {
     return (float) $this->value;
   }
diff --git a/core/lib/Drupal/Core/TypedData/Plugin/DataType/IntegerData.php b/core/lib/Drupal/Core/TypedData/Plugin/DataType/IntegerData.php
index bef643ee2c7fd678d278c0facd41e8109b0f024e..ed0733e9c2cfd6341f8e8cdc6e9a7826f69364e9 100644
--- a/core/lib/Drupal/Core/TypedData/Plugin/DataType/IntegerData.php
+++ b/core/lib/Drupal/Core/TypedData/Plugin/DataType/IntegerData.php
@@ -2,6 +2,7 @@
 
 namespace Drupal\Core\TypedData\Plugin\DataType;
 
+use Drupal\Core\Serialization\Attribute\JsonSchema;
 use Drupal\Core\StringTranslation\TranslatableMarkup;
 use Drupal\Core\TypedData\Attribute\DataType;
 use Drupal\Core\TypedData\PrimitiveBase;
@@ -22,6 +23,7 @@ class IntegerData extends PrimitiveBase implements IntegerInterface {
   /**
    * {@inheritdoc}
    */
+  #[JsonSchema(['type' => 'integer'])]
   public function getCastedValue() {
     return (int) $this->value;
   }
diff --git a/core/lib/Drupal/Core/TypedData/Plugin/DataType/StringData.php b/core/lib/Drupal/Core/TypedData/Plugin/DataType/StringData.php
index c1832e24e400e5630d56daa326c63be892e2d65b..8823d9771ff1ea76d64bdb0b2943a5e713a7ef5c 100644
--- a/core/lib/Drupal/Core/TypedData/Plugin/DataType/StringData.php
+++ b/core/lib/Drupal/Core/TypedData/Plugin/DataType/StringData.php
@@ -2,6 +2,7 @@
 
 namespace Drupal\Core\TypedData\Plugin\DataType;
 
+use Drupal\Core\Serialization\Attribute\JsonSchema;
 use Drupal\Core\StringTranslation\TranslatableMarkup;
 use Drupal\Core\TypedData\Attribute\DataType;
 use Drupal\Core\TypedData\PrimitiveBase;
@@ -22,6 +23,7 @@ class StringData extends PrimitiveBase implements StringInterface {
   /**
    * {@inheritdoc}
    */
+  #[JsonSchema(['type' => 'string'])]
   public function getCastedValue() {
     return $this->getString();
   }
diff --git a/core/lib/Drupal/Core/TypedData/Plugin/DataType/TimeSpan.php b/core/lib/Drupal/Core/TypedData/Plugin/DataType/TimeSpan.php
index b488558f8ca975c5c35556f720533005028c1986..112bdd7356cff079cefbd7ec8876533c2e04b1f1 100644
--- a/core/lib/Drupal/Core/TypedData/Plugin/DataType/TimeSpan.php
+++ b/core/lib/Drupal/Core/TypedData/Plugin/DataType/TimeSpan.php
@@ -2,6 +2,7 @@
 
 namespace Drupal\Core\TypedData\Plugin\DataType;
 
+use Drupal\Core\Serialization\Attribute\JsonSchema;
 use Drupal\Core\StringTranslation\TranslatableMarkup;
 use Drupal\Core\TypedData\Attribute\DataType;
 use Drupal\Core\TypedData\Type\DurationInterface;
@@ -30,10 +31,18 @@ public function getDuration() {
     if ($this->value) {
       // Keep the duration in seconds as there is generally no valid way to
       // convert it to days, months or years.
-      return new \DateInterval('PT' . $this->value . 'S');
+      return new \DateInterval($this->getDurationAsIso8601Abnf());
     }
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  #[JsonSchema(['type' => 'string', 'format' => 'duration'])]
+  public function getDurationAsIso8601Abnf(): string {
+    return 'PT' . $this->value . 'S';
+  }
+
   /**
    * {@inheritdoc}
    */
diff --git a/core/lib/Drupal/Core/TypedData/Plugin/DataType/Uri.php b/core/lib/Drupal/Core/TypedData/Plugin/DataType/Uri.php
index 50a9b2c4bdb65d55f16adfddb3b4e8561f8ab609..5c6c1dc06d881772a318ef903250f32640544808 100644
--- a/core/lib/Drupal/Core/TypedData/Plugin/DataType/Uri.php
+++ b/core/lib/Drupal/Core/TypedData/Plugin/DataType/Uri.php
@@ -2,6 +2,7 @@
 
 namespace Drupal\Core\TypedData\Plugin\DataType;
 
+use Drupal\Core\Serialization\Attribute\JsonSchema;
 use Drupal\Core\StringTranslation\TranslatableMarkup;
 use Drupal\Core\TypedData\Attribute\DataType;
 use Drupal\Core\TypedData\Type\UriInterface;
@@ -17,4 +18,12 @@
 )]
 class Uri extends StringData implements UriInterface {
 
+  /**
+   * {@inheritdoc}
+   */
+  #[JsonSchema(['type' => 'string', 'format' => 'uri'])]
+  public function getCastedValue() {
+    return parent::getCastedValue();
+  }
+
 }
diff --git a/core/lib/Drupal/Core/TypedData/Type/DurationInterface.php b/core/lib/Drupal/Core/TypedData/Type/DurationInterface.php
index 90d6192246252366ba1f1165536784959e9c0285..5a1ff9a99d6a4ca5d37eed0cd0c1d76ca76c6a5b 100644
--- a/core/lib/Drupal/Core/TypedData/Type/DurationInterface.php
+++ b/core/lib/Drupal/Core/TypedData/Type/DurationInterface.php
@@ -19,6 +19,16 @@ interface DurationInterface {
    */
   public function getDuration();
 
+  /**
+   * Returns the duration as an ISO 8601 ABNF string.
+   *
+   * @return string
+   *   ABNF-formatted duration.
+   *
+   * @see https://datatracker.ietf.org/doc/html/rfc3339#appendix-A
+   */
+  public function getDurationAsIso8601Abnf(): string;
+
   /**
    * Sets the duration.
    *
diff --git a/core/misc/cspell/dictionary.txt b/core/misc/cspell/dictionary.txt
index ffb7877c3036a88a504de617f95e79797026d7d2..ec7da3aed475fbf91acad4d488478a3ac00fdfc2 100644
--- a/core/misc/cspell/dictionary.txt
+++ b/core/misc/cspell/dictionary.txt
@@ -1,3 +1,4 @@
+abnf
 absolutezero
 adamson
 addedline
diff --git a/core/modules/jsonapi/jsonapi.services.yml b/core/modules/jsonapi/jsonapi.services.yml
index 90c4854205bc36346af9ff35605b8b04b0f057d7..fcab4b2bd2cfa2c9f32032e62438da8e085e36a8 100644
--- a/core/modules/jsonapi/jsonapi.services.yml
+++ b/core/modules/jsonapi/jsonapi.services.yml
@@ -49,6 +49,8 @@ services:
     arguments:
       - '@jsonapi.normalization_cacher'
       - '@event_dispatcher'
+      - '@entity_field.manager'
+      - '@entity_type.manager'
     tags:
       - { name: jsonapi_normalizer }
   jsonapi.normalization_cacher:
@@ -80,6 +82,7 @@ services:
     class: Drupal\jsonapi\Normalizer\RelationshipNormalizer
     tags:
       - { name: jsonapi_normalizer }
+    arguments: ['@jsonapi.resource_type.repository']
   serializer.encoder.jsonapi:
     class: Drupal\jsonapi\Encoder\JsonEncoder
     tags:
diff --git a/core/modules/jsonapi/schema.json b/core/modules/jsonapi/schema.json
index ebd5eb2a9c034b1a1e19d0e1cc3f8150bedc4953..8c2d439ce711e61b362066a370a64d8b5a2ec1ec 100644
--- a/core/modules/jsonapi/schema.json
+++ b/core/modules/jsonapi/schema.json
@@ -1,158 +1,492 @@
 {
-  "$schema": "http://json-schema.org/draft-06/schema#",
-  "title": "JSON:API Schema",
-  "description": "This is a schema for responses in the JSON:API format. For more, see http://jsonapi.org",
-  "oneOf": [
+  "$schema": "https://json-schema.org/draft/2020-12/schema",
+  "$id": "https://jsonapi.org/schemas/spec/v1.1/draft",
+  "title": "JSON:API Schema v1.1",
+  "description": "This schema only validates RESPONSES from a request.",
+  "allOf": [
     {
-      "$ref": "#/definitions/success"
+      "$ref": "#/definitions/requiredTopLevelMembers"
     },
     {
-      "$ref": "#/definitions/failure"
-    },
+      "$ref": "#/definitions/oneOfDataOrErrors"
+    }
+  ],
+  "anyOf": [
     {
-      "$ref": "#/definitions/info"
+      "$ref": "#/definitions/atMemberName"
     }
   ],
-
+  "type": "object",
+  "properties": {
+    "data": {
+      "$ref": "#/definitions/data"
+    },
+    "errors": {
+      "$ref": "#/definitions/errors"
+    },
+    "included": {
+      "$ref": "#/definitions/included"
+    },
+    "jsonapi": {
+      "$ref": "#/definitions/jsonapi"
+    },
+    "links": {
+      "description": "Link members related to the primary data.",
+      "$ref": "#/definitions/topLevelLinks"
+    },
+    "meta": {
+      "$ref": "#/definitions/meta"
+    }
+  },
+  "dependencies": {
+    "included": [
+      "data"
+    ]
+  },
+  "unevaluatedProperties": false,
   "definitions": {
-    "success": {
+    "atMemberName": {
+      "description": "@member name may contain any valid JSON value.",
       "type": "object",
-      "required": [
-        "data"
+      "patternProperties": {
+        "^@[a-zA-Z0-9]{1}(?:[-\\w]*[a-zA-Z0-9])?$": true
+      }
+    },
+    "memberName": {
+      "description": "Member name may contain any valid JSON value.",
+      "type": "object",
+      "patternProperties": {
+        "^[a-zA-Z0-9]{1}(?:[-\\w]*[a-zA-Z0-9])?$": true
+      }
+    },
+    "attributes": {
+      "description": "Members of the attributes object (\"attributes\") represent information about the resource object in which it's defined.",
+      "type": "object",
+      "anyOf": [
+        {
+          "$ref": "#/definitions/memberName"
+        },
+        {
+          "$ref": "#/definitions/atMemberName"
+        }
+      ],
+      "not": {
+        "$comment": "This is what the specification requires, but it seems bad. https://github.com/json-api/json-api/issues/1553",
+        "anyOf": [
+          {
+            "required": [
+              "type"
+            ]
+          },
+          {
+            "required": [
+              "id"
+            ]
+          }
+        ]
+      },
+      "unevaluatedProperties": false
+    },
+    "data": {
+      "description": "The document's \"primary data\" is a representation of the resource or collection of resources targeted by a request.",
+      "oneOf": [
+        {
+          "$ref": "#/definitions/resource"
+        },
+        {
+          "description": "An array of resource objects, an array of resource identifier objects, or an empty array ([]), for requests that target resource collections.",
+          "$ref": "#/definitions/resourceCollection"
+        },
+        {
+          "description": "null if the request is one that might correspond to a single resource, but doesn't currently.",
+          "type": "null"
+        }
+      ]
+    },
+    "empty": {
+      "description": "Describes an empty to-one relationship.",
+      "type": "null"
+    },
+    "error": {
+      "type": "object",
+      "anyOf": [
+        {
+          "required": [
+            "id"
+          ]
+        },
+        {
+          "required": [
+            "links"
+          ]
+        },
+        {
+          "required": [
+            "status"
+          ]
+        },
+        {
+          "required": [
+            "code"
+          ]
+        },
+        {
+          "required": [
+            "title"
+          ]
+        },
+        {
+          "required": [
+            "detail"
+          ]
+        },
+        {
+          "required": [
+            "source"
+          ]
+        },
+        {
+          "required": [
+            "meta"
+          ]
+        }
       ],
       "properties": {
-        "data": {
-          "$ref": "#/definitions/data"
+        "id": {
+          "description": "A unique identifier for this particular occurrence of the problem.",
+          "type": "string"
         },
-        "included": {
-          "description": "To reduce the number of HTTP requests, servers **MAY** allow responses that include related resources along with the requested primary resources. Such responses are called \"compound documents\".",
-          "type": "array",
-          "items": {
-            "$ref": "#/definitions/resource"
-          },
-          "uniqueItems": true
+        "links": {
+          "$ref": "#/definitions/errorLinks"
+        },
+        "status": {
+          "description": "The HTTP status code applicable to this problem, expressed as a string value.",
+          "type": "string"
+        },
+        "code": {
+          "description": "An application-specific error code, expressed as a string value.",
+          "type": "string"
+        },
+        "title": {
+          "description": "A short, human-readable summary of the problem. It **SHOULD NOT** change from occurrence to occurrence of the problem, except for purposes of localization.",
+          "type": "string"
+        },
+        "detail": {
+          "description": "A human-readable explanation specific to this occurrence of the problem.",
+          "type": "string"
+        },
+        "source": {
+          "$ref": "#/definitions/errorSource"
         },
         "meta": {
           "$ref": "#/definitions/meta"
+        }
+      },
+      "unevaluatedProperties": false
+    },
+    "errorLinks": {
+      "description": "The error links object **MAY** contain the following members: about.",
+      "type": "object",
+      "allOf": [
+        {
+          "$ref": "#/definitions/links"
+        }
+      ],
+      "anyOf": [
+        {
+          "$ref": "#/definitions/atMemberName"
+        }
+      ],
+      "properties": {
+        "about": {
+          "description": "A link that leads to further details about this particular occurrence of the problem.",
+          "$ref": "#/definitions/link"
         },
-        "links": {
-          "description": "Link members related to the primary data.",
-          "allOf": [
-            {
-              "$ref": "#/definitions/links"
-            },
-            {
-              "$ref": "#/definitions/pagination"
-            }
-          ]
+        "type": {
+          "description": "A link that identifies the type of error that this particular error is an instance of.",
+          "$ref": "#/definitions/link"
+        }
+      },
+      "unevaluatedProperties": false
+    },
+    "errorSource": {
+      "type": "object",
+      "properties": {
+        "pointer": {
+          "description": "A JSON Pointer [RFC6901] to the associated entity in the request document [e.g. \"/data\" for a primary data object, or \"/data/attributes/title\" for a specific attribute].",
+          "type": "string",
+          "pattern": "^(?:\\/(?:[^~/]|~0|~1)*)*$"
         },
-        "jsonapi": {
-          "$ref": "#/definitions/jsonapi"
+        "parameter": {
+          "description": "A string indicating which query parameter caused the error.",
+          "type": "string"
+        },
+        "header": {
+          "description": "A string indicating the name of a single request header which caused the error.",
+          "type": "string"
         }
+      }
+    },
+    "errors": {
+      "type": "array",
+      "items": {
+        "$ref": "#/definitions/error"
       },
-      "additionalProperties": false
+      "uniqueItems": true
     },
-    "failure": {
+    "included": {
+      "description": "To reduce the number of HTTP requests, servers **MAY** allow responses that include related resources along with the requested primary resources. Such responses are called \"compound documents\".",
+      "type": "array",
+      "items": {
+        "$ref": "#/definitions/resource"
+      },
+      "uniqueItems": true
+    },
+    "jsonapi": {
+      "description": "An object describing the server's implementation",
       "type": "object",
-      "required": [
-        "errors"
+      "anyOf": [
+        {
+          "$ref": "#/definitions/atMemberName"
+        }
       ],
       "properties": {
-        "errors": {
-          "type": "array",
-          "items": {
-            "$ref": "#/definitions/error"
-          },
-          "uniqueItems": true
+        "version": {
+          "type": "string"
         },
         "meta": {
           "$ref": "#/definitions/meta"
         },
-        "jsonapi": {
-          "$ref": "#/definitions/jsonapi"
+        "ext": {
+          "type": "array",
+          "items": {
+            "$ref": "#/definitions/linkUrl"
+          },
+          "uniqueItems": true
         },
-        "links": {
-          "$ref": "#/definitions/links"
+        "profile": {
+          "type": "array",
+          "items": {
+            "$ref": "#/definitions/linkUrl"
+          },
+          "uniqueItems": true
         }
       },
-      "additionalProperties": false
+      "unevaluatedProperties": false
     },
-    "info": {
+    "linkObject": {
       "type": "object",
       "required": [
-        "meta"
+        "href"
       ],
       "properties": {
+        "href": {
+          "$ref": "#/definitions/linkUrl"
+        },
         "meta": {
           "$ref": "#/definitions/meta"
         },
-        "links": {
-          "$ref": "#/definitions/links"
+        "rel": {
+          "type": "string"
+        },
+        "title": {
+          "type": "string"
+        },
+        "type": {
+          "type": "string"
         },
-        "jsonapi": {
-          "$ref": "#/definitions/jsonapi"
+        "hreflang": {
+          "type": "string"
+        },
+        "describedby": {
+          "$ref": "#/definitions/link"
         }
-      },
-      "additionalProperties": false
+      }
+    },
+    "linkUrl": {
+      "description": "A string containing the link's URL.",
+      "type": "string",
+      "format": "uri",
+      "$comment": "URI regex as per https://tools.ietf.org/html/rfc3986#appendix-B",
+      "pattern": "^(([^:/?#]+):)?(//([^/?#]*))?([^?#]*)(\\?([^#]*))?(#(.*))?"
+    },
+    "link": {
+      "description": "A link **MUST** be represented as either: a string containing the link's URL or a link object.",
+      "oneOf": [
+        {
+          "$ref": "#/definitions/linkUrl"
+        },
+        {
+          "$ref": "#/definitions/linkObject"
+        }
+      ]
+    },
+    "linkage": {
+      "description": "Resource linkage in a compound document allows a client to link together all of the included resource objects without having to GET any URLs via links.",
+      "oneOf": [
+        {
+          "$ref": "#/definitions/relationshipToOne"
+        },
+        {
+          "$ref": "#/definitions/relationshipToMany"
+        }
+      ]
+    },
+    "links": {
+      "type": "object"
     },
-
     "meta": {
       "description": "Non-standard meta-information that can not be represented as an attribute or relationship.",
       "type": "object",
-      "additionalProperties": true
-    },
-    "data": {
-      "description": "The document's \"primary data\" is a representation of the resource or collection of resources targeted by a request.",
-      "oneOf": [
+      "anyOf": [
         {
-          "$ref": "#/definitions/resource"
+          "$ref": "#/definitions/memberName"
         },
         {
-          "description": "An array of resource objects, an array of resource identifier objects, or an empty array ([]), for requests that target resource collections.",
-          "type": "array",
-          "items": {
-            "$ref": "#/definitions/resource"
-          },
-          "uniqueItems": true
+          "$ref": "#/definitions/atMemberName"
+        }
+      ],
+      "unevaluatedProperties": false
+    },
+    "oneOfDataOrErrors": {
+      "type": "object",
+      "dependentSchemas": {
+        "data": {
+          "not": {
+            "required": [
+              "errors"
+            ]
+          }
+        }
+      }
+    },
+    "pagination": {
+      "type": "object",
+      "properties": {
+        "first": {
+          "description": "The first page of data",
+          "oneOf": [
+            {
+              "$ref": "#/definitions/link"
+            },
+            {
+              "type": "null"
+            }
+          ]
+        },
+        "last": {
+          "description": "The last page of data",
+          "oneOf": [
+            {
+              "$ref": "#/definitions/link"
+            },
+            {
+              "type": "null"
+            }
+          ]
+        },
+        "prev": {
+          "description": "The previous page of data",
+          "oneOf": [
+            {
+              "$ref": "#/definitions/link"
+            },
+            {
+              "type": "null"
+            }
+          ]
         },
+        "next": {
+          "description": "The next page of data",
+          "oneOf": [
+            {
+              "$ref": "#/definitions/link"
+            },
+            {
+              "type": "null"
+            }
+          ]
+        }
+      }
+    },
+    "relationship": {
+      "type": "object",
+      "properties": {
+        "links": {
+          "$ref": "#/definitions/relationshipLinks"
+        },
+        "data": {
+          "$ref": "#/definitions/linkage"
+        },
+        "meta": {
+          "$ref": "#/definitions/meta"
+        }
+      },
+      "allOf": [
         {
-          "description": "null if the request is one that might correspond to a single resource, but doesn't currently.",
-          "type": "null"
+          "anyOf": [
+            {
+              "required": [
+                "data"
+              ]
+            },
+            {
+              "required": [
+                "meta"
+              ]
+            },
+            {
+              "required": [
+                "links"
+              ]
+            }
+          ]
         }
-      ]
+      ],
+      "anyOf": [
+        {
+          "$ref": "#/definitions/atMemberName"
+        }
+      ],
+      "unevaluatedProperties": false
     },
-    "resource": {
-      "description": "\"Resource objects\" appear in a JSON:API document to represent resources.",
+    "relationshipFromRequest": {
       "type": "object",
-      "required": [
-        "type",
-        "id"
-      ],
       "properties": {
-        "type": {
-          "type": "string"
-        },
-        "id": {
-          "type": "string"
-        },
-        "attributes": {
-          "$ref": "#/definitions/attributes"
-        },
-        "relationships": {
-          "$ref": "#/definitions/relationships"
-        },
-        "links": {
-          "$ref": "#/definitions/links"
+        "data": {
+          "$ref": "#/definitions/linkage"
         },
         "meta": {
           "$ref": "#/definitions/meta"
         }
       },
-      "additionalProperties": false
+      "required": [
+        "data"
+      ],
+      "anyOf": [
+        {
+          "$ref": "#/definitions/atMemberName"
+        }
+      ],
+      "unevaluatedProperties": false
     },
     "relationshipLinks": {
       "description": "A resource object **MAY** contain references to other resource objects (\"relationships\"). Relationships may be to-one or to-many. Relationships can be specified by including a member in a resource's links object.",
       "type": "object",
+      "allOf": [
+        {
+          "$ref": "#/definitions/links"
+        }
+      ],
+      "anyOf": [
+        {
+          "description": "Pagination links for the relationship data.",
+          "$ref": "#/definitions/pagination"
+        },
+        {
+          "$ref": "#/definitions/atMemberName"
+        }
+      ],
       "properties": {
         "self": {
           "description": "A `self` member, whose value is a URL for the relationship itself (a \"relationship URL\"). This URL allows the client to directly manipulate the relationship. For example, it would allow a client to remove an `author` from an `article` without deleting the people resource itself.",
@@ -162,221 +496,272 @@
           "$ref": "#/definitions/link"
         }
       },
-      "additionalProperties": true
+      "unevaluatedProperties": false
     },
-    "links": {
-      "type": "object",
-      "additionalProperties": {
-        "$ref": "#/definitions/link"
+    "relationshipToMany": {
+      "description": "An array of objects each containing \"type\" and \"id\" members for to-many relationships.",
+      "type": "array",
+      "items": {
+        "$ref": "#/definitions/resourceIdentifier"
       }
     },
-    "link": {
-      "description": "A link **MUST** be represented as either: a string containing the link's URL or a link object.",
+    "relationshipToOne": {
+      "description": "References to other resource objects in a to-one (\"relationship\"). Relationships can be specified by including a member in a resource's links object.",
       "oneOf": [
         {
-          "description": "A string containing the link's URL.",
-          "type": "string",
-          "format": "uri-reference"
+          "$ref": "#/definitions/empty"
         },
         {
-          "type": "object",
-          "required": [
-            "href"
-          ],
-          "properties": {
-            "href": {
-              "description": "A string containing the link's URL.",
-              "type": "string",
-              "format": "uri-reference"
-            },
-            "meta": {
-              "$ref": "#/definitions/meta"
-            }
-          }
+          "$ref": "#/definitions/resourceIdentifier"
         }
       ]
     },
-
-    "attributes": {
-      "description": "Members of the attributes object (\"attributes\") represent information about the resource object in which it's defined.",
+    "relationships": {
+      "description": "Members of the relationships object (\"relationships\") represent references from the resource object in which it's defined to other resource objects.",
       "type": "object",
+      "allOf": [
+        {
+          "$ref": "#/definitions/relationshipsForbiddenMemberName"
+        }
+      ],
+      "anyOf": [
+        {
+          "$ref": "#/definitions/atMemberName"
+        }
+      ],
       "patternProperties": {
-        "^(?!relationships$|links$|id$|type$)\\w[-\\w_]*$": {
-          "description": "Attributes may contain any valid JSON value."
+        "^[a-zA-Z0-9]{1}(?:[-\\w]*[a-zA-Z0-9])?$": {
+          "$ref": "#/definitions/relationship"
         }
       },
-      "additionalProperties": false
+      "unevaluatedProperties": false
     },
-
-    "relationships": {
+    "relationshipsFromRequest": {
       "description": "Members of the relationships object (\"relationships\") represent references from the resource object in which it's defined to other resource objects.",
       "type": "object",
+      "allOf": [
+        {
+          "$ref": "#/definitions/relationshipsForbiddenMemberName"
+        }
+      ],
+      "anyOf": [
+        {
+          "$ref": "#/definitions/atMemberName"
+        }
+      ],
       "patternProperties": {
-        "^(?!id$|type$)\\w[-\\w_]*$": {
-          "properties": {
-            "links": {
-              "$ref": "#/definitions/relationshipLinks"
-            },
-            "data": {
-              "description": "Member, whose value represents \"resource linkage\".",
-              "oneOf": [
-                {
-                  "$ref": "#/definitions/relationshipToOne"
-                },
-                {
-                  "$ref": "#/definitions/relationshipToMany"
-                }
-              ]
-            },
-            "meta": {
-              "$ref": "#/definitions/meta"
-            }
-          },
-          "anyOf": [
-            {"required": ["data"]},
-            {"required": ["meta"]},
-            {"required": ["links"]}
-          ],
-          "additionalProperties": false
+        "^[a-zA-Z0-9]{1}(?:[-\\w]*[a-zA-Z0-9])?$": {
+          "$ref": "#/definitions/relationshipFromRequest"
         }
       },
-      "additionalProperties": false
+      "unevaluatedProperties": false
     },
-    "relationshipToOne": {
-      "description": "References to other resource objects in a to-one (\"relationship\"). Relationships can be specified by including a member in a resource's links object.",
+    "relationshipsForbiddenMemberName": {
+      "not": {
+        "anyOf": [
+          {
+            "type": "object",
+            "required": [
+              "type"
+            ]
+          },
+          {
+            "type": "object",
+            "required": [
+              "id"
+            ]
+          }
+        ]
+      }
+    },
+    "requiredTopLevelMembers": {
       "anyOf": [
         {
-          "$ref": "#/definitions/empty"
+          "type": "object",
+          "required": [
+            "meta"
+          ]
         },
         {
-          "$ref": "#/definitions/linkage"
+          "type": "object",
+          "required": [
+            "data"
+          ]
+        },
+        {
+          "type": "object",
+          "required": [
+            "errors"
+          ]
         }
       ]
     },
-    "relationshipToMany": {
-      "description": "An array of objects each containing \"type\" and \"id\" members for to-many relationships.",
+    "resource": {
+      "description": "\"Resource objects\" appear in a JSON:API document to represent resources.",
+      "$comment": "The id member is not required when the resource object originates at the client and represents a new resource to be created on the server.",
+      "type": "object",
+      "allOf": [
+        {
+          "$ref": "#/definitions/resourceIdentification"
+        }
+      ],
+      "properties": {
+        "attributes": {
+          "$ref": "#/definitions/attributes"
+        },
+        "links": {
+          "$ref": "#/definitions/resourceLinks"
+        },
+        "meta": {
+          "$ref": "#/definitions/meta"
+        },
+        "relationships": {
+          "$ref": "#/definitions/relationships"
+        }
+      },
+      "unevaluatedProperties": false
+    },
+    "resourceCollection": {
+      "description": "An array of resource objects.",
       "type": "array",
       "items": {
-        "$ref": "#/definitions/linkage"
+        "$ref": "#/definitions/resource"
       },
       "uniqueItems": true
     },
-    "empty": {
-      "description": "Describes an empty to-one relationship.",
-      "type": "null"
+    "resourceIdentification": {
+      "allOf": [
+        {
+          "$ref": "#/definitions/resourceIdentificationNew"
+        },
+        {
+          "type": "object",
+          "required": [
+            "type",
+            "id"
+          ]
+        },
+        {
+          "not": {
+            "type": "object",
+            "required": [
+              "lid"
+            ]
+          }
+        }
+      ]
     },
-    "linkage": {
-      "description": "The \"type\" and \"id\" to non-empty members.",
+    "resourceIdentificationNew": {
+      "$comment": "The id member is not required when the resource object originates at the client and represents a new resource to be created on the server.",
       "type": "object",
       "required": [
-        "type",
-        "id"
+        "type"
       ],
       "properties": {
         "type": {
-          "type": "string"
+          "type": "string",
+          "pattern": "^[a-zA-Z0-9]{1}(?:[-\\w]*[a-zA-Z0-9])?$"
         },
         "id": {
           "type": "string"
         },
-        "meta": {
-          "$ref": "#/definitions/meta"
+        "lid": {
+          "type": "string"
         }
       },
-      "additionalProperties": false
+      "anyOf": [
+        {
+          "$ref": "#/definitions/atMemberName"
+        }
+      ]
     },
-    "pagination": {
+    "resourceIdentifier": {
+      "description": "A \"resource identifier object\" is an object that identifies an individual resource.",
       "type": "object",
-      "properties": {
-        "first": {
-          "description": "The first page of data",
-          "oneOf": [
-            { "$ref": "#/definitions/link" },
-            { "type": "null" }
-          ]
-        },
-        "last": {
-          "description": "The last page of data",
-          "oneOf": [
-            { "$ref": "#/definitions/link" },
-            { "type": "null" }
-          ]
+      "allOf": [
+        {
+          "$ref": "#/definitions/resourceIdentificationNew"
         },
-        "prev": {
-          "description": "The previous page of data",
-          "oneOf": [
-            { "$ref": "#/definitions/link" },
-            { "type": "null" }
+        {
+          "required": [
+            "type"
           ]
         },
-        "next": {
-          "description": "The next page of data",
+        {
           "oneOf": [
-            { "$ref": "#/definitions/link" },
-            { "type": "null" }
+            {
+              "required": [
+                "id"
+              ]
+            },
+            {
+              "required": [
+                "lid"
+              ]
+            }
           ]
         }
-      }
-    },
-
-    "jsonapi": {
-      "description": "An object describing the server's implementation",
-      "type": "object",
+      ],
       "properties": {
-        "version": {
-          "type": "string"
-        },
         "meta": {
           "$ref": "#/definitions/meta"
         }
       },
-      "additionalProperties": false
+      "unevaluatedProperties": false
     },
-
-    "error": {
+    "resourceLinks": {
+      "description": "The top-level links object **MAY** contain the following members: self, related, pagination links.",
       "type": "object",
+      "allOf": [
+        {
+          "$ref": "#/definitions/links"
+        }
+      ],
+      "anyOf": [
+        {
+          "$ref": "#/definitions/atMemberName"
+        }
+      ],
       "properties": {
-        "id": {
-          "description": "A unique identifier for this particular occurrence of the problem.",
-          "type": "string"
-        },
-        "links": {
+        "self": {
+          "description": "",
+          "$ref": "#/definitions/link"
+        }
+      },
+      "unevaluatedProperties": false
+    },
+    "topLevelLinks": {
+      "description": "The top-level links object **MAY** contain the following members: self, related, pagination links.",
+      "type": "object",
+      "allOf": [
+        {
           "$ref": "#/definitions/links"
+        }
+      ],
+      "anyOf": [
+        {
+          "description": "Pagination links for the primary data.",
+          "$ref": "#/definitions/pagination"
         },
-        "status": {
-          "description": "The HTTP status code applicable to this problem, expressed as a string value.",
-          "type": "string"
-        },
-        "code": {
-          "description": "An application-specific error code, expressed as a string value.",
-          "type": "string"
-        },
-        "title": {
-          "description": "A short, human-readable summary of the problem. It **SHOULD NOT** change from occurrence to occurrence of the problem, except for purposes of localization.",
-          "type": "string"
-        },
-        "detail": {
-          "description": "A human-readable explanation specific to this occurrence of the problem.",
-          "type": "string"
+        {
+          "$ref": "#/definitions/atMemberName"
+        }
+      ],
+      "properties": {
+        "self": {
+          "description": "The link that generated the current response document.",
+          "$ref": "#/definitions/link"
         },
-        "source": {
-          "type": "object",
-          "properties": {
-            "pointer": {
-              "description": "A JSON Pointer [RFC6901] to the associated entity in the request document [e.g. \"/data\" for a primary data object, or \"/data/attributes/title\" for a specific attribute].",
-              "type": "string"
-            },
-            "parameter": {
-              "description": "A string indicating which query parameter caused the error.",
-              "type": "string"
-            }
-          }
+        "related": {
+          "description": "A related resource link when the primary data represents a resource relationship.",
+          "$ref": "#/definitions/link"
         },
-        "meta": {
-          "$ref": "#/definitions/meta"
+        "describedby": {
+          "description": "A link to a description document (e.g. OpenAPI or JSON Schema) for the current document.",
+          "$ref": "#/definitions/link"
         }
       },
-      "additionalProperties": false
+      "unevaluatedProperties": false
     }
   }
 }
diff --git a/core/modules/jsonapi/src/JsonApiResource/Data.php b/core/modules/jsonapi/src/JsonApiResource/Data.php
index bac3a7fc5f39c103ce8c6013f25c0508c203826c..f132929503dc57966d60fb5e0ee482bc64752312 100644
--- a/core/modules/jsonapi/src/JsonApiResource/Data.php
+++ b/core/modules/jsonapi/src/JsonApiResource/Data.php
@@ -100,8 +100,8 @@ public function setTotalCount($count) {
   /**
    * Returns the collection as an array.
    *
-   * @return \Drupal\Core\Entity\EntityInterface[]
-   *   The array of entities.
+   * @return \Drupal\jsonapi\JsonApiResource\ResourceIdentifierInterface[]
+   *   Array of contained data.
    */
   public function toArray() {
     return $this->data;
diff --git a/core/modules/jsonapi/src/JsonApiSpec.php b/core/modules/jsonapi/src/JsonApiSpec.php
index 274b3f4deb1736b5df2ca68314cbf0cfc20b785d..bb06be289d7d1a69da364fc0fefef356c1c4fd5e 100644
--- a/core/modules/jsonapi/src/JsonApiSpec.php
+++ b/core/modules/jsonapi/src/JsonApiSpec.php
@@ -20,12 +20,17 @@ class JsonApiSpec {
    *
    * @see http://jsonapi.org/format/#document-jsonapi-object
    */
-  const SUPPORTED_SPECIFICATION_VERSION = '1.0';
+  const SUPPORTED_SPECIFICATION_VERSION = '1.1';
 
   /**
    * The URI of the supported specification document.
    */
-  const SUPPORTED_SPECIFICATION_PERMALINK = 'http://jsonapi.org/format/1.0/';
+  const SUPPORTED_SPECIFICATION_PERMALINK = 'http://jsonapi.org/format/1.1/';
+
+  /**
+   * The URI of the supported specification's JSON Schema.
+   */
+  const SUPPORTED_SPECIFICATION_JSON_SCHEMA = 'https://jsonapi.org/schemas/spec/v1.1/draft';
 
   /**
    * Member name: globally allowed characters.
diff --git a/core/modules/jsonapi/src/Normalizer/FieldItemNormalizer.php b/core/modules/jsonapi/src/Normalizer/FieldItemNormalizer.php
index b8089ccf6a9e6d681d6ce9e4bfa41304fe65c909..1ec6ee141926559869f8156d51437a2259100b36 100644
--- a/core/modules/jsonapi/src/Normalizer/FieldItemNormalizer.php
+++ b/core/modules/jsonapi/src/Normalizer/FieldItemNormalizer.php
@@ -12,7 +12,10 @@
 use Drupal\jsonapi\Normalizer\Value\CacheableNormalization;
 use Drupal\jsonapi\ResourceType\ResourceType;
 use Drupal\serialization\Normalizer\CacheableNormalizerInterface;
+use Drupal\serialization\Normalizer\JsonSchemaReflectionTrait;
+use Drupal\serialization\Normalizer\SchematicNormalizerTrait;
 use Drupal\serialization\Normalizer\SerializedColumnNormalizerTrait;
+use Drupal\serialization\Serializer\JsonSchemaProviderSerializerInterface;
 use Symfony\Component\Serializer\Exception\UnexpectedValueException;
 use Symfony\Component\Serializer\Normalizer\DenormalizerInterface;
 
@@ -28,6 +31,8 @@
 class FieldItemNormalizer extends NormalizerBase implements DenormalizerInterface {
 
   use SerializedColumnNormalizerTrait;
+  use SchematicNormalizerTrait;
+  use JsonSchemaReflectionTrait;
 
   /**
    * The entity type manager.
@@ -54,25 +59,24 @@ public function __construct(EntityTypeManagerInterface $entity_type_manager) {
    * cacheability in mind, and hence bubbles cacheability out of band. This must
    * catch it, and pass it to the value object that JSON:API uses.
    */
-  public function normalize($field_item, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
-    assert($field_item instanceof FieldItemInterface);
+  public function doNormalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
+    assert($object instanceof FieldItemInterface);
     /** @var \Drupal\Core\TypedData\TypedDataInterface $property */
-    $values = [];
     $context[CacheableNormalizerInterface::SERIALIZATION_CONTEXT_CACHEABILITY] = new CacheableMetadata();
-    if (!empty($field_item->getProperties(TRUE))) {
+    // Default: The field has only internal (or no) properties but has a public
+    // value.
+    $values = $object->getValue();
+    // There are non-internal properties. Normalize those.
+    if ($field_properties = TypedDataInternalPropertiesHelper::getNonInternalProperties($object)) {
       // We normalize each individual value, so each can do their own casting,
       // if needed.
-      $field_properties = TypedDataInternalPropertiesHelper::getNonInternalProperties($field_item);
-      foreach ($field_properties as $property_name => $property) {
-        $values[$property_name] = $this->serializer->normalize($property, $format, $context);
-      }
+      $values = array_map(function ($property) use ($format, $context) {
+        return $this->serializer->normalize($property, $format, $context);
+      }, $field_properties);
       // Flatten if there is only a single property to normalize.
-      $flatten = count($field_properties) === 1 && $field_item::mainPropertyName() !== NULL;
+      $flatten = count($field_properties) === 1 && $object::mainPropertyName() !== NULL;
       $values = static::rasterizeValueRecursive($flatten ? reset($values) : $values);
     }
-    else {
-      $values = $field_item->getValue();
-    }
     $normalization = new CacheableNormalization(
       $context[CacheableNormalizerInterface::SERIALIZATION_CONTEXT_CACHEABILITY],
       $values
@@ -233,6 +237,45 @@ protected function getFieldItemInstance(ResourceType $resource_type, FieldItemDa
     return $field_item;
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  protected function getNormalizationSchema(mixed $object, array $context = []): array {
+    $schema = ['type' => 'object'];
+    if (is_string($object)) {
+      return ['$comment' => 'No detailed schema available.'] + $schema;
+    }
+    assert($object instanceof FieldItemInterface);
+    $field_properties = TypedDataInternalPropertiesHelper::getNonInternalProperties($object);
+    if (count($field_properties) === 0) {
+      // The field item has only internal (or no) properties. In this case, the
+      // value is normalized from ::getValue(). Use a schema from the method or
+      // interface, if available.
+      return $this->getJsonSchemaForMethod(
+        $object,
+        'getValue',
+        ['$comment' => sprintf('Cannot determine schema for %s::getValue().', $object::class)]
+      );
+    }
+    // If we did not early return, iterate over the non-internal properties.
+    foreach ($field_properties as $property_name => $property) {
+      $property_schema = [
+        'title' => (string) $property->getDataDefinition()->getLabel(),
+      ];
+      assert($this->serializer instanceof JsonSchemaProviderSerializerInterface);
+      $property_schema = array_merge(
+        $this->serializer->getJsonSchema($property, $context),
+        $property_schema,
+      );
+      $schema['properties'][$property_name] = $property_schema;
+    }
+    // Flatten if there is only a single property to normalize.
+    if (count($field_properties) === 1 && $object::mainPropertyName() !== NULL) {
+      $schema = $schema['properties'][$object::mainPropertyName()] ?? [];
+    }
+    return $schema;
+  }
+
   /**
    * {@inheritdoc}
    */
diff --git a/core/modules/jsonapi/src/Normalizer/FieldNormalizer.php b/core/modules/jsonapi/src/Normalizer/FieldNormalizer.php
index 188f24c6bd44c1fbd5a73994ab231b514e21f129..93c0352991bc31483c1d3f2174ce586f9ed6781c 100644
--- a/core/modules/jsonapi/src/Normalizer/FieldNormalizer.php
+++ b/core/modules/jsonapi/src/Normalizer/FieldNormalizer.php
@@ -4,9 +4,12 @@
 
 use Drupal\Core\Field\FieldDefinitionInterface;
 use Drupal\Core\Field\FieldItemListInterface;
+use Drupal\Core\Field\FieldStorageDefinitionInterface;
 use Drupal\jsonapi\JsonApiResource\ResourceObject;
 use Drupal\jsonapi\Normalizer\Value\CacheableNormalization;
 use Drupal\jsonapi\ResourceType\ResourceType;
+use Drupal\serialization\Normalizer\SchematicNormalizerTrait;
+use Drupal\serialization\Serializer\JsonSchemaProviderSerializerInterface;
 use Symfony\Component\Serializer\Normalizer\DenormalizerInterface;
 
 /**
@@ -20,10 +23,12 @@
  */
 class FieldNormalizer extends NormalizerBase implements DenormalizerInterface {
 
+  use SchematicNormalizerTrait;
+
   /**
    * {@inheritdoc}
    */
-  public function normalize($field, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
+  public function doNormalize($field, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
     /** @var \Drupal\Core\Field\FieldItemListInterface $field */
     $normalized_items = $this->normalizeFieldItems($field, $format, $context);
     assert($context['resource_object'] instanceof ResourceObject);
@@ -76,7 +81,7 @@ public function denormalize($data, $class, $format = NULL, array $context = []):
    * @param array $context
    *   The context array.
    *
-   * @return \Drupal\jsonapi\Normalizer\FieldItemNormalizer[]
+   * @return \Drupal\jsonapi\Normalizer\Value\CacheableNormalization[]
    *   The array of normalized field items.
    */
   protected function normalizeFieldItems(FieldItemListInterface $field, $format, array $context) {
@@ -89,6 +94,40 @@ protected function normalizeFieldItems(FieldItemListInterface $field, $format, a
     return $normalizer_items;
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  protected function getNormalizationSchema(mixed $object, array $context = []): array {
+    assert($object instanceof FieldItemListInterface);
+    // Some aspects of the schema are determined by the field config.
+    $cardinality = $object->getFieldDefinition()->getFieldStorageDefinition()->getCardinality();
+    $schema = [];
+    // Normalizers are resolved by the class/object being normalized. Even
+    // without data, we must retrieve a representative field item.
+    $field_item = $object->appendItem($object->generateSampleItems());
+    assert($this->serializer instanceof JsonSchemaProviderSerializerInterface);
+    $item_schema = $this->serializer->getJsonSchema($field_item, $context);
+    $object->removeItem(count($object) - 1);
+    unset($field_item);
+
+    $schema = $item_schema;
+    if ($cardinality !== 1) {
+      $schema['type'] = 'array';
+      if ($object->getFieldDefinition()->isRequired()) {
+        $schema['minItems'] = 1;
+      }
+      if ($cardinality !== FieldStorageDefinitionInterface::CARDINALITY_UNLIMITED) {
+        $schema['maxItems'] = $cardinality;
+      }
+      if (!empty($item_schema)) {
+        $schema['items'] = $item_schema;
+      }
+    }
+    return !$object->getFieldDefinition()->isRequired()
+      ? ['oneOf' => [$schema, ['type' => 'null']]]
+      : $schema;
+  }
+
   /**
    * {@inheritdoc}
    */
diff --git a/core/modules/jsonapi/src/Normalizer/JsonApiDocumentTopLevelNormalizer.php b/core/modules/jsonapi/src/Normalizer/JsonApiDocumentTopLevelNormalizer.php
index f9ea4a5f5cf1bfbbd7e10f0be9dc5c2176221c3c..54b36fefe5f0d6f0137fe5aa282f546e28bf3fe6 100644
--- a/core/modules/jsonapi/src/Normalizer/JsonApiDocumentTopLevelNormalizer.php
+++ b/core/modules/jsonapi/src/Normalizer/JsonApiDocumentTopLevelNormalizer.php
@@ -7,13 +7,22 @@
 use Drupal\Component\Uuid\Uuid;
 use Drupal\Core\Cache\CacheableMetadata;
 use Drupal\Core\Entity\EntityTypeManagerInterface;
+use Drupal\Core\Field\FieldStorageDefinitionInterface;
+use Drupal\jsonapi\JsonApiResource\Data;
 use Drupal\jsonapi\JsonApiResource\ErrorCollection;
+use Drupal\jsonapi\JsonApiResource\IncludedData;
+use Drupal\jsonapi\JsonApiResource\NullIncludedData;
 use Drupal\jsonapi\JsonApiResource\OmittedData;
+use Drupal\jsonapi\JsonApiResource\RelationshipData;
+use Drupal\jsonapi\JsonApiResource\ResourceIdentifier;
+use Drupal\jsonapi\JsonApiResource\ResourceObjectData;
 use Drupal\jsonapi\JsonApiSpec;
 use Drupal\jsonapi\Normalizer\Value\CacheableOmission;
 use Drupal\jsonapi\JsonApiResource\JsonApiDocumentTopLevel;
 use Drupal\jsonapi\Normalizer\Value\CacheableNormalization;
 use Drupal\jsonapi\ResourceType\ResourceType;
+use Drupal\serialization\Normalizer\SchematicNormalizerTrait;
+use Drupal\serialization\Serializer\JsonSchemaProviderSerializerInterface;
 use Symfony\Component\HttpKernel\Exception\BadRequestHttpException;
 use Symfony\Component\HttpKernel\Exception\HttpExceptionInterface;
 use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
@@ -35,6 +44,8 @@
  */
 class JsonApiDocumentTopLevelNormalizer extends NormalizerBase implements DenormalizerInterface, NormalizerInterface {
 
+  use SchematicNormalizerTrait;
+
   /**
    * The entity type manager.
    *
@@ -167,7 +178,7 @@ public function denormalize($data, $class, $format = NULL, array $context = []):
   /**
    * {@inheritdoc}
    */
-  public function normalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
+  public function doNormalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
     assert($object instanceof JsonApiDocumentTopLevel);
     $data = $object->getData();
     $document['jsonapi'] = CacheableNormalization::permanent([
@@ -327,6 +338,150 @@ protected static function getLinkHash($salt, $link_href) {
     return substr(str_replace(['-', '_'], '', Crypt::hashBase64($salt . $link_href)), 0, 7);
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  public function getNormalizationSchema(mixed $object, array $context = []): array {
+    // If we are providing a schema based only on an interface, we lack context
+    // to provide anything more than a ref to the JSON:API top-level schema.
+    $fallbackSchema = [
+      '$ref' => JsonApiSpec::SUPPORTED_SPECIFICATION_JSON_SCHEMA,
+    ];
+    if (is_string($object)) {
+      return $fallbackSchema;
+    }
+    assert($object instanceof JsonApiDocumentTopLevel);
+    if ($object->getData() instanceof OmittedData) {
+      // A top-level omitted data object is a bit weird but it will only contain
+      // information in the 'links' property, so we can fall back.
+      return $fallbackSchema;
+    }
+    $schema = [
+      'allOf' => [
+        ['$ref' => JsonApiSpec::SUPPORTED_SPECIFICATION_JSON_SCHEMA],
+      ],
+    ];
+
+    // Top-level JSON:API documents may contain top-level data or an error
+    // collection.
+    $data = $object->getData();
+
+    if ($data instanceof ErrorCollection) {
+      // There's not much else to state here, because errors are a known schema.
+      $schema['required'] = ['errors'];
+    }
+
+    // Relationship data - "resource identifier object(s)"
+    if ($data instanceof RelationshipData) {
+      if ($data->getCardinality() === 1) {
+        $schema['properties']['data'] = [
+          '$ref' => JsonApiSpec::SUPPORTED_SPECIFICATION_JSON_SCHEMA . '#/definitions/relationship',
+        ];
+      }
+      else {
+        $schema['properties']['data'] = [
+          'type' => 'array',
+          'items' => [
+            '$ref' => JsonApiSpec::SUPPORTED_SPECIFICATION_JSON_SCHEMA . '#/definitions/relationship',
+          ],
+          'unevaluatedItems' => FALSE,
+        ];
+        if ($data->getCardinality() !== FieldStorageDefinitionInterface::CARDINALITY_UNLIMITED) {
+          $schema['properties']['data']['maxContains'] = $data->getCardinality();
+        }
+        // We can't do a minContains with the data available in this context.
+      }
+    }
+
+    if ($data instanceof IncludedData) {
+      if ($data instanceof NullIncludedData) {
+        $schema['properties']['included'] = [
+          'type' => 'array',
+          'maxContains' => 0,
+        ];
+      }
+      else {
+        $schema['properties']['included'] = [
+          // 'included' member is always an array.
+          'type' => 'array',
+          'items' => [
+            'oneOf' => $this->getSchemasForDataCollection($data->getData(), $context),
+          ],
+        ];
+      }
+    }
+
+    if ($data instanceof ResourceObjectData) {
+      if ($data->getCardinality() === 1) {
+        assert($data->count() === 1);
+        $schema['properties']['data'] = [
+          'oneOf' => [
+            ...$this->getSchemasForDataCollection($data->getData(), $context),
+            ['type' => 'null'],
+          ],
+        ];
+      }
+      else {
+        $schema['properties']['data'] = [
+          'type' => 'array',
+          'items' => [
+            'oneOf' => $this->getSchemasForDataCollection($data->getData(), $context),
+          ],
+        ];
+        if ($data->getCardinality() !== FieldStorageDefinitionInterface::CARDINALITY_UNLIMITED) {
+          $schema['properties']['data']['maxContains'] = $data->getCardinality();
+        }
+      }
+    }
+
+    return $schema;
+  }
+
+  /**
+   * Retrieve an array of schemas for the resource types in a data object.
+   *
+   * @param \Drupal\jsonapi\JsonApiResource\Data $data
+   *   JSON:API data value objects.
+   * @param array $context
+   *   Normalization context.
+   *
+   * @return array
+   *   Schemas for all types represented in the collection.
+   */
+  protected function getSchemasForDataCollection(Data $data, array $context): array {
+    $schemas = [];
+    if ($data->count() === 0) {
+      return [
+        // We lack sufficient information about if the data would be a
+        // collection or a single resource, so allow either.
+        ['type' => ['array', 'null']],
+      ];
+    }
+    $members = $data->toArray();
+    assert($this->serializer instanceof JsonSchemaProviderSerializerInterface);
+    // Per the spec, data must either be comprised of a single instance or
+    // collection of resource objects OR resource identifiers, but not both.
+    foreach ($members as $member) {
+      $resourceType = $member->getResourceType();
+      if (array_key_exists($resourceType->getTypeName(), $schemas)) {
+        continue;
+      }
+      $schemas[$resourceType->getTypeName()] = $member instanceof ResourceIdentifier
+        ? [
+          'allOf' => [
+            ['$ref' => JsonApiSpec::SUPPORTED_SPECIFICATION_JSON_SCHEMA . '#/definitions/resourceIdentifier'],
+          ],
+          'properties' => [
+            'type' => [
+              'const' => $resourceType->getTypeName(),
+            ],
+          ],
+        ]
+        : $this->serializer->getJsonSchema($member, $context);
+    }
+    return array_values($schemas);
+  }
+
   /**
    * {@inheritdoc}
    */
diff --git a/core/modules/jsonapi/src/Normalizer/RelationshipNormalizer.php b/core/modules/jsonapi/src/Normalizer/RelationshipNormalizer.php
index b406cd4d749ffa2ce20ddd476270fba12f486ff4..126aeaa39c897f91bcb0ae026b9770e35c2401ab 100644
--- a/core/modules/jsonapi/src/Normalizer/RelationshipNormalizer.php
+++ b/core/modules/jsonapi/src/Normalizer/RelationshipNormalizer.php
@@ -2,8 +2,12 @@
 
 namespace Drupal\jsonapi\Normalizer;
 
+use Drupal\Core\Field\Plugin\Field\FieldType\EntityReferenceItemInterface;
 use Drupal\jsonapi\JsonApiResource\Relationship;
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\jsonapi\Normalizer\Value\CacheableNormalization;
+use Drupal\jsonapi\ResourceType\ResourceTypeRepositoryInterface;
+use Drupal\serialization\Normalizer\SchematicNormalizerTrait;
 
 /**
  * Normalizes a JSON:API relationship object.
@@ -12,10 +16,22 @@
  */
 class RelationshipNormalizer extends NormalizerBase {
 
+  use SchematicNormalizerTrait;
+
+  /**
+   * Constructor.
+   *
+   * @param \Drupal\jsonapi\ResourceType\ResourceTypeRepositoryInterface $resourceTypeRepository
+   *   Resource type repository.
+   */
+  public function __construct(
+    protected ResourceTypeRepositoryInterface $resourceTypeRepository,
+  ) {}
+
   /**
    * {@inheritdoc}
    */
-  public function normalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
+  public function doNormalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
     assert($object instanceof Relationship);
     return CacheableNormalization::aggregate([
       'data' => $this->serializer->normalize($object->getData(), $format, $context),
@@ -24,6 +40,41 @@ public function normalize($object, $format = NULL, array $context = []): array|s
     ]);
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  public function getNormalizationSchema(mixed $object, array $context = []): array {
+    assert($object instanceof Relationship);
+    $schema = [
+      'allOf' => [
+        ['$ref' => JsonApiSpec::SUPPORTED_SPECIFICATION_JSON_SCHEMA . '#/definitions/relationship'],
+      ],
+    ];
+    $field_definition = $object->getContext()->getField($object->getFieldName())?->getFieldDefinition();
+    $item_class = $field_definition?->getItemDefinition()->getClass();
+    assert($item_class, sprintf('The context ResourceObject for Relationship being normalized is missing field %s.', $object->getFieldName()));
+    if (!$item_class || !is_subclass_of($item_class, EntityReferenceItemInterface::class)) {
+      return $schema;
+    }
+    $targets = $item_class::getReferenceableBundles($field_definition);
+    $target_types = array_reduce(array_keys($targets), function (array $carry, string $entity_type_id) use ($targets) {
+      foreach ($targets[$entity_type_id] as $bundle) {
+        // Even if a resource is internal, it can be referenced.
+        if ((!$resource = $this->resourceTypeRepository->get($entity_type_id, $bundle)) || in_array($resource->getTypeName(), $carry)) {
+          continue;
+        }
+        $carry[] = $resource->getTypeName();
+      }
+      return $carry;
+    }, []);
+    if ($target_types) {
+      $schema['properties']['type'] = [
+        'oneOf' => array_map(fn(string $resource_type_name) => ['const' => $resource_type_name], $target_types),
+      ];
+    }
+    return $schema;
+  }
+
   /**
    * {@inheritdoc}
    */
diff --git a/core/modules/jsonapi/src/Normalizer/ResourceObjectNormalizer.php b/core/modules/jsonapi/src/Normalizer/ResourceObjectNormalizer.php
index f5623be8fb1892df757cba932e536ba87a006755..7e978071e4ef3ec8649e4f15f9fc98eb9c88b92e 100644
--- a/core/modules/jsonapi/src/Normalizer/ResourceObjectNormalizer.php
+++ b/core/modules/jsonapi/src/Normalizer/ResourceObjectNormalizer.php
@@ -3,6 +3,9 @@
 namespace Drupal\jsonapi\Normalizer;
 
 use Drupal\Core\Cache\CacheableMetadata;
+use Drupal\Core\Entity\EntityFieldManagerInterface;
+use Drupal\Core\Entity\EntityTypeManagerInterface;
+use Drupal\Core\Entity\FieldableEntityInterface;
 use Drupal\Core\Field\EntityReferenceFieldItemListInterface;
 use Drupal\Core\Field\FieldItemListInterface;
 use Drupal\jsonapi\Events\CollectRelationshipMetaEvent;
@@ -10,8 +13,13 @@
 use Drupal\jsonapi\EventSubscriber\ResourceObjectNormalizationCacher;
 use Drupal\jsonapi\JsonApiResource\Relationship;
 use Drupal\jsonapi\JsonApiResource\ResourceObject;
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\jsonapi\Normalizer\Value\CacheableNormalization;
 use Drupal\jsonapi\Normalizer\Value\CacheableOmission;
+use Drupal\jsonapi\ResourceType\ResourceType;
+use Drupal\jsonapi\ResourceType\ResourceTypeField;
+use Drupal\jsonapi\Serializer\Serializer;
+use Drupal\serialization\Normalizer\SchematicNormalizerTrait;
 use Symfony\Component\EventDispatcher\EventDispatcherInterface;
 
 /**
@@ -25,6 +33,8 @@
  */
 class ResourceObjectNormalizer extends NormalizerBase {
 
+  use SchematicNormalizerTrait;
+
   /**
    * The entity normalization cacher.
    *
@@ -37,6 +47,16 @@ class ResourceObjectNormalizer extends NormalizerBase {
    */
   private EventDispatcherInterface $eventDispatcher;
 
+  /**
+   * @var mixed|\Drupal\Core\Entity\EntityFieldManagerInterface|null
+   */
+  private EntityFieldManagerInterface $entityFieldManager;
+
+  /**
+   * @var mixed|\Drupal\Core\Entity\EntityTypeManagerInterface|null
+   */
+  private EntityTypeManagerInterface $entityTypeManager;
+
   /**
    * Constructs a ResourceObjectNormalizer object.
    *
@@ -44,8 +64,12 @@ class ResourceObjectNormalizer extends NormalizerBase {
    *   The entity normalization cacher.
    * @param \Symfony\Component\EventDispatcher\EventDispatcherInterface $event_dispatcher
    *   The event dispatcher.
+   * @param \Drupal\Core\Entity\EntityFieldManagerInterface $entity_field_manager
+   *   The entity field manager.
+   * @param \Drupal\Core\Entity\EntityTypeManagerInterface $entity_type_manager
+   *   The entity type manager.
    */
-  public function __construct(ResourceObjectNormalizationCacher $cacher, ?EventDispatcherInterface $event_dispatcher = NULL) {
+  public function __construct(ResourceObjectNormalizationCacher $cacher, ?EventDispatcherInterface $event_dispatcher = NULL, ?EntityFieldManagerInterface $entity_field_manager = NULL, ?EntityTypeManagerInterface $entity_type_manager = NULL) {
     $this->cacher = $cacher;
 
     if ($event_dispatcher === NULL) {
@@ -53,6 +77,18 @@ public function __construct(ResourceObjectNormalizationCacher $cacher, ?EventDis
       $event_dispatcher = \Drupal::service('event_dispatcher');
     }
     $this->eventDispatcher = $event_dispatcher;
+
+    if ($entity_field_manager === NULL) {
+      @trigger_error('Calling ' . __METHOD__ . '() without the $entity_field_manager argument is deprecated in drupal:11.2.0 and will be required in drupal:12.0.0. See https://www.drupal.org/node/3031367', E_USER_DEPRECATED);
+      $entity_field_manager = \Drupal::service('entity_field_manager');
+    }
+    $this->entityFieldManager = $entity_field_manager;
+
+    if ($entity_type_manager === NULL) {
+      @trigger_error('Calling ' . __METHOD__ . '() without the $entity_type_manager argument is deprecated in drupal:11.2.0 and will be required in drupal:12.0.0. See https://www.drupal.org/node/3031367', E_USER_DEPRECATED);
+      $entity_type_manager = \Drupal::service('entity_type_manager');
+    }
+    $this->entityTypeManager = $entity_type_manager;
   }
 
   /**
@@ -65,7 +101,7 @@ public function supportsDenormalization($data, string $type, ?string $format = N
   /**
    * {@inheritdoc}
    */
-  public function normalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
+  public function doNormalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
     assert($object instanceof ResourceObject);
     // If the fields to use were specified, only output those field values.
     $context['resource_object'] = $object;
@@ -223,6 +259,132 @@ protected function serializeField($field, array $context, $format) {
     }
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  public function getNormalizationSchema(mixed $object, array $context = []): array {
+    if (is_string($object)) {
+      // Without a true object we can only provide a generic schema.
+      return [
+        '$ref' => JsonApiSpec::SUPPORTED_SPECIFICATION_JSON_SCHEMA . '#/definitions/resource',
+      ];
+    }
+    // ResourceObject is usually instantiated from a specific entity, however
+    // a placeholder can be created for purposes of schema generation which
+    // would provide access to the resource type, but not contain any "live"
+    // data.
+    assert($object instanceof ResourceObject);
+
+    $attributes_schema = [];
+    $relationships_schema = [];
+
+    $this->entityTypeManager->getDefinition($object->getResourceType()->getEntityTypeId())->entityClassImplements(FieldableEntityInterface::class)
+      ? $this->processContentEntitySchema($object, $context, $attributes_schema, $relationships_schema)
+      : $this->processConfigEntitySchema($object->getResourceType(), $context, $attributes_schema);
+
+    return [
+      'allOf' => [
+        ['$ref' => JsonApiSpec::SUPPORTED_SPECIFICATION_JSON_SCHEMA . '#/definitions/resourceIdentification'],
+      ],
+      'properties' => [
+        'meta' => [
+          '$ref' => JsonApiSpec::SUPPORTED_SPECIFICATION_JSON_SCHEMA . '#/definitions/meta',
+        ],
+        'links' => [
+          '$ref' => JsonApiSpec::SUPPORTED_SPECIFICATION_JSON_SCHEMA . '#/definitions/resourceLinks',
+        ],
+        // If the array is empty we must return an object so it won't encode as
+        // an array; we don't cast the value here so the returned value is still
+        // traversable as an array when accessed in PHP.
+        'attributes' => $attributes_schema ?: new \ArrayObject(),
+        'relationships' => $relationships_schema ?: new \ArrayObject(),
+      ],
+      'unevaluatedProperties' => FALSE,
+    ];
+  }
+
+  protected function processConfigEntitySchema(ResourceType $resource_type, array $context, array &$attributes_schema): void {
+    // This is largely the same as in ResourceObject but without a real entity.
+    $fields = $resource_type->getFields();
+    // Filter the array based on the field names.
+    $enabled_field_names = array_filter(array_keys($fields), static fn (string $internal_field_name) => $resource_type->isFieldEnabled($internal_field_name));
+    // Return a sub-array of $output containing the keys in $enabled_fields.
+    $input = array_intersect_key($fields, array_flip($enabled_field_names));
+    foreach ($input as $field_name => $field_value) {
+      $attributes_schema['properties'][$resource_type->getPublicName($field_name)] = [
+        'title' => $field_name,
+        // @todo Potentially introspect schema to give more information.
+        // @see https://www.drupal.org/project/drupal/issues/3426508
+        // Right now, this will validate to anything.
+      ];
+    }
+  }
+
+  protected function processContentEntitySchema(ResourceObject $resource_object, array $context, array &$attributes_schema, &$relationships_schema): void {
+    // Actual normalization supports sparse fieldsets, however we provide schema
+    // for all possible fields that may be retrieved.
+    $resource_type = $resource_object->getResourceType();
+    $field_definitions = $this->entityFieldManager
+      ->getFieldDefinitions($resource_type->getEntityTypeId(), $resource_type->getBundle());
+
+    $resource_fields = $resource_type->getFields();
+    // User resource objects contain a read-only attribute that is not a real
+    // field on the user entity type.
+    // @see \Drupal\jsonapi\JsonApiResource\ResourceObject::extractContentEntityFields()
+    // @todo Eliminate this special casing in https://www.drupal.org/project/drupal/issues/3079254.
+    if ($resource_type->getEntityTypeId() === 'user') {
+      $resource_fields = array_diff_key($resource_fields, array_flip([$resource_type->getPublicName('display_name')]));
+    }
+
+    /** @var \Drupal\Core\Field\FieldDefinitionInterface[] $fields */
+    $fields = array_reduce(
+      $resource_fields,
+      function (array $carry, ResourceTypeField $resource_field) use ($field_definitions) {
+        if (!$resource_field->isFieldEnabled()) {
+          return $carry;
+        }
+        $carry[$resource_field->getPublicName()] = $field_definitions[$resource_field->getInternalName()];
+        return $carry;
+      },
+      []
+    );
+    assert($this->serializer instanceof Serializer);
+    $relationship_field_names = array_keys($resource_type->getRelatableResourceTypes());
+
+    $create_values = [];
+    if ($bundle_key = $this->entityTypeManager->getDefinition($resource_type->getEntityTypeId())->getKey('bundle')) {
+      $create_values = [$bundle_key => $resource_type->getBundle()];
+    }
+    $stub_entity = $this->entityTypeManager
+      ->getStorage($resource_type->getEntityTypeId())->create($create_values);
+    foreach ($fields as $field_name => $field) {
+      $stub_field = $stub_entity->get($field->getName());
+      if ($stub_field instanceof EntityReferenceFieldItemListInterface) {
+        // Build the relationship object based on the entity reference and
+        // retrieve normalizer for that object instead.
+        // @see ::serializeField()
+        $relationship = Relationship::createFromEntityReferenceField($resource_object, $stub_field);
+        $schema = $this->serializer->getJsonSchema($relationship, $context);
+      }
+      else {
+        $schema = $this->serializer->getJsonSchema($stub_field, $context);
+      }
+      // Fallback basic annotations.
+      if (empty($schema['title']) && $title = $field->getLabel()) {
+        $schema['title'] = (string) $title;
+      }
+      if (empty($schema['description']) && $description = $field->getFieldStorageDefinition()->getDescription()) {
+        $schema['description'] = (string) $description;
+      }
+      if (in_array($field_name, $relationship_field_names, TRUE)) {
+        $relationships_schema['properties'][$field_name] = $schema;
+      }
+      else {
+        $attributes_schema['properties'][$field_name] = $schema;
+      }
+    }
+  }
+
   /**
    * {@inheritdoc}
    */
diff --git a/core/modules/jsonapi/src/Serializer/Serializer.php b/core/modules/jsonapi/src/Serializer/Serializer.php
index b622f5315e88ec6926074ed34d9de58e3d4327f1..93669b50402fd713b8ef806ad9a9b62a09bb37ae 100644
--- a/core/modules/jsonapi/src/Serializer/Serializer.php
+++ b/core/modules/jsonapi/src/Serializer/Serializer.php
@@ -2,6 +2,8 @@
 
 namespace Drupal\jsonapi\Serializer;
 
+use Drupal\serialization\Serializer\JsonSchemaProviderSerializerInterface;
+use Drupal\serialization\Serializer\JsonSchemaProviderSerializerTrait;
 use Symfony\Component\Serializer\Normalizer\NormalizerInterface;
 use Symfony\Component\Serializer\Serializer as SymfonySerializer;
 
@@ -20,7 +22,9 @@
  * @see https://www.drupal.org/project/drupal/issues/3032787
  * @see jsonapi.api.php
  */
-final class Serializer extends SymfonySerializer {
+final class Serializer extends SymfonySerializer implements JsonSchemaProviderSerializerInterface {
+
+  use JsonSchemaProviderSerializerTrait;
 
   /**
    * A normalizer to fall back on when JSON:API cannot normalize an object.
diff --git a/core/modules/jsonapi/tests/src/Functional/ActionTest.php b/core/modules/jsonapi/tests/src/Functional/ActionTest.php
index f2f293ad6c337ed241a2afe998dfac8fafd54142..8a961624d9c97ba5d9e2131b9e0645ce571d6364 100644
--- a/core/modules/jsonapi/tests/src/Functional/ActionTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/ActionTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Url;
 use Drupal\system\Entity\Action;
 use Drupal\user\RoleInterface;
@@ -76,10 +77,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/BaseFieldOverrideTest.php b/core/modules/jsonapi/tests/src/Functional/BaseFieldOverrideTest.php
index c26bb17a3f1d5e6c016a99d04ead28ee37e9b232..d2ee111c28c107c95728f8417620eca35b4d5bd8 100644
--- a/core/modules/jsonapi/tests/src/Functional/BaseFieldOverrideTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/BaseFieldOverrideTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Field\Entity\BaseFieldOverride;
 use Drupal\Core\Url;
 use Drupal\node\Entity\NodeType;
@@ -79,10 +80,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/BlockContentTest.php b/core/modules/jsonapi/tests/src/Functional/BlockContentTest.php
index 3678644af4726181103263e9be10a64dc14d44e1..c30aaa0057c24b31fbf181d1ddb12de95f570d4c 100644
--- a/core/modules/jsonapi/tests/src/Functional/BlockContentTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/BlockContentTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\block_content\Entity\BlockContent;
 use Drupal\block_content\Entity\BlockContentType;
 use Drupal\Core\Cache\Cache;
@@ -140,10 +141,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $base_url->toString()],
diff --git a/core/modules/jsonapi/tests/src/Functional/BlockContentTypeTest.php b/core/modules/jsonapi/tests/src/Functional/BlockContentTypeTest.php
index 4fb58c343f46ca2c6732966bd4a821ae2a5badfa..43986170104f089abb7cf3a577e0e175dfd231a1 100644
--- a/core/modules/jsonapi/tests/src/Functional/BlockContentTypeTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/BlockContentTypeTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\block_content\Entity\BlockContentType;
 use Drupal\Core\Url;
 
@@ -73,10 +74,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/BlockTest.php b/core/modules/jsonapi/tests/src/Functional/BlockTest.php
index 1543feb48edbadde58d61c8cfcfcfe4dd7b1bef4..b16c830afe4bb5a3ee145ece93aab7232332fb10 100644
--- a/core/modules/jsonapi/tests/src/Functional/BlockTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/BlockTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\block\Entity\Block;
 use Drupal\Core\Session\AccountInterface;
 use Drupal\Core\Url;
@@ -93,10 +94,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/CommentTest.php b/core/modules/jsonapi/tests/src/Functional/CommentTest.php
index 82c97d9f1f328062c62cf2d6fa9d9363e8dde91d..6396330801cd3d7ad0a5594bac4f05dad17e539f 100644
--- a/core/modules/jsonapi/tests/src/Functional/CommentTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/CommentTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\comment\Entity\Comment;
 use Drupal\comment\Entity\CommentType;
 use Drupal\comment\Plugin\Field\FieldType\CommentItemInterface;
@@ -150,10 +151,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/CommentTypeTest.php b/core/modules/jsonapi/tests/src/Functional/CommentTypeTest.php
index 4c63521082e5e6b005b3ea70152a2386d8cda1ef..bd80c09b8424f0761dbebc2af2cb8e4e6596b36c 100644
--- a/core/modules/jsonapi/tests/src/Functional/CommentTypeTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/CommentTypeTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\comment\Entity\CommentType;
 use Drupal\Core\Url;
 
@@ -74,10 +75,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/ConfigTestTest.php b/core/modules/jsonapi/tests/src/Functional/ConfigTestTest.php
index e6fcf2230c17afc7a106ab120c1a8b0ebf48ad9d..c65b703ca4da6f2f33f27d3755c652ae13f460de 100644
--- a/core/modules/jsonapi/tests/src/Functional/ConfigTestTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/ConfigTestTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\config_test\Entity\ConfigTest;
 use Drupal\Core\Url;
 
@@ -83,10 +84,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/ConfigurableLanguageTest.php b/core/modules/jsonapi/tests/src/Functional/ConfigurableLanguageTest.php
index ca11e4dfdfe790c42bac5c6cf59281b6bc706edc..94e14919227c7787df7620ef68b820a6d08ca547 100644
--- a/core/modules/jsonapi/tests/src/Functional/ConfigurableLanguageTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/ConfigurableLanguageTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Component\Utility\NestedArray;
 use Drupal\Core\Cache\Cache;
 use Drupal\Core\Url;
@@ -73,10 +74,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/ContactFormTest.php b/core/modules/jsonapi/tests/src/Functional/ContactFormTest.php
index 6601401603ff05107df0e8923ad0ee34926dc7fa..6426ef195fe1a291ba4f36148558291534168418 100644
--- a/core/modules/jsonapi/tests/src/Functional/ContactFormTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/ContactFormTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\contact\Entity\ContactForm;
 use Drupal\Core\Url;
 
@@ -76,10 +77,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/ContentLanguageSettingsTest.php b/core/modules/jsonapi/tests/src/Functional/ContentLanguageSettingsTest.php
index a3446d7257d06de46eecc45cbf1a1aaa04411f30..d34186faa3c93adab3436883c7c4843e343d8fa9 100644
--- a/core/modules/jsonapi/tests/src/Functional/ContentLanguageSettingsTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/ContentLanguageSettingsTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Cache\Cache;
 use Drupal\Core\Session\AccountInterface;
 use Drupal\Core\Url;
@@ -81,10 +82,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/DateFormatTest.php b/core/modules/jsonapi/tests/src/Functional/DateFormatTest.php
index 52575b69150a000a7be74582f7e215a3bdb1a107..084a2e9916210a3e3caf330f6d5c370f65f3d251 100644
--- a/core/modules/jsonapi/tests/src/Functional/DateFormatTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/DateFormatTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Datetime\Entity\DateFormat;
 use Drupal\Core\Url;
 
@@ -78,10 +79,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/EditorTest.php b/core/modules/jsonapi/tests/src/Functional/EditorTest.php
index 0cdd52931f75d687463d4dcf6d8b47c744205c2b..09028368af9ca3714dbeff4a791cc2bab6775b2a 100644
--- a/core/modules/jsonapi/tests/src/Functional/EditorTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/EditorTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\ckeditor5\Plugin\CKEditor5Plugin\Heading;
 use Drupal\Core\Entity\EntityInterface;
 use Drupal\Core\Session\AccountInterface;
@@ -106,10 +107,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/EntityFormDisplayTest.php b/core/modules/jsonapi/tests/src/Functional/EntityFormDisplayTest.php
index 45127dba6538d58c1b170bbe7ce7d562a065259a..24a34c6fa2ddff1498e2ec732d4f56974e298a26 100644
--- a/core/modules/jsonapi/tests/src/Functional/EntityFormDisplayTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/EntityFormDisplayTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Entity\Entity\EntityFormDisplay;
 use Drupal\Core\Url;
 use Drupal\node\Entity\NodeType;
@@ -80,10 +81,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/EntityFormModeTest.php b/core/modules/jsonapi/tests/src/Functional/EntityFormModeTest.php
index 46df804ba6f1277a447305c060250496b901a72e..cadf6f6f271d9877411f5384dc2f9e0e0e3fcade 100644
--- a/core/modules/jsonapi/tests/src/Functional/EntityFormModeTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/EntityFormModeTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Entity\Entity\EntityFormMode;
 use Drupal\Core\Url;
 
@@ -73,10 +74,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/EntityTestComputedFieldTest.php b/core/modules/jsonapi/tests/src/Functional/EntityTestComputedFieldTest.php
index 2163c58d365d9a30366df295eb75269d38070b98..14d00fcbef6c1d332b1b448288fb2b6257e85231 100644
--- a/core/modules/jsonapi/tests/src/Functional/EntityTestComputedFieldTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/EntityTestComputedFieldTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Cache\Cache;
 use Drupal\Core\Url;
 use Drupal\entity_test\Entity\EntityTestComputedField;
@@ -95,10 +96,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/EntityTestMapFieldTest.php b/core/modules/jsonapi/tests/src/Functional/EntityTestMapFieldTest.php
index c710fb0c3fb11b2e977b8ad5c594b8323a9de58c..f8092f7aeea062ca879d937fc5b735830ec0d63d 100644
--- a/core/modules/jsonapi/tests/src/Functional/EntityTestMapFieldTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/EntityTestMapFieldTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Url;
 use Drupal\entity_test\Entity\EntityTestMapField;
 use Drupal\user\Entity\User;
@@ -96,10 +97,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/EntityTestTest.php b/core/modules/jsonapi/tests/src/Functional/EntityTestTest.php
index c468322f44a124612a90a8188929482ab928eadc..b176f41b6933d815456eaf58e01563e6c173c114 100644
--- a/core/modules/jsonapi/tests/src/Functional/EntityTestTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/EntityTestTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Field\BaseFieldDefinition;
 use Drupal\Core\Session\AccountInterface;
 use Drupal\Core\Url;
@@ -107,10 +108,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/EntityViewDisplayTest.php b/core/modules/jsonapi/tests/src/Functional/EntityViewDisplayTest.php
index 628232d7d59030211c6ca5f99db55e1c5c3d8f88..d344634c191f407b13147c48cff1fda6762bacda 100644
--- a/core/modules/jsonapi/tests/src/Functional/EntityViewDisplayTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/EntityViewDisplayTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Entity\Entity\EntityViewDisplay;
 use Drupal\Core\Url;
 use Drupal\node\Entity\NodeType;
@@ -81,10 +82,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/EntityViewModeTest.php b/core/modules/jsonapi/tests/src/Functional/EntityViewModeTest.php
index c85c4f55888dc09f53bc2ec259ec0056b5912592..353109cf8c8f08c2d6af6ddc143174832c41e2a7 100644
--- a/core/modules/jsonapi/tests/src/Functional/EntityViewModeTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/EntityViewModeTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Entity\Entity\EntityViewMode;
 use Drupal\Core\Url;
 
@@ -73,10 +74,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/FieldConfigTest.php b/core/modules/jsonapi/tests/src/Functional/FieldConfigTest.php
index d1bdc3d98b00598be4da0acc71054a22f30f197d..6d48f81ed965feb9fbea194c6b3774441e8fc140 100644
--- a/core/modules/jsonapi/tests/src/Functional/FieldConfigTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/FieldConfigTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Entity\EntityInterface;
 use Drupal\Core\Session\AccountInterface;
 use Drupal\Core\Url;
@@ -87,10 +88,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/FieldStorageConfigTest.php b/core/modules/jsonapi/tests/src/Functional/FieldStorageConfigTest.php
index dfe693e5aa279d1db78f0fff9209b0dbb5afe8b6..84b110fcbb4664580406a942bb36d0451dc2128e 100644
--- a/core/modules/jsonapi/tests/src/Functional/FieldStorageConfigTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/FieldStorageConfigTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Url;
 use Drupal\field\Entity\FieldStorageConfig;
 
@@ -70,10 +71,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/FileTest.php b/core/modules/jsonapi/tests/src/Functional/FileTest.php
index 1c89a7bcffcd0cccd24216ee6bbe1e891eb65a1f..a36b91602844e8a16409d500838b67a7589bad2a 100644
--- a/core/modules/jsonapi/tests/src/Functional/FileTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/FileTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Component\Utility\NestedArray;
 use Drupal\Core\Url;
 use Drupal\file\Entity\File;
@@ -138,10 +139,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/FileUploadTest.php b/core/modules/jsonapi/tests/src/Functional/FileUploadTest.php
index 1a4b0a2d151072533a458a873497659e885bee55..bd6efb306d3f08e54e90bc9d885aaf332796ce0c 100644
--- a/core/modules/jsonapi/tests/src/Functional/FileUploadTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/FileUploadTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Component\Render\PlainTextOutput;
 use Drupal\Component\Serialization\Json;
 use Drupal\Component\Utility\NestedArray;
@@ -299,10 +300,10 @@ public function testPostFileUploadAndUseInSingleRequest(): void {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => Url::fromUri('base:/jsonapi/entity_test/entity_test/' . $this->entity->uuid() . '/field_rest_file_test')->setAbsolute(TRUE)->toString()],
@@ -760,10 +761,10 @@ protected function getExpectedDocument($fid = 1, $expected_filename = 'example.t
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/FilterFormatTest.php b/core/modules/jsonapi/tests/src/Functional/FilterFormatTest.php
index 174a1bb4b20eaecc352b887ce067470a4c9980ef..4321e4424b9be7887d18dfc2e2b42c8f045507a3 100644
--- a/core/modules/jsonapi/tests/src/Functional/FilterFormatTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/FilterFormatTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Url;
 use Drupal\filter\Entity\FilterFormat;
 
@@ -78,10 +79,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/ImageStyleTest.php b/core/modules/jsonapi/tests/src/Functional/ImageStyleTest.php
index 23629ffa40b2ed69485fe43d4dcc819d3c5d1f52..8476c9e18ad3b2183d69dba7c881d08ba0f36a9f 100644
--- a/core/modules/jsonapi/tests/src/Functional/ImageStyleTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/ImageStyleTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Url;
 use Drupal\image\Entity\ImageStyle;
 
@@ -90,10 +91,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/MediaTest.php b/core/modules/jsonapi/tests/src/Functional/MediaTest.php
index 7b98a0ad36f835d8c9133e39bb54a5cda8afc9dc..da5e7ba55d7e2c356d507d776743a1694ea5d71e 100644
--- a/core/modules/jsonapi/tests/src/Functional/MediaTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/MediaTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Entity\EntityInterface;
 use Drupal\Core\Url;
 use Drupal\file\Entity\File;
@@ -165,10 +166,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $base_url->toString()],
diff --git a/core/modules/jsonapi/tests/src/Functional/MediaTypeTest.php b/core/modules/jsonapi/tests/src/Functional/MediaTypeTest.php
index 6cfbc20edec0f2b7cab5b1f641e9e3421dccbb29..a0957561112b516ff0e9c85f937bf58bf1e53ffb 100644
--- a/core/modules/jsonapi/tests/src/Functional/MediaTypeTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/MediaTypeTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Url;
 use Drupal\media\Entity\MediaType;
 
@@ -74,10 +75,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/MenuLinkContentTest.php b/core/modules/jsonapi/tests/src/Functional/MenuLinkContentTest.php
index ed41e22b6c9f49f4db2becdaff0f758d14a4eb0a..2dc19ac03c012546a06fc424d82e2ac3174be37c 100644
--- a/core/modules/jsonapi/tests/src/Functional/MenuLinkContentTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/MenuLinkContentTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Component\Serialization\Json;
 use Drupal\Component\Utility\NestedArray;
 use Drupal\Core\Url;
@@ -96,10 +97,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $base_url->toString()],
diff --git a/core/modules/jsonapi/tests/src/Functional/MenuTest.php b/core/modules/jsonapi/tests/src/Functional/MenuTest.php
index ad0836bcb68e02223c357ea44106b352dd582d5f..94c984689dbaadf8a3ffcc4cb6461aaf3479fd88 100644
--- a/core/modules/jsonapi/tests/src/Functional/MenuTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/MenuTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Url;
 use Drupal\system\Entity\Menu;
 
@@ -76,10 +77,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/NodeTest.php b/core/modules/jsonapi/tests/src/Functional/NodeTest.php
index f10b97f0cbc5750552e3b23dbc6a9a8e0b15f170..aa40c2446c5c8f55890d9ab56923839cbc55d9d7 100644
--- a/core/modules/jsonapi/tests/src/Functional/NodeTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/NodeTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Component\Serialization\Json;
 use Drupal\Component\Utility\NestedArray;
 use Drupal\Core\Cache\Cache;
@@ -153,10 +154,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $base_url->toString()],
diff --git a/core/modules/jsonapi/tests/src/Functional/NodeTypeTest.php b/core/modules/jsonapi/tests/src/Functional/NodeTypeTest.php
index da15111ab95c5d5ee28c17655f917676162d4dd9..303879625c369b83a2cec402e2fc80700643610f 100644
--- a/core/modules/jsonapi/tests/src/Functional/NodeTypeTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/NodeTypeTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Url;
 use Drupal\node\Entity\NodeType;
 
@@ -73,10 +74,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/PathAliasTest.php b/core/modules/jsonapi/tests/src/Functional/PathAliasTest.php
index 6618515e3e59eb7e7852a05083bd3bfdb4fa30d6..05bee497fc0357add90a8dbbe4d62ab936143c49 100644
--- a/core/modules/jsonapi/tests/src/Functional/PathAliasTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/PathAliasTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\path_alias\Entity\PathAlias;
 use Drupal\Core\Url;
 
@@ -84,10 +85,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $base_url->toString()],
diff --git a/core/modules/jsonapi/tests/src/Functional/ResourceResponseTestTrait.php b/core/modules/jsonapi/tests/src/Functional/ResourceResponseTestTrait.php
index 07b63ded251d66a925813c93db1626c641a6f382..a21c3b8959466d59d966c828cfbf286410904c76 100644
--- a/core/modules/jsonapi/tests/src/Functional/ResourceResponseTestTrait.php
+++ b/core/modules/jsonapi/tests/src/Functional/ResourceResponseTestTrait.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Component\Serialization\Json;
 use Drupal\Component\Utility\Crypt;
 use Drupal\Core\Access\AccessResultInterface;
@@ -79,10 +80,10 @@ protected static function toCollectionResourceResponse(array $responses, $self_l
     $merged_document['jsonapi'] = [
       'meta' => [
         'links' => [
-          'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+          'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
         ],
       ],
-      'version' => '1.0',
+      'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
     ];
     // Until we can reasonably know what caused an error, we shouldn't include
     // 'self' links in error documents. For example, a 404 shouldn't have a
diff --git a/core/modules/jsonapi/tests/src/Functional/ResourceTestBase.php b/core/modules/jsonapi/tests/src/Functional/ResourceTestBase.php
index 09e360438d119be81cc0801802ba65147e919555..9dd535bedd5e76bf3e7cbff5ad665b3e5506e7a6 100644
--- a/core/modules/jsonapi/tests/src/Functional/ResourceTestBase.php
+++ b/core/modules/jsonapi/tests/src/Functional/ResourceTestBase.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Component\Serialization\Json;
 use Drupal\Component\Utility\NestedArray;
 use Drupal\Component\Utility\Random;
@@ -171,9 +172,9 @@ abstract class ResourceTestBase extends BrowserTestBase {
    * @var array
    */
   protected static $jsonApiMember = [
-    'version' => '1.0',
+    'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
     'meta' => [
-      'links' => ['self' => ['href' => 'http://jsonapi.org/format/1.0/']],
+      'links' => ['self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK]],
     ],
   ];
 
diff --git a/core/modules/jsonapi/tests/src/Functional/ResponsiveImageStyleTest.php b/core/modules/jsonapi/tests/src/Functional/ResponsiveImageStyleTest.php
index db68b40dd818ce2dce396ca8adbeb6311ddc4233..6a96266ebcc6f6593d35d1dec2de0305d5b00969 100644
--- a/core/modules/jsonapi/tests/src/Functional/ResponsiveImageStyleTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/ResponsiveImageStyleTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Url;
 use Drupal\responsive_image\Entity\ResponsiveImageStyle;
 
@@ -87,10 +88,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/RestResourceConfigTest.php b/core/modules/jsonapi/tests/src/Functional/RestResourceConfigTest.php
index 0a2e60201f46622a2eab4e1d86971f3d4e4586d1..48ac1cbcf7e1dd9b3660d68c1fae6fd0eed9a985 100644
--- a/core/modules/jsonapi/tests/src/Functional/RestResourceConfigTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/RestResourceConfigTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Url;
 use Drupal\rest\Entity\RestResourceConfig;
 
@@ -81,10 +82,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/RoleTest.php b/core/modules/jsonapi/tests/src/Functional/RoleTest.php
index f43a2486e75037b5dcdcb8749492f138862d9674..49a1ab219e59c371d4184d80d87288413b206b9e 100644
--- a/core/modules/jsonapi/tests/src/Functional/RoleTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/RoleTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Url;
 use Drupal\user\Entity\Role;
 
@@ -70,10 +71,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/SearchPageTest.php b/core/modules/jsonapi/tests/src/Functional/SearchPageTest.php
index a6975401864c1a0a6f756ec4c5bc7ab38f4009d3..0808250f4196f790ac231e61590a63d5a484481f 100644
--- a/core/modules/jsonapi/tests/src/Functional/SearchPageTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/SearchPageTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Url;
 use Drupal\search\Entity\SearchPage;
 
@@ -83,10 +84,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/ShortcutSetTest.php b/core/modules/jsonapi/tests/src/Functional/ShortcutSetTest.php
index 7172b8ce65e98bc32fee058b3e46ece4dc395507..5173641b8fcae693371d162e12e47eee852764af 100644
--- a/core/modules/jsonapi/tests/src/Functional/ShortcutSetTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/ShortcutSetTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Url;
 use Drupal\shortcut\Entity\ShortcutSet;
 
@@ -95,10 +96,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/ShortcutTest.php b/core/modules/jsonapi/tests/src/Functional/ShortcutTest.php
index d7b111043188f1bd176e02b8f8f1a9691d14e4d4..9be8ff2fc3b78133a5009231a3919d303bbcc069 100644
--- a/core/modules/jsonapi/tests/src/Functional/ShortcutTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/ShortcutTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Component\Utility\NestedArray;
 use Drupal\Core\Session\AccountInterface;
 use Drupal\Core\Url;
@@ -86,10 +87,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/TermTest.php b/core/modules/jsonapi/tests/src/Functional/TermTest.php
index c76de970eb0c608f0990c3036433d4b7255610a6..030e5fc697c1d47006c54ca3df51cba7e03feac6 100644
--- a/core/modules/jsonapi/tests/src/Functional/TermTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/TermTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Component\Serialization\Json;
 use Drupal\Component\Utility\NestedArray;
 use Drupal\Core\Cache\Cache;
@@ -249,10 +250,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $base_url->toString()],
diff --git a/core/modules/jsonapi/tests/src/Functional/UserTest.php b/core/modules/jsonapi/tests/src/Functional/UserTest.php
index bbedfecc140ac8195f1c52f94acd006828aa1500..20076123569eec558efe8df86cedd985d607dfbb 100644
--- a/core/modules/jsonapi/tests/src/Functional/UserTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/UserTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Component\Serialization\Json;
 use Drupal\Component\Utility\NestedArray;
 use Drupal\Core\Cache\Cache;
@@ -142,10 +143,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/ViewTest.php b/core/modules/jsonapi/tests/src/Functional/ViewTest.php
index 828ce5da3df7bdfb8df6e8f88df0870cf14a9d0e..730ed4b9559a75cbd677adf22213c4750d352bbb 100644
--- a/core/modules/jsonapi/tests/src/Functional/ViewTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/ViewTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Url;
 use Drupal\views\Entity\View;
 
@@ -69,10 +70,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/VocabularyTest.php b/core/modules/jsonapi/tests/src/Functional/VocabularyTest.php
index 73e516f083081f6f268ec45850cc5d70268d686f..f455dea709d1572270ff5afa45920aba5664b9fe 100644
--- a/core/modules/jsonapi/tests/src/Functional/VocabularyTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/VocabularyTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Url;
 use Drupal\taxonomy\Entity\Vocabulary;
 
@@ -70,10 +71,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/WorkflowTest.php b/core/modules/jsonapi/tests/src/Functional/WorkflowTest.php
index dc3051a67f33e411cf536710661f335c36e68f05..c9eeafba18707e27e0e5434a03216ed1b23b2c75 100644
--- a/core/modules/jsonapi/tests/src/Functional/WorkflowTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/WorkflowTest.php
@@ -4,6 +4,7 @@
 
 namespace Drupal\Tests\jsonapi\Functional;
 
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\Core\Url;
 use Drupal\workflows\Entity\Workflow;
 
@@ -78,10 +79,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $self_url],
diff --git a/core/modules/jsonapi/tests/src/Functional/WorkspaceTest.php b/core/modules/jsonapi/tests/src/Functional/WorkspaceTest.php
index 80162d9a9b9216a1f670f24cca469dc8df25c3d3..97eec557d22d9eaa197c2823bc5fc8e1e9fe695a 100644
--- a/core/modules/jsonapi/tests/src/Functional/WorkspaceTest.php
+++ b/core/modules/jsonapi/tests/src/Functional/WorkspaceTest.php
@@ -7,6 +7,7 @@
 use Drupal\Core\Cache\CacheableMetadata;
 use Drupal\Core\Entity\EntityInterface;
 use Drupal\Core\Url;
+use Drupal\jsonapi\JsonApiSpec;
 use Drupal\user\Entity\User;
 use Drupal\workspaces\Entity\Workspace;
 
@@ -122,10 +123,10 @@ protected function getExpectedDocument(): array {
       'jsonapi' => [
         'meta' => [
           'links' => [
-            'self' => ['href' => 'http://jsonapi.org/format/1.0/'],
+            'self' => ['href' => JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK],
           ],
         ],
-        'version' => '1.0',
+        'version' => JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION,
       ],
       'links' => [
         'self' => ['href' => $base_url->toString()],
diff --git a/core/modules/jsonapi/tests/src/Kernel/JsonapiKernelTestBase.php b/core/modules/jsonapi/tests/src/Kernel/JsonapiKernelTestBase.php
index cbc472892f79750a14b34625fd850511b53fb3fc..7e1f5e29655a575cd8655b7c5cdb375c2e9a65e7 100644
--- a/core/modules/jsonapi/tests/src/Kernel/JsonapiKernelTestBase.php
+++ b/core/modules/jsonapi/tests/src/Kernel/JsonapiKernelTestBase.php
@@ -24,7 +24,7 @@ abstract class JsonapiKernelTestBase extends KernelTestBase {
   protected static $modules = ['jsonapi', 'file'];
 
   /**
-   * Creates a field of an entity reference field storage on the bundle.
+   * Creates a field of a text field storage on the bundle.
    *
    * @param string $entity_type
    *   The type of entity the field will be attached to.
diff --git a/core/modules/jsonapi/tests/src/Kernel/Normalizer/JsonApiDocumentTopLevelNormalizerTest.php b/core/modules/jsonapi/tests/src/Kernel/Normalizer/JsonApiTopLevelResourceNormalizerTest.php
similarity index 82%
rename from core/modules/jsonapi/tests/src/Kernel/Normalizer/JsonApiDocumentTopLevelNormalizerTest.php
rename to core/modules/jsonapi/tests/src/Kernel/Normalizer/JsonApiTopLevelResourceNormalizerTest.php
index d1eeef1d25c4b361119930eafe70ec094ea80982..ced2507e1384a8ff9d788dd824c58ee75510fa40 100644
--- a/core/modules/jsonapi/tests/src/Kernel/Normalizer/JsonApiDocumentTopLevelNormalizerTest.php
+++ b/core/modules/jsonapi/tests/src/Kernel/Normalizer/JsonApiTopLevelResourceNormalizerTest.php
@@ -17,12 +17,16 @@
 use Drupal\jsonapi\JsonApiResource\ResourceObject;
 use Drupal\jsonapi\JsonApiResource\JsonApiDocumentTopLevel;
 use Drupal\jsonapi\JsonApiResource\ResourceObjectData;
+use Drupal\jsonapi\JsonApiSpec;
+use Drupal\jsonapi\Normalizer\JsonApiDocumentTopLevelNormalizer;
 use Drupal\node\Entity\Node;
 use Drupal\node\Entity\NodeType;
+use Drupal\system\Entity\Action;
 use Drupal\taxonomy\Entity\Term;
 use Drupal\taxonomy\Entity\Vocabulary;
 use Drupal\Tests\image\Kernel\ImageFieldCreationTrait;
 use Drupal\Tests\jsonapi\Kernel\JsonapiKernelTestBase;
+use Drupal\Tests\jsonapi\Traits\JsonApiJsonSchemaTestTrait;
 use Drupal\user\Entity\Role;
 use Drupal\user\Entity\User;
 use Drupal\user\RoleInterface;
@@ -35,9 +39,10 @@
  *
  * @internal
  */
-class JsonApiDocumentTopLevelNormalizerTest extends JsonapiKernelTestBase {
+class JsonApiTopLevelResourceNormalizerTest extends JsonapiKernelTestBase {
 
   use ImageFieldCreationTrait;
+  use JsonApiJsonSchemaTestTrait;
 
   /**
    * {@inheritdoc}
@@ -258,14 +263,56 @@ protected function tearDown(): void {
   }
 
   /**
-   * @covers ::normalize
+   * Get a test resource type, resource object and includes.
+   *
+   * @return array
+   *   Indexed array with values:
+   *     - Resource type.
+   *     - Resource object.
+   *     - Includes.
    */
-  public function testNormalize(): void {
-    $resource_type = $this->container->get('jsonapi.resource_type.repository')->get('node', 'article');
+  protected function getTestContentEntityResource(): array {
+    $resource_type = $this->container->get('jsonapi.resource_type.repository')
+      ->get('node', 'article');
 
     $resource_object = ResourceObject::createFromEntity($resource_type, $this->node);
     $includes = $this->includeResolver->resolve($resource_object, 'uid,field_tags,field_image');
+    return [$resource_type, $resource_object, $includes];
+  }
+
+  /**
+   * Get a test resource type, resource object and includes for config entity.
+   *
+   * @return array
+   *   Indexed array with values:
+   *     - Resource type.
+   *     - Resource object.
+   *     - Includes.
+   */
+  protected function getTestConfigEntityResource(): array {
+    $resource_type = $this->container->get('jsonapi.resource_type.repository')
+      ->get('action', 'action');
+
+    $resource_object = ResourceObject::createFromEntity(
+      $resource_type,
+      Action::create([
+        'id' => 'user_add_role_action.' . RoleInterface::ANONYMOUS_ID,
+        'type' => 'user',
+        'label' => 'Add the anonymous role to the selected users',
+        'configuration' => [
+          'rid' => RoleInterface::ANONYMOUS_ID,
+        ],
+        'plugin' => 'user_add_role_action',
+      ])
+    );
+    return [$resource_type, $resource_object, new NullIncludedData()];
+  }
 
+  /**
+   * @covers ::normalize
+   */
+  public function testNormalize(): void {
+    [$resource_type, $resource_object, $includes] = $this->getTestContentEntityResource();
     $jsonapi_doc_object = $this
       ->getNormalizer()
       ->normalize(
@@ -296,8 +343,8 @@ public function testNormalize(): void {
     $normalized = $jsonapi_doc_object->getNormalization();
 
     // @see http://jsonapi.org/format/#document-jsonapi-object
-    $this->assertEquals('1.0', $normalized['jsonapi']['version']);
-    $this->assertEquals('http://jsonapi.org/format/1.0/', $normalized['jsonapi']['meta']['links']['self']['href']);
+    $this->assertEquals(JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION, $normalized['jsonapi']['version']);
+    $this->assertEquals(JsonApiSpec::SUPPORTED_SPECIFICATION_PERMALINK, $normalized['jsonapi']['meta']['links']['self']['href']);
 
     $this->assertSame($normalized['data']['attributes']['title'], 'dummy_title');
     $this->assertEquals($normalized['data']['id'], $this->node->uuid());
@@ -777,7 +824,7 @@ public function testCacheableMetadata(CacheableMetadata $expected_metadata): voi
   /**
    * Provides test cases for asserting cacheable metadata behavior.
    */
-  public static function testCacheableMetadataProvider() {
+  public static function testCacheableMetadataProvider(): array {
     $cacheable_metadata = function ($metadata) {
       return CacheableMetadata::createFromRenderArray(['#cache' => $metadata]);
     };
@@ -793,7 +840,7 @@ public static function testCacheableMetadataProvider() {
   /**
    * Helper to load the normalizer.
    */
-  protected function getNormalizer() {
+  protected function getNormalizer(): JsonApiDocumentTopLevelNormalizer {
     $normalizer_service = $this->container->get('jsonapi_test_normalizers_kernel.jsonapi_document_toplevel');
     // Simulate what happens when this normalizer service is used via the
     // serializer service, as it is meant to be used.
@@ -801,4 +848,87 @@ protected function getNormalizer() {
     return $normalizer_service;
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  public static function jsonSchemaDataProvider(): array {
+    return [
+      'Empty collection top-level document' => [
+        new JsonApiDocumentTopLevel(
+          new ResourceObjectData([]),
+          new NullIncludedData(),
+          new LinkCollection([])
+        ),
+      ],
+    ];
+  }
+
+  /**
+   * Test the generated resource object normalization against the schema.
+   *
+   * @covers \Drupal\jsonapi\Normalizer\ResourceObjectNormalizer::normalize
+   * @covers \Drupal\jsonapi\Normalizer\ResourceObjectNormalizer::getNormalizationSchema
+   */
+  public function testResourceObjectSchema(): void {
+    [, $resource_object] = $this->getTestContentEntityResource();
+    $serializer = $this->container->get('jsonapi.serializer');
+    $context = ['account' => NULL];
+    $format = $this->getJsonSchemaTestNormalizationFormat();
+    $schema = $serializer->normalize($resource_object, 'json_schema', $context);
+    $this->doCheckSchemaAgainstMetaSchema($schema);
+    $normalized = json_decode(json_encode($serializer->normalize(
+      $resource_object,
+      $format,
+      $context
+    )->getNormalization()));
+    $validator = $this->getValidator();
+    $validator->validate($normalized, json_decode(json_encode($schema)));
+    $this->assertSame([], $validator->getErrors(), 'Validation errors on object ' . print_r($normalized, TRUE) . ' with schema ' . print_r($schema, TRUE));
+  }
+
+  /**
+   * Test the generated config resource object normalization against the schema.
+   *
+   * @covers \Drupal\jsonapi\Normalizer\ResourceObjectNormalizer::normalize
+   * @covers \Drupal\jsonapi\Normalizer\ResourceObjectNormalizer::getNormalizationSchema
+   */
+  public function testConfigEntityResourceObjectSchema(): void {
+    [, $resource_object] = $this->getTestConfigEntityResource();
+    $serializer = $this->container->get('jsonapi.serializer');
+    $context = ['account' => NULL];
+    $format = $this->getJsonSchemaTestNormalizationFormat();
+    $schema = $serializer->normalize($resource_object, 'json_schema', $context);
+    $this->doCheckSchemaAgainstMetaSchema($schema);
+    $normalized = json_decode(json_encode($serializer->normalize(
+      $resource_object,
+      $format,
+      $context
+    )->getNormalization()));
+    $validator = $this->getValidator();
+    $validator->validate($normalized, json_decode(json_encode($schema)));
+    $this->assertSame([], $validator->getErrors(), 'Validation errors on object ' . print_r($normalized, TRUE) . ' with schema ' . print_r($schema, TRUE));
+  }
+
+  public function testTopLevelResourceWithSingleResource(): void {
+    [, $resource_object] = $this->getTestContentEntityResource();
+    $serializer = $this->container->get('jsonapi.serializer');
+    $context = ['account' => NULL];
+    $format = $this->getJsonSchemaTestNormalizationFormat();
+    $topLevel = new JsonApiDocumentTopLevel(
+      new ResourceObjectData([$resource_object]),
+      new NullIncludedData(),
+      new LinkCollection([])
+    );
+    $schema = $serializer->normalize($topLevel, 'json_schema', $context);
+    $this->doCheckSchemaAgainstMetaSchema($schema);
+    $normalized = json_decode(json_encode($serializer->normalize(
+      $topLevel,
+      $format,
+      $context
+    )->getNormalization()));
+    $validator = $this->getValidator();
+    $validator->validate($normalized, json_decode(json_encode($schema)));
+    $this->assertSame([], $validator->getErrors(), 'Validation errors on object ' . print_r($normalized, TRUE) . ' with schema ' . print_r($schema, TRUE));
+  }
+
 }
diff --git a/core/modules/jsonapi/tests/src/Kernel/Normalizer/RelationshipNormalizerTest.php b/core/modules/jsonapi/tests/src/Kernel/Normalizer/RelationshipNormalizerTest.php
index 6fe7a65870f9604b8396c052a5a59016e55e6170..e5abd16c4be4411526da0cbe8dc51645b6adc743 100644
--- a/core/modules/jsonapi/tests/src/Kernel/Normalizer/RelationshipNormalizerTest.php
+++ b/core/modules/jsonapi/tests/src/Kernel/Normalizer/RelationshipNormalizerTest.php
@@ -214,8 +214,9 @@ protected function setUp(): void {
     $this->referencer->save();
 
     // Set up the test dependencies.
-    $this->referencingResourceType = $this->container->get('jsonapi.resource_type.repository')->get('node', 'referencer');
-    $this->normalizer = new RelationshipNormalizer();
+    $resource_type_repository = $this->container->get('jsonapi.resource_type.repository');
+    $this->referencingResourceType = $resource_type_repository->get('node', 'referencer');
+    $this->normalizer = new RelationshipNormalizer($resource_type_repository);
     $this->normalizer->setSerializer($this->container->get('jsonapi.serializer'));
   }
 
diff --git a/core/modules/jsonapi/tests/src/Traits/JsonApiJsonSchemaTestTrait.php b/core/modules/jsonapi/tests/src/Traits/JsonApiJsonSchemaTestTrait.php
new file mode 100644
index 0000000000000000000000000000000000000000..5c073834e99835e04267f83877e4618b9742710d
--- /dev/null
+++ b/core/modules/jsonapi/tests/src/Traits/JsonApiJsonSchemaTestTrait.php
@@ -0,0 +1,52 @@
+<?php
+
+declare(strict_types=1);
+
+namespace Drupal\Tests\jsonapi\Traits;
+
+use Drupal\jsonapi\JsonApiSpec;
+use Drupal\jsonapi\Normalizer\Value\CacheableNormalization;
+use Drupal\Tests\serialization\Traits\JsonSchemaTestTrait;
+use JsonSchema\Constraints\Factory;
+use JsonSchema\Uri\UriRetriever;
+use JsonSchema\Validator;
+
+trait JsonApiJsonSchemaTestTrait {
+
+  use JsonSchemaTestTrait {
+    getNormalizationForValue as parentGetNormalizationForValue;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function getJsonSchemaTestNormalizationFormat(): ?string {
+    return 'api_json';
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function getValidator(): Validator {
+    $uriRetriever = new UriRetriever();
+    $uriRetriever->setTranslation(
+      '|^' . JsonApiSpec::SUPPORTED_SPECIFICATION_JSON_SCHEMA . '#?|',
+      sprintf('file://%s/schema.json', realpath(__DIR__ . '/../../..'))
+    );
+    return new Validator(new Factory(
+      uriRetriever: $uriRetriever,
+    ));
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function getNormalizationForValue(mixed $value): mixed {
+    $normalization = $this->parentGetNormalizationForValue($value);
+    if ($normalization instanceof CacheableNormalization) {
+      return $normalization->getNormalization();
+    }
+    return $normalization;
+  }
+
+}
diff --git a/core/modules/serialization/serialization.services.yml b/core/modules/serialization/serialization.services.yml
index bc67d05596e280359e37fe6607aa38183267f91f..9dc3a849ce9a7e92570cd1c873cff4c3d8e058b0 100644
--- a/core/modules/serialization/serialization.services.yml
+++ b/core/modules/serialization/serialization.services.yml
@@ -5,7 +5,7 @@ services:
   _defaults:
     autoconfigure: true
   serializer:
-    class: Symfony\Component\Serializer\Serializer
+    class: Drupal\serialization\Serializer\Serializer
     arguments: [{  }, {  }]
   serializer.normalizer.config_entity:
     class: Drupal\serialization\Normalizer\ConfigEntityNormalizer
diff --git a/core/modules/serialization/src/Normalizer/DateTimeIso8601Normalizer.php b/core/modules/serialization/src/Normalizer/DateTimeIso8601Normalizer.php
index 12e79be415b53d3e44be392541afc7b8a1a90142..187494a66a76c5827c38b8914e5aabfab205a45b 100644
--- a/core/modules/serialization/src/Normalizer/DateTimeIso8601Normalizer.php
+++ b/core/modules/serialization/src/Normalizer/DateTimeIso8601Normalizer.php
@@ -2,6 +2,7 @@
 
 namespace Drupal\serialization\Normalizer;
 
+use Drupal\Core\Serialization\Attribute\JsonSchema;
 use Drupal\Core\TypedData\Plugin\DataType\DateTimeIso8601;
 use Drupal\datetime\Plugin\Field\FieldType\DateTimeItem;
 use Drupal\datetime\Plugin\Field\FieldType\DateTimeItemInterface;
@@ -14,6 +15,8 @@
  */
 class DateTimeIso8601Normalizer extends DateTimeNormalizer {
 
+  use SchematicNormalizerHelperTrait;
+
   /**
    * {@inheritdoc}
    */
@@ -32,18 +35,22 @@ class DateTimeIso8601Normalizer extends DateTimeNormalizer {
   /**
    * {@inheritdoc}
    */
-  public function normalize($datetime, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
-    assert($datetime instanceof DateTimeIso8601);
-    $field_item = $datetime->getParent();
+  #[JsonSchema(['type' => 'string', 'format' => 'date'])]
+  public function normalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
+    if ($format === 'json_schema') {
+      return $this->getNormalizationSchema($object, $context);
+    }
+    assert($object instanceof DateTimeIso8601);
+    $field_item = $object->getParent();
     // @todo Remove this in https://www.drupal.org/project/drupal/issues/2958416.
     if ($field_item instanceof DateTimeItem && $field_item->getFieldDefinition()->getFieldStorageDefinition()->getSetting('datetime_type') === DateTimeItem::DATETIME_TYPE_DATE) {
-      $drupal_date_time = $datetime->getDateTime();
+      $drupal_date_time = $object->getDateTime();
       if ($drupal_date_time === NULL) {
         return $drupal_date_time;
       }
       return $drupal_date_time->format($this->allowedFormats['date-only']);
     }
-    return parent::normalize($datetime, $format, $context);
+    return parent::normalize($object, $format, $context);
   }
 
   /**
diff --git a/core/modules/serialization/src/Normalizer/DateTimeNormalizer.php b/core/modules/serialization/src/Normalizer/DateTimeNormalizer.php
index cfbef830bc0a3b8bd9b82ece397775bcd12707e9..3e93629980bffebe23165f86de05550419af42ac 100644
--- a/core/modules/serialization/src/Normalizer/DateTimeNormalizer.php
+++ b/core/modules/serialization/src/Normalizer/DateTimeNormalizer.php
@@ -3,6 +3,7 @@
 namespace Drupal\serialization\Normalizer;
 
 use Drupal\Core\Config\ConfigFactoryInterface;
+use Drupal\Core\Serialization\Attribute\JsonSchema;
 use Drupal\Core\TypedData\Type\DateTimeInterface;
 use Symfony\Component\Serializer\Exception\UnexpectedValueException;
 use Symfony\Component\Serializer\Normalizer\DenormalizerInterface;
@@ -14,6 +15,8 @@
  */
 class DateTimeNormalizer extends NormalizerBase implements DenormalizerInterface {
 
+  use SchematicNormalizerTrait;
+
   /**
    * Allowed datetime formats for the denormalizer.
    *
@@ -49,9 +52,10 @@ public function __construct(ConfigFactoryInterface $config_factory) {
   /**
    * {@inheritdoc}
    */
-  public function normalize($datetime, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
-    assert($datetime instanceof DateTimeInterface);
-    $drupal_date_time = $datetime->getDateTime();
+  #[JsonSchema(['type' => 'string', 'format' => 'date-time'])]
+  public function doNormalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
+    assert($object instanceof DateTimeInterface);
+    $drupal_date_time = $object->getDateTime();
     if ($drupal_date_time === NULL) {
       return $drupal_date_time;
     }
diff --git a/core/modules/serialization/src/Normalizer/JsonSchemaReflectionTrait.php b/core/modules/serialization/src/Normalizer/JsonSchemaReflectionTrait.php
new file mode 100644
index 0000000000000000000000000000000000000000..672213756501f190b6d1d989342c85f95217d6c5
--- /dev/null
+++ b/core/modules/serialization/src/Normalizer/JsonSchemaReflectionTrait.php
@@ -0,0 +1,44 @@
+<?php
+
+declare(strict_types=1);
+
+namespace Drupal\serialization\Normalizer;
+
+use Drupal\Core\Serialization\Attribute\JsonSchema;
+
+trait JsonSchemaReflectionTrait {
+
+  /**
+   * Get a JSON Schema based on method reflection.
+   *
+   * @param object $object
+   *   Object to reflect.
+   * @param string $method
+   *   Method to reflect.
+   * @param array $fallback
+   *   Fallback. Defaults to an empty array, which is a matches-all schema.
+   * @param bool $nullable
+   *   If a schema is returned from reflection, whether to add a null option.
+   *
+   * @return array
+   *   JSON Schema.
+   */
+  protected function getJsonSchemaForMethod(mixed $object, string $method, array $fallback = [], bool $nullable = FALSE): array {
+    $schemas = [];
+    if ((is_object($object) || class_exists($object)) && method_exists($object, $method)) {
+      $reflection = new \ReflectionMethod($object, $method);
+      $schemas = $reflection->getAttributes(JsonSchema::class);
+    }
+    if (count($schemas) === 0) {
+      return $fallback;
+    }
+    $schemas = array_values(array_filter([
+      ...array_map(fn ($schema) => $schema->newInstance()->getJsonSchema(), $schemas),
+      $nullable ? ['type' => 'null'] : NULL,
+    ]));
+    return count($schemas) === 1
+      ? current($schemas)
+      : ['oneOf' => $schemas];
+  }
+
+}
diff --git a/core/modules/serialization/src/Normalizer/MarkupNormalizer.php b/core/modules/serialization/src/Normalizer/MarkupNormalizer.php
index e3df030735ea8a8eda9e3d4968726a0be7e8e0ed..d6bacd72d2281097bd504d90dcf64c8cd4c16ecc 100644
--- a/core/modules/serialization/src/Normalizer/MarkupNormalizer.php
+++ b/core/modules/serialization/src/Normalizer/MarkupNormalizer.php
@@ -9,13 +9,30 @@
  */
 class MarkupNormalizer extends NormalizerBase {
 
+  use SchematicNormalizerTrait;
+  use JsonSchemaReflectionTrait;
+
   /**
    * {@inheritdoc}
    */
-  public function normalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
+  public function doNormalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
     return (string) $object;
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  public function getNormalizationSchema(mixed $object, array $context = []): array {
+    return $this->getJsonSchemaForMethod(
+      $object,
+      '__toString',
+      [
+        'type' => 'string',
+        'description' => 'May contain HTML markup.',
+      ]
+    );
+  }
+
   /**
    * {@inheritdoc}
    */
diff --git a/core/modules/serialization/src/Normalizer/NormalizerBase.php b/core/modules/serialization/src/Normalizer/NormalizerBase.php
index ff4ab29ef366d4383c96a71fddf85c9253330ae0..b0220ce933f78bb288da2a816dad7b4695ad8cb9 100644
--- a/core/modules/serialization/src/Normalizer/NormalizerBase.php
+++ b/core/modules/serialization/src/Normalizer/NormalizerBase.php
@@ -69,7 +69,9 @@ public function supportsDenormalization($data, string $type, ?string $format = N
    *   specified this will return TRUE.
    */
   protected function checkFormat($format = NULL) {
-    if (!isset($format) || !isset($this->format)) {
+    // The format 'json_schema' is special-cased as it requires explicit
+    // support, as opposed to a permissive default-case value normalization.
+    if (!isset($format) || (!isset($this->format) && $format !== 'json_schema')) {
       return TRUE;
     }
 
diff --git a/core/modules/serialization/src/Normalizer/NullNormalizer.php b/core/modules/serialization/src/Normalizer/NullNormalizer.php
index 63d7c24e728398221d954dbaf96fe4fb57c56d47..e4c5a11a5cb8cff39b9cf81de4976f49ed548f72 100644
--- a/core/modules/serialization/src/Normalizer/NullNormalizer.php
+++ b/core/modules/serialization/src/Normalizer/NullNormalizer.php
@@ -2,11 +2,15 @@
 
 namespace Drupal\serialization\Normalizer;
 
+use Drupal\Core\Serialization\Attribute\JsonSchema;
+
 /**
  * Null normalizer.
  */
 class NullNormalizer extends NormalizerBase {
 
+  use SchematicNormalizerTrait;
+
   /**
    * The interface or class that this Normalizer supports.
    *
@@ -27,10 +31,18 @@ public function __construct($supported_interface_of_class) {
   /**
    * {@inheritdoc}
    */
-  public function normalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
+  #[JsonSchema(['type' => 'null'])]
+  public function doNormalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
     return NULL;
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  protected function getNormalizationSchema(mixed $object, array $context = []): array {
+    return ['type' => 'null'];
+  }
+
   /**
    * {@inheritdoc}
    */
diff --git a/core/modules/serialization/src/Normalizer/PrimitiveDataNormalizer.php b/core/modules/serialization/src/Normalizer/PrimitiveDataNormalizer.php
index a34784953795c3295e220eac0e717efe50066fa4..d4b2e50514dee8329805d88e3a97cdc52bbb3554 100644
--- a/core/modules/serialization/src/Normalizer/PrimitiveDataNormalizer.php
+++ b/core/modules/serialization/src/Normalizer/PrimitiveDataNormalizer.php
@@ -4,6 +4,7 @@
 
 use Drupal\Core\Field\FieldItemInterface;
 use Drupal\Core\TypedData\PrimitiveInterface;
+use Drupal\Core\TypedData\TypedDataInterface;
 
 /**
  * Converts primitive data objects to their casted values.
@@ -11,11 +12,13 @@
 class PrimitiveDataNormalizer extends NormalizerBase {
 
   use SerializedColumnNormalizerTrait;
+  use SchematicNormalizerTrait;
+  use JsonSchemaReflectionTrait;
 
   /**
    * {@inheritdoc}
    */
-  public function normalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
+  public function doNormalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
     // Add cacheability if applicable.
     $this->addCacheableDependency($context, $object);
 
@@ -36,6 +39,19 @@ public function normalize($object, $format = NULL, array $context = []): array|s
     return $object->getValue() === NULL ? NULL : $object->getCastedValue();
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  public function getNormalizationSchema(mixed $object, array $context = []): array {
+    $nullable = !$object instanceof TypedDataInterface || !$object->getDataDefinition()->isRequired();
+    return $this->getJsonSchemaForMethod(
+      $object,
+      'getCastedValue',
+      ['$comment' => 'Unable to provide schema, no type specified.'],
+      $nullable,
+    );
+  }
+
   /**
    * {@inheritdoc}
    */
diff --git a/core/modules/serialization/src/Normalizer/SchematicNormalizerFallbackTrait.php b/core/modules/serialization/src/Normalizer/SchematicNormalizerFallbackTrait.php
new file mode 100644
index 0000000000000000000000000000000000000000..161a19438f968948c0bca7a5fd15c503372de54c
--- /dev/null
+++ b/core/modules/serialization/src/Normalizer/SchematicNormalizerFallbackTrait.php
@@ -0,0 +1,19 @@
+<?php
+
+declare(strict_types=1);
+
+namespace Drupal\serialization\Normalizer;
+
+/**
+ * Trait for generating helpful schema-generation fallback messages.
+ */
+trait SchematicNormalizerFallbackTrait {
+
+  public static function generateNoSchemaAvailableMessage(mixed $object): string {
+    $baseMessage = 'See https://www.drupal.org/node/3424710 for information on implementing schemas in your program code.';
+    return is_object($object)
+      ? sprintf('No schema is defined for property of type %s. %s', $object::class, $baseMessage)
+      : sprintf('No schema defined for this property. %s', $baseMessage);
+  }
+
+}
diff --git a/core/modules/serialization/src/Normalizer/SchematicNormalizerHelperTrait.php b/core/modules/serialization/src/Normalizer/SchematicNormalizerHelperTrait.php
new file mode 100644
index 0000000000000000000000000000000000000000..81842e93b3dd16314caca4d927e2f4e12b5ee9a1
--- /dev/null
+++ b/core/modules/serialization/src/Normalizer/SchematicNormalizerHelperTrait.php
@@ -0,0 +1,46 @@
+<?php
+
+declare(strict_types=1);
+
+namespace Drupal\serialization\Normalizer;
+
+trait SchematicNormalizerHelperTrait {
+
+  use JsonSchemaReflectionTrait;
+
+  /**
+   * Retrieve JSON Schema for the normalization.
+   *
+   * @param mixed $object
+   *   Supported object or class/interface name being normalized.
+   * @param array $context
+   *   Context options. Well-defined keys include:
+   *   - dialect: Used to specify a dialect for the desired schema being
+   *     generated. The dialect meta-schema MUST extend JSON Schema draft
+   *     2020-12 or later. Normalizers MAY choose to return a schema with
+   *     keywords supported by a dialect it supports, but only when they
+   *     are supported by the dialect specified in this key. For instance,
+   *     normalizers may return a schema with a 'discriminator' as supported
+   *     by OpenAPI if that dialect is passed, but return a more permissive but
+   *     less specific schema when it is not.
+   *
+   * @return array
+   *   JSON Schema for the normalization, conforming to version draft 2020-12.
+   *
+   * @see https://json-schema.org/specification#specification-documents
+   */
+  protected function getNormalizationSchema(mixed $object, array $context = []): array {
+    return $this->getJsonSchemaForMethod($this, 'normalize', ['$comment' => 'No schema available.']);
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function checkFormat($format = NULL) {
+    if ($format === 'json_schema') {
+      return TRUE;
+    }
+    return parent::checkFormat($format);
+  }
+
+}
diff --git a/core/modules/serialization/src/Normalizer/SchematicNormalizerTrait.php b/core/modules/serialization/src/Normalizer/SchematicNormalizerTrait.php
new file mode 100644
index 0000000000000000000000000000000000000000..bec0024cff31fe276abedc1ad5e236a10e1aeea8
--- /dev/null
+++ b/core/modules/serialization/src/Normalizer/SchematicNormalizerTrait.php
@@ -0,0 +1,57 @@
+<?php
+
+declare(strict_types=1);
+
+namespace Drupal\serialization\Normalizer;
+
+/**
+ * Trait for normalizers which can also provide JSON Schema.
+ *
+ * To implement this trait, convert the existing normalizer's ::normalize()
+ * method to ::doNormalize().
+ *
+ * Due to trait inheritance rules, this trait cannot be used with normalizers
+ * which call parent::normalize() during normalization (will result in infinite
+ * recursion). Instead, use SchematicNormalizerHelperTrait and conditionally
+ * call ::getNormalizationSchema() in ::normalize(). See
+ * DateTimeIso8601Normalizer::normalize() for an example.
+ */
+trait SchematicNormalizerTrait {
+
+  use SchematicNormalizerHelperTrait;
+  use SchematicNormalizerFallbackTrait;
+
+  /**
+   * {@inheritdoc}
+   */
+  public function normalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
+    if ($format === 'json_schema') {
+      return $this->getNormalizationSchema($object, $context);
+    }
+    return $this->doNormalize($object, $format, $context);
+  }
+
+  /**
+   * Normalizes an object into a set of arrays/scalars.
+   *
+   * @param mixed $object
+   *   Object to normalize.
+   * @param string|null $format
+   *   Format the normalization result will be encoded as.
+   * @param array $context
+   *   Context options for the normalizer.
+   *
+   * @return array|string|int|float|bool|\ArrayObject|null
+   *   The normalization. An \ArrayObject is used to make sure an empty object
+   *   is encoded as an object not an array.
+   */
+  abstract protected function doNormalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL;
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function getNormalizationSchema(mixed $object, array $context = []): array {
+    return $this->getJsonSchemaForMethod($this, 'doNormalize', ['$comment' => static::generateNoSchemaAvailableMessage($object)]);
+  }
+
+}
diff --git a/core/modules/serialization/src/Normalizer/TimestampNormalizer.php b/core/modules/serialization/src/Normalizer/TimestampNormalizer.php
index abc9f2ba8079b68b149f6c34e1312f5f1318861e..46d13e51a7b4be917cdd198e8e549110d74890ab 100644
--- a/core/modules/serialization/src/Normalizer/TimestampNormalizer.php
+++ b/core/modules/serialization/src/Normalizer/TimestampNormalizer.php
@@ -15,6 +15,8 @@
  */
 class TimestampNormalizer extends DateTimeNormalizer {
 
+  use SchematicNormalizerTrait;
+
   /**
    * {@inheritdoc}
    */
diff --git a/core/modules/serialization/src/Normalizer/TypedDataNormalizer.php b/core/modules/serialization/src/Normalizer/TypedDataNormalizer.php
index 96866b23685856d777bee00f42e662a6d3bcb6db..8aa6a9f673667f7cb9c91a62ed89ad1c3ee0287a 100644
--- a/core/modules/serialization/src/Normalizer/TypedDataNormalizer.php
+++ b/core/modules/serialization/src/Normalizer/TypedDataNormalizer.php
@@ -5,14 +5,17 @@
 use Drupal\Core\TypedData\TypedDataInterface;
 
 /**
- * Converts typed data objects to arrays.
+ * Normalizes typed data objects into strings or arrays.
  */
 class TypedDataNormalizer extends NormalizerBase {
 
+  use SchematicNormalizerTrait;
+  use JsonSchemaReflectionTrait;
+
   /**
    * {@inheritdoc}
    */
-  public function normalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
+  public function doNormalize($object, $format = NULL, array $context = []): array|string|int|float|bool|\ArrayObject|NULL {
     $this->addCacheableDependency($context, $object);
     $value = $object->getValue();
     // Support for stringable value objects: avoid numerous custom normalizers.
@@ -22,6 +25,27 @@ public function normalize($object, $format = NULL, array $context = []): array|s
     return $value;
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  protected function getNormalizationSchema(mixed $object, array $context = []): array {
+    assert($object instanceof TypedDataInterface);
+    $value = $object->getValue();
+    $nullable = !$object->getDataDefinition()->isRequired();
+    // Match the special-cased logic in ::normalize().
+    if (is_object($value) && method_exists($value, '__toString')) {
+      return $nullable
+        ? ['oneOf' => ['string', 'null']]
+        : ['type' => 'string'];
+    }
+    return $this->getJsonSchemaForMethod(
+      $object,
+      'getValue',
+      ['$comment' => static::generateNoSchemaAvailableMessage($object)],
+      $nullable
+    );
+  }
+
   /**
    * {@inheritdoc}
    */
diff --git a/core/modules/serialization/src/Serializer/JsonSchemaProviderSerializerInterface.php b/core/modules/serialization/src/Serializer/JsonSchemaProviderSerializerInterface.php
new file mode 100644
index 0000000000000000000000000000000000000000..dfb134d40e216f59301a7476df49b15947a3e8d8
--- /dev/null
+++ b/core/modules/serialization/src/Serializer/JsonSchemaProviderSerializerInterface.php
@@ -0,0 +1,25 @@
+<?php
+
+declare(strict_types=1);
+
+namespace Drupal\serialization\Serializer;
+
+interface JsonSchemaProviderSerializerInterface {
+
+  /**
+   * Convenience method to get a JSON schema.
+   *
+   * Unlike calling ::normalize() with $format of 'json_schema' directly, this
+   * method always returns a schema, even if it's empty.
+   *
+   * @param mixed $object
+   *   Object or interface/class name for which to retrieve a schema.
+   * @param array $context
+   *   Normalization context.
+   *
+   * @return array
+   *   Schema.
+   */
+  public function getJsonSchema(mixed $object, array $context): array;
+
+}
diff --git a/core/modules/serialization/src/Serializer/JsonSchemaProviderSerializerTrait.php b/core/modules/serialization/src/Serializer/JsonSchemaProviderSerializerTrait.php
new file mode 100644
index 0000000000000000000000000000000000000000..77e00675cfbf93272a9584f8749314bfe2d67b68
--- /dev/null
+++ b/core/modules/serialization/src/Serializer/JsonSchemaProviderSerializerTrait.php
@@ -0,0 +1,27 @@
+<?php
+
+declare(strict_types=1);
+
+namespace Drupal\serialization\Serializer;
+
+use Drupal\serialization\Normalizer\SchematicNormalizerFallbackTrait;
+use Symfony\Component\Serializer\Exception\NotNormalizableValueException;
+
+trait JsonSchemaProviderSerializerTrait {
+
+  use SchematicNormalizerFallbackTrait;
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getJsonSchema(mixed $object, array $context): array {
+    try {
+      $normalizer_schema = $this->normalize($object, 'json_schema', $context);
+    }
+    catch (NotNormalizableValueException) {
+      $normalizer_schema = ['$comment' => static::generateNoSchemaAvailableMessage($object)];
+    }
+    return $normalizer_schema;
+  }
+
+}
diff --git a/core/modules/serialization/src/Serializer/Serializer.php b/core/modules/serialization/src/Serializer/Serializer.php
new file mode 100644
index 0000000000000000000000000000000000000000..1bba229647f6da2628e57add0b8185a925d7b686
--- /dev/null
+++ b/core/modules/serialization/src/Serializer/Serializer.php
@@ -0,0 +1,16 @@
+<?php
+
+declare(strict_types=1);
+
+namespace Drupal\serialization\Serializer;
+
+use Symfony\Component\Serializer\Serializer as SymfonySerializer;
+
+/**
+ * Serializer with JSON Schema generation convenience methods.
+ */
+class Serializer extends SymfonySerializer implements JsonSchemaProviderSerializerInterface {
+
+  use JsonSchemaProviderSerializerTrait;
+
+}
diff --git a/core/modules/serialization/src/json-schema-draft-04-meta-schema.json b/core/modules/serialization/src/json-schema-draft-04-meta-schema.json
new file mode 100644
index 0000000000000000000000000000000000000000..bcbb84743e3838fab7cbec5f0a5bcbafcfc99136
--- /dev/null
+++ b/core/modules/serialization/src/json-schema-draft-04-meta-schema.json
@@ -0,0 +1,149 @@
+{
+    "id": "http://json-schema.org/draft-04/schema#",
+    "$schema": "http://json-schema.org/draft-04/schema#",
+    "description": "Core schema meta-schema",
+    "definitions": {
+        "schemaArray": {
+            "type": "array",
+            "minItems": 1,
+            "items": { "$ref": "#" }
+        },
+        "positiveInteger": {
+            "type": "integer",
+            "minimum": 0
+        },
+        "positiveIntegerDefault0": {
+            "allOf": [ { "$ref": "#/definitions/positiveInteger" }, { "default": 0 } ]
+        },
+        "simpleTypes": {
+            "enum": [ "array", "boolean", "integer", "null", "number", "object", "string" ]
+        },
+        "stringArray": {
+            "type": "array",
+            "items": { "type": "string" },
+            "minItems": 1,
+            "uniqueItems": true
+        }
+    },
+    "type": "object",
+    "properties": {
+        "id": {
+            "type": "string"
+        },
+        "$schema": {
+            "type": "string"
+        },
+        "title": {
+            "type": "string"
+        },
+        "description": {
+            "type": "string"
+        },
+        "default": {},
+        "multipleOf": {
+            "type": "number",
+            "minimum": 0,
+            "exclusiveMinimum": true
+        },
+        "maximum": {
+            "type": "number"
+        },
+        "exclusiveMaximum": {
+            "type": "boolean",
+            "default": false
+        },
+        "minimum": {
+            "type": "number"
+        },
+        "exclusiveMinimum": {
+            "type": "boolean",
+            "default": false
+        },
+        "maxLength": { "$ref": "#/definitions/positiveInteger" },
+        "minLength": { "$ref": "#/definitions/positiveIntegerDefault0" },
+        "pattern": {
+            "type": "string",
+            "format": "regex"
+        },
+        "additionalItems": {
+            "anyOf": [
+                { "type": "boolean" },
+                { "$ref": "#" }
+            ],
+            "default": {}
+        },
+        "items": {
+            "anyOf": [
+                { "$ref": "#" },
+                { "$ref": "#/definitions/schemaArray" }
+            ],
+            "default": {}
+        },
+        "maxItems": { "$ref": "#/definitions/positiveInteger" },
+        "minItems": { "$ref": "#/definitions/positiveIntegerDefault0" },
+        "uniqueItems": {
+            "type": "boolean",
+            "default": false
+        },
+        "maxProperties": { "$ref": "#/definitions/positiveInteger" },
+        "minProperties": { "$ref": "#/definitions/positiveIntegerDefault0" },
+        "required": { "$ref": "#/definitions/stringArray" },
+        "additionalProperties": {
+            "anyOf": [
+                { "type": "boolean" },
+                { "$ref": "#" }
+            ],
+            "default": {}
+        },
+        "definitions": {
+            "type": "object",
+            "additionalProperties": { "$ref": "#" },
+            "default": {}
+        },
+        "properties": {
+            "type": "object",
+            "additionalProperties": { "$ref": "#" },
+            "default": {}
+        },
+        "patternProperties": {
+            "type": "object",
+            "additionalProperties": { "$ref": "#" },
+            "default": {}
+        },
+        "dependencies": {
+            "type": "object",
+            "additionalProperties": {
+                "anyOf": [
+                    { "$ref": "#" },
+                    { "$ref": "#/definitions/stringArray" }
+                ]
+            }
+        },
+        "enum": {
+            "type": "array",
+            "minItems": 1,
+            "uniqueItems": true
+        },
+        "type": {
+            "anyOf": [
+                { "$ref": "#/definitions/simpleTypes" },
+                {
+                    "type": "array",
+                    "items": { "$ref": "#/definitions/simpleTypes" },
+                    "minItems": 1,
+                    "uniqueItems": true
+                }
+            ]
+        },
+        "format": { "type": "string" },
+        "allOf": { "$ref": "#/definitions/schemaArray" },
+        "anyOf": { "$ref": "#/definitions/schemaArray" },
+        "oneOf": { "$ref": "#/definitions/schemaArray" },
+        "not": { "$ref": "#" }
+    },
+    "dependencies": {
+        "exclusiveMaximum": [ "maximum" ],
+        "exclusiveMinimum": [ "minimum" ]
+    },
+    "default": {}
+}
diff --git a/core/modules/serialization/src/json-schema-draft-04-meta-schema.json.LICENSE.txt b/core/modules/serialization/src/json-schema-draft-04-meta-schema.json.LICENSE.txt
new file mode 100644
index 0000000000000000000000000000000000000000..3f1bab7bc7ff0341e9ec711d0f3c16eae81d4e2a
--- /dev/null
+++ b/core/modules/serialization/src/json-schema-draft-04-meta-schema.json.LICENSE.txt
@@ -0,0 +1,11 @@
+Copyright 2023 JSON Schema Specification Authors
+
+Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
+
+1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
+
+2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
+
+3. Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS “AS IS” AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
diff --git a/core/modules/serialization/tests/src/Traits/JsonSchemaTestTrait.php b/core/modules/serialization/tests/src/Traits/JsonSchemaTestTrait.php
new file mode 100644
index 0000000000000000000000000000000000000000..a9c5020025b67ecef3ec321c743466a52e3b1c58
--- /dev/null
+++ b/core/modules/serialization/tests/src/Traits/JsonSchemaTestTrait.php
@@ -0,0 +1,171 @@
+<?php
+
+declare(strict_types=1);
+
+namespace Drupal\Tests\serialization\Traits;
+
+use Drupal\serialization\Normalizer\PrimitiveDataNormalizer;
+use JsonSchema\Validator;
+use Prophecy\Prophecy\ObjectProphecy;
+use Symfony\Component\Serializer\Normalizer\NormalizerInterface;
+
+/**
+ * Trait for testing JSON Schema validity and fit to sample data.
+ *
+ * In most cases, tests need only implement the abstract method for providing
+ * a full set of representative normalized values.
+ */
+trait JsonSchemaTestTrait {
+
+  /**
+   * Format that should be used when performing test normalizations.
+   */
+  protected function getJsonSchemaTestNormalizationFormat(): ?string {
+    return NULL;
+  }
+
+  /**
+   * Data provider for ::testNormalizedValuesAgainstJsonSchema.
+   *
+   * @return array
+   *   Array of possible normalized values to validate the JSON schema against.
+   */
+  abstract public static function jsonSchemaDataProvider(): array;
+
+  /**
+   * Method to make prophecy public for use in data provider closures.
+   */
+  public function doProphesize(?string $classOrInterface = NULL): ObjectProphecy {
+    return $this->prophesize($classOrInterface);
+  }
+
+  /**
+   * Test that a valid schema is returned for the explicitly supported types.
+   *
+   * This is in many cases an interface, which would not be normalized directly,
+   * however the schema should never return an invalid type. An empty array or
+   * a type with only a '$comment' member is valid.
+   *
+   * @dataProvider supportedTypesDataProvider
+   */
+  public function testSupportedTypesSchemaIsValid(string $type): void {
+    $this->doTestJsonSchemaIsValid($type, TRUE);
+  }
+
+  /**
+   * Check a schema is valid against the meta-schema.
+   *
+   * @param array $defined_schema
+   *   Defined schema.
+   * @param bool $accept_no_schema_type
+   *   Whether to accept a schema with no meaningful type construct.
+   */
+  protected function doCheckSchemaAgainstMetaSchema(array $defined_schema, bool $accept_no_schema_type = FALSE): void {
+    $validator = $this->getValidator();
+    // Ensure the schema contains a meaningful type construct.
+    if (!$accept_no_schema_type) {
+      $this->assertFalse(empty(array_filter(array_keys($defined_schema), fn($key) => in_array($key, ['type', 'allOf', 'oneOf', 'anyOf', 'not', '$ref']))));
+    }
+    // All associative arrays must be encoded as objects.
+    $schema = json_decode(json_encode($defined_schema));
+    $validator->validate(
+      $schema,
+      // Schemas must be compatible with draft 2020-12, however the validation
+      // library, justinrainbow/json-schema, only supports up to draft-04.
+      // Generally speaking this isn't an issue as there are few changes to the
+      // spec that will affect core-provided normalization schemas, and we have
+      // little other option in the PHP ecosystem for runtime validation.
+      // @see https://www.drupal.org/project/drupal/issues/3350943
+      (object) ['$ref' => 'file://' . __DIR__ . '/../../../src/json-schema-draft-04-meta-schema.json']
+    );
+    $this->assertTrue($validator->isValid());
+  }
+
+  /**
+   * Validate the normalizer's JSON schema.
+   *
+   * @param mixed $type
+   *   Object/type being normalized.
+   * @param bool $accept_no_schema_type
+   *   Whether to accept a schema with no meaningful type.
+   *
+   * @return array
+   *   Schema, so later tests can avoid retrieving it again.
+   */
+  public function doTestJsonSchemaIsValid(mixed $type, bool $accept_no_schema_type = FALSE): array {
+    $defined_schema = $this->getNormalizer()->normalize($type, 'json_schema');
+    $this->doCheckSchemaAgainstMetaSchema($defined_schema, $accept_no_schema_type);
+    return $defined_schema;
+  }
+
+  /**
+   * @return array
+   *   Supported types for which to test schema generation.
+   */
+  public static function supportedTypesDataProvider(): array {
+    return array_map(fn ($type) => [$type], array_keys((new PrimitiveDataNormalizer())->getSupportedTypes(NULL)));
+  }
+
+  /**
+   * Test normalized values against the JSON schema.
+   *
+   * @dataProvider jsonSchemaDataProvider
+   */
+  public function testNormalizedValuesAgainstJsonSchema(mixed $value): void {
+    // Explicitly test the JSON Schema's validity here, because it will depend
+    // on the type of the data being normalized, e.g. a class implementing the
+    // interface defined in ::getSupportedTypes().
+    if ($value instanceof \Closure) {
+      $value = $value($this);
+    }
+    $schema = $this->doTestJsonSchemaIsValid($value);
+    $validator = $this->getValidator();
+    // Test the value validates to the schema.
+    // All associative arrays must be encoded as objects.
+    $normalized = json_decode(json_encode($this->getNormalizationForValue($value)));
+    $validator->validate($normalized, $schema);
+    $this->assertSame([], $validator->getErrors(), 'Validation errors on object ' . print_r($normalized, TRUE) . ' with schema ' . print_r($schema, TRUE));
+  }
+
+  /**
+   * Helper method to retrieve the normalizer.
+   *
+   * Override this method if the normalizer has a custom getter or is not
+   * already present at $this->normalizer.
+   *
+   * @return \Symfony\Component\Serializer\Normalizer\NormalizerInterface
+   *   The normalizer under test.
+   */
+  protected function getNormalizer(): NormalizerInterface {
+    return $this->normalizer;
+  }
+
+  /**
+   * Get the normalization for a value.
+   *
+   * Override this method if the normalization needs further processing, e.g.
+   * in the case of JSON:API module's CacheableDependencyInterface.
+   *
+   * @param mixed $value
+   *   Value to be normalized.
+   *
+   * @return mixed
+   *   Final normalized value.
+   */
+  protected function getNormalizationForValue(mixed $value): mixed {
+    return $this->getNormalizer()->normalize($value, $this->getJsonSchemaTestNormalizationFormat());
+  }
+
+  /**
+   * Get the JSON Schema Validator.
+   *
+   * Override this method to add additional schema translations to the loader.
+   *
+   * @return \JsonSchema\Validator
+   *   Schema validator.
+   */
+  protected function getValidator(): Validator {
+    return new Validator();
+  }
+
+}
diff --git a/core/modules/serialization/tests/src/Unit/Normalizer/DateTimeIso8601NormalizerTest.php b/core/modules/serialization/tests/src/Unit/Normalizer/DateTimeIso8601NormalizerTest.php
index 59b01a1bbbd4c0530ec8d90c494770814fd699e5..0cf16102e6b832cc1418baa496438de70bef9eb3 100644
--- a/core/modules/serialization/tests/src/Unit/Normalizer/DateTimeIso8601NormalizerTest.php
+++ b/core/modules/serialization/tests/src/Unit/Normalizer/DateTimeIso8601NormalizerTest.php
@@ -15,6 +15,7 @@
 use Drupal\Core\TypedData\Type\DateTimeInterface;
 use Drupal\datetime\Plugin\Field\FieldType\DateTimeItem;
 use Drupal\serialization\Normalizer\DateTimeIso8601Normalizer;
+use Drupal\Tests\serialization\Traits\JsonSchemaTestTrait;
 use Drupal\Tests\UnitTestCase;
 use Prophecy\Argument;
 use Symfony\Component\Serializer\Exception\InvalidArgumentException;
@@ -30,6 +31,8 @@
  */
 class DateTimeIso8601NormalizerTest extends UnitTestCase {
 
+  use JsonSchemaTestTrait;
+
   /**
    * The tested data type's normalizer.
    *
@@ -255,6 +258,39 @@ public function testDenormalizeNoTargetInstanceOrFieldDefinitionException(): voi
     $this->normalizer->denormalize('', DateTimeIso8601::class, NULL, []);
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  public static function jsonSchemaDataProvider(): array {
+    $case = function (UnitTestCase $test) {
+      assert(in_array(JsonSchemaTestTrait::class, class_uses($test)));
+      $field_item = $test->doProphesize(DateTimeItem::class);
+      $data = $test->doProphesize(DateTimeIso8601::class);
+
+      $field_storage_definition = $test->doProphesize(FieldStorageDefinitionInterface::class);
+      $field_storage_definition->getSetting('datetime_type')
+        ->willReturn(DateTimeItem::DATETIME_TYPE_DATE);
+      $field_definition = $test->doProphesize(FieldDefinitionInterface::class);
+      $field_definition->getFieldStorageDefinition()
+        ->willReturn($field_storage_definition);
+      $field_item->getFieldDefinition()
+        ->willReturn($field_definition);
+      $data->getParent()
+        ->willReturn($field_item);
+      $drupal_date_time = $test->doProphesize(DateTimeIso8601NormalizerTestDrupalDateTime::class);
+      $drupal_date_time->setTimezone(new \DateTimeZone('Australia/Sydney'))
+        ->willReturn($drupal_date_time->reveal());
+      $drupal_date_time->format('Y-m-d')
+        ->willReturn('1991-09-19');
+      $data->getDateTime()
+        ->willReturn($drupal_date_time->reveal());
+      return $data->reveal();
+    };
+    return [
+      'ISO 8601 date-only' => [fn (UnitTestCase $test) => $case($test)],
+    ];
+  }
+
 }
 
 /**
diff --git a/core/modules/serialization/tests/src/Unit/Normalizer/DateTimeNormalizerTest.php b/core/modules/serialization/tests/src/Unit/Normalizer/DateTimeNormalizerTest.php
index f12d82eaf1665354935a4c75f8b4763df138df9e..39e96006c06828926ae37d6ef3f8f3c979b72f68 100644
--- a/core/modules/serialization/tests/src/Unit/Normalizer/DateTimeNormalizerTest.php
+++ b/core/modules/serialization/tests/src/Unit/Normalizer/DateTimeNormalizerTest.php
@@ -11,6 +11,7 @@
 use Drupal\Core\TypedData\Plugin\DataType\IntegerData;
 use Drupal\Core\TypedData\Type\DateTimeInterface;
 use Drupal\serialization\Normalizer\DateTimeNormalizer;
+use Drupal\Tests\serialization\Traits\JsonSchemaTestTrait;
 use Drupal\Tests\UnitTestCase;
 use Symfony\Component\Serializer\Exception\UnexpectedValueException;
 
@@ -23,6 +24,8 @@
  */
 class DateTimeNormalizerTest extends UnitTestCase {
 
+  use JsonSchemaTestTrait;
+
   /**
    * The tested data type's normalizer.
    *
@@ -176,6 +179,30 @@ public function testDenormalizeException(): void {
     $this->normalizer->denormalize($normalized, DateTimeInterface::class, NULL, []);
   }
 
+  /**
+   * Generate test data for date data providers.
+   *
+   * @return array
+   *   Test data for time formats supported by DateTimeNormalizer.
+   */
+  public static function jsonSchemaDataProvider(): array {
+    $case = function (UnitTestCase $test) {
+      $drupal_date_time = $test->prophesize(DateTimeNormalizerTestDrupalDateTime::class);
+      $drupal_date_time->setTimezone(new \DateTimeZone('Australia/Sydney'))
+        ->willReturn($drupal_date_time->reveal());
+      $drupal_date_time->format(\DateTime::RFC3339)
+        ->willReturn('1983-07-12T05:00:00-05:00');
+
+      $data = $test->prophesize(DateTimeInterface::class);
+      $data->getDateTime()
+        ->willReturn($drupal_date_time->reveal());
+      return $data->reveal();
+    };
+    return [
+      'RFC 3339' => [fn (UnitTestCase $test) => $case($test)],
+    ];
+  }
+
 }
 
 
diff --git a/core/modules/serialization/tests/src/Unit/Normalizer/MarkupNormalizerTest.php b/core/modules/serialization/tests/src/Unit/Normalizer/MarkupNormalizerTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..88505cf5c190ce8de66dc647ed63ac0ead2b3c9f
--- /dev/null
+++ b/core/modules/serialization/tests/src/Unit/Normalizer/MarkupNormalizerTest.php
@@ -0,0 +1,55 @@
+<?php
+
+declare(strict_types=1);
+
+namespace Drupal\Tests\serialization\Unit\Normalizer;
+
+use Drupal\Core\Render\Markup;
+use Drupal\Core\Template\Attribute;
+use Drupal\serialization\Normalizer\MarkupNormalizer;
+use Drupal\Tests\serialization\Traits\JsonSchemaTestTrait;
+use Drupal\Tests\UnitTestCase;
+
+/**
+ * @coversDefaultClass \Drupal\serialization\Normalizer\MarkupNormalizer
+ * @group serialization
+ */
+final class MarkupNormalizerTest extends UnitTestCase {
+
+  use JsonSchemaTestTrait;
+
+  /**
+   * The TypedDataNormalizer instance.
+   *
+   * @var \Drupal\serialization\Normalizer\TypedDataNormalizer
+   */
+  protected $normalizer;
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function setUp(): void {
+    parent::setUp();
+
+    $this->normalizer = new MarkupNormalizer();
+  }
+
+  /**
+   * Test the normalizer properly delegates schema discovery to its subject.
+   */
+  public function testDelegatedSchemaDiscovery(): void {
+    $schema = $this->normalizer->getNormalizationSchema(new Attribute(['data-test' => 'testing']));
+    $this->assertEquals('Rendered HTML element attributes', $schema['description']);
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function jsonSchemaDataProvider(): array {
+    return [
+      'markup' => [Markup::create('Generic Markup')],
+      'attribute' => [new Attribute(['data-test' => 'testing'])],
+    ];
+  }
+
+}
diff --git a/core/modules/serialization/tests/src/Unit/Normalizer/NullNormalizerTest.php b/core/modules/serialization/tests/src/Unit/Normalizer/NullNormalizerTest.php
index 288a49354578a650830f5608dd0c6a3de47e6403..1ada30d01a419bc5941b6928a94dec7a911eef67 100644
--- a/core/modules/serialization/tests/src/Unit/Normalizer/NullNormalizerTest.php
+++ b/core/modules/serialization/tests/src/Unit/Normalizer/NullNormalizerTest.php
@@ -4,7 +4,9 @@
 
 namespace Drupal\Tests\serialization\Unit\Normalizer;
 
+use Drupal\Core\TypedData\TypedDataInterface;
 use Drupal\serialization\Normalizer\NullNormalizer;
+use Drupal\Tests\serialization\Traits\JsonSchemaTestTrait;
 use Drupal\Tests\UnitTestCase;
 
 /**
@@ -13,6 +15,8 @@
  */
 class NullNormalizerTest extends UnitTestCase {
 
+  use JsonSchemaTestTrait;
+
   /**
    * The NullNormalizer instance.
    *
@@ -55,4 +59,13 @@ public function testNormalize(): void {
     $this->assertNull($this->normalizer->normalize($mock));
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  public static function jsonSchemaDataProvider(): array {
+    return [
+      'null' => [TypedDataInterface::class],
+    ];
+  }
+
 }
diff --git a/core/modules/serialization/tests/src/Unit/Normalizer/PrimitiveDataNormalizerTest.php b/core/modules/serialization/tests/src/Unit/Normalizer/PrimitiveDataNormalizerTest.php
index 3d55447e823cdb94b90dc234c5abb972794af013..4fe7f883b8bb2c6595e1895b91cb07ed67fb37ce 100644
--- a/core/modules/serialization/tests/src/Unit/Normalizer/PrimitiveDataNormalizerTest.php
+++ b/core/modules/serialization/tests/src/Unit/Normalizer/PrimitiveDataNormalizerTest.php
@@ -6,8 +6,14 @@
 
 use Drupal\Core\TypedData\DataDefinition;
 use Drupal\Core\TypedData\Plugin\DataType\BooleanData;
+use Drupal\Core\TypedData\Plugin\DataType\DecimalData;
+use Drupal\Core\TypedData\Plugin\DataType\DurationIso8601;
+use Drupal\Core\TypedData\Plugin\DataType\Email;
+use Drupal\Core\TypedData\Plugin\DataType\FloatData;
 use Drupal\Core\TypedData\Plugin\DataType\IntegerData;
 use Drupal\Core\TypedData\Plugin\DataType\StringData;
+use Drupal\Core\TypedData\Plugin\DataType\Uri;
+use Drupal\Tests\serialization\Traits\JsonSchemaTestTrait;
 use Drupal\Tests\UnitTestCase;
 use Drupal\serialization\Normalizer\PrimitiveDataNormalizer;
 
@@ -17,6 +23,8 @@
  */
 class PrimitiveDataNormalizerTest extends UnitTestCase {
 
+  use JsonSchemaTestTrait;
+
   /**
    * The TypedDataNormalizer instance.
    *
@@ -102,4 +110,30 @@ public static function dataProviderPrimitiveData() {
     return $data;
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  public static function jsonSchemaDataProvider(): array {
+    $email = new Email(DataDefinition::createFromDataType('email'));
+    $email->setValue('test@example.com');
+    $float = new FloatData(DataDefinition::createFromDataType('float'));
+    $float->setValue(9.99);
+    $uri = new Uri(DataDefinition::createFromDataType('uri'));
+    $uri->setValue('https://example.com');
+    $decimal = new DecimalData(DataDefinition::createFromDataType('decimal'));
+    $decimal->setValue('9.99');
+    // TimeSpan normalizes to an integer, however Iso8601 matches a format.
+    $duration = new DurationIso8601(DataDefinition::createFromDataType('duration_iso8601'));
+    $duration->setValue('P1D');
+
+    return [
+      'email' => [$email],
+      'float' => [$float],
+      'uri' => [$uri],
+      'decimal' => [$decimal],
+      'duration' => [$duration],
+      ...array_map(fn ($value) => [$value[0]], static::dataProviderPrimitiveData()),
+    ];
+  }
+
 }
diff --git a/core/modules/serialization/tests/src/Unit/Normalizer/TimestampNormalizerTest.php b/core/modules/serialization/tests/src/Unit/Normalizer/TimestampNormalizerTest.php
index 398e458f4eb06f100c1d223715a42436dce7e9d9..c200bbde64feba8468407e872013beb4d420497d 100644
--- a/core/modules/serialization/tests/src/Unit/Normalizer/TimestampNormalizerTest.php
+++ b/core/modules/serialization/tests/src/Unit/Normalizer/TimestampNormalizerTest.php
@@ -10,6 +10,7 @@
 use Drupal\Core\TypedData\Plugin\DataType\Timestamp;
 use Drupal\Core\TypedData\Type\DateTimeInterface;
 use Drupal\serialization\Normalizer\TimestampNormalizer;
+use Drupal\Tests\serialization\Traits\JsonSchemaTestTrait;
 use Drupal\Tests\UnitTestCase;
 use Symfony\Component\Serializer\Exception\UnexpectedValueException;
 
@@ -22,6 +23,8 @@
  */
 class TimestampNormalizerTest extends UnitTestCase {
 
+  use JsonSchemaTestTrait;
+
   /**
    * The tested data type's normalizer.
    *
@@ -132,6 +135,28 @@ public function testDenormalizeException(): void {
     $this->normalizer->denormalize($normalized, Timestamp::class, NULL, []);
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  public static function jsonSchemaDataProvider(): array {
+    $case = function (UnitTestCase $test) {
+      assert(in_array(JsonSchemaTestTrait::class, class_uses($test)));
+      $drupal_date_time = $test->doProphesize(TimestampNormalizerTestDrupalDateTime::class);
+      $drupal_date_time->setTimezone(new \DateTimeZone('UTC'))
+        ->willReturn($drupal_date_time->reveal());
+      $drupal_date_time->format(\DateTime::RFC3339)
+        ->willReturn('1983-07-12T09:05:00-05:00');
+
+      $data = $test->doProphesize(Timestamp::class);
+      $data->getDateTime()
+        ->willReturn($drupal_date_time->reveal());
+      return $data->reveal();
+    };
+    return [
+      'RFC 3339' => [fn (UnitTestCase $test) => $case($test)],
+    ];
+  }
+
 }
 
 /**
diff --git a/core/modules/text/src/TextProcessed.php b/core/modules/text/src/TextProcessed.php
index 212c11440dcd19130589765efd823ed3ed01db67..fea487051c9c8f9cd0af23595dbab9cdae7847a3 100644
--- a/core/modules/text/src/TextProcessed.php
+++ b/core/modules/text/src/TextProcessed.php
@@ -3,6 +3,7 @@
 namespace Drupal\text;
 
 use Drupal\Core\Cache\CacheableDependencyInterface;
+use Drupal\Core\Serialization\Attribute\JsonSchema;
 use Drupal\Core\TypedData\DataDefinitionInterface;
 use Drupal\Core\TypedData\TypedDataInterface;
 use Drupal\Core\TypedData\TypedData;
@@ -38,6 +39,7 @@ public function __construct(DataDefinitionInterface $definition, $name = NULL, ?
   /**
    * {@inheritdoc}
    */
+  #[JsonSchema(['type' => 'string', 'description' => 'May contain HTML markup.'])]
   public function getValue() {
     if ($this->processed !== NULL) {
       return FilteredMarkup::create($this->processed->getProcessedText());