diff --git a/composer.lock b/composer.lock
index bb5244b40b479fbfd3082c941b3a6c2b77a07279..5b6777f9a78d9a6aa86d0ec3590be123343d2ab2 100644
--- a/composer.lock
+++ b/composer.lock
@@ -452,7 +452,7 @@
             "dist": {
                 "type": "path",
                 "url": "core",
-                "reference": "34bfd0597745e88afb64bf5294f6af4975930fdc"
+                "reference": "25a2835ba98c19a6fa4d60089d813a81de36e5e5"
             },
             "require": {
                 "asm89/stack-cors": "^1.1",
@@ -670,7 +670,6 @@
                     "lib/Drupal/Core/Cache/DatabaseCacheTagsChecksum.php",
                     "lib/Drupal/Core/Database/Connection.php",
                     "lib/Drupal/Core/Database/Database.php",
-                    "lib/Drupal/Core/Database/Statement.php",
                     "lib/Drupal/Core/Database/StatementInterface.php",
                     "lib/Drupal/Core/DependencyInjection/Container.php",
                     "lib/Drupal/Core/DrupalKernel.php",
diff --git a/core/composer.json b/core/composer.json
index 71b0f6a7ca30e78280ac304cd359d9f7eb6c2895..64078e9d8aa62ef5b935edd783b845ef62188a4c 100644
--- a/core/composer.json
+++ b/core/composer.json
@@ -172,8 +172,7 @@
     "autoload": {
         "psr-4": {
             "Drupal\\Core\\": "lib/Drupal/Core",
-            "Drupal\\Component\\": "lib/Drupal/Component",
-            "Drupal\\Driver\\": "../drivers/lib/Drupal/Driver"
+            "Drupal\\Component\\": "lib/Drupal/Component"
         },
         "classmap": [
             "lib/Drupal.php",
@@ -191,7 +190,6 @@
             "lib/Drupal/Core/Cache/DatabaseCacheTagsChecksum.php",
             "lib/Drupal/Core/Database/Connection.php",
             "lib/Drupal/Core/Database/Database.php",
-            "lib/Drupal/Core/Database/Statement.php",
             "lib/Drupal/Core/Database/StatementInterface.php",
             "lib/Drupal/Core/DependencyInjection/Container.php",
             "lib/Drupal/Core/DrupalKernel.php",
diff --git a/core/lib/Drupal/Core/Database/Connection.php b/core/lib/Drupal/Core/Database/Connection.php
index cea7fc4028eb1a3fc362ef6cacce8274788417e5..5cb43e5bc47b668e9b24685a3fcf09edcd23a979 100644
--- a/core/lib/Drupal/Core/Database/Connection.php
+++ b/core/lib/Drupal/Core/Database/Connection.php
@@ -70,19 +70,6 @@ abstract class Connection {
    */
   protected $driverClasses = [];
 
-  /**
-   * The name of the Statement class for this connection.
-   *
-   * @var string|null
-   *
-   * @deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Database
-   *   drivers should use or extend StatementWrapper instead, and encapsulate
-   *   client-level statement objects.
-   *
-   * @see https://www.drupal.org/node/3177488
-   */
-  protected $statementClass = 'Drupal\Core\Database\Statement';
-
   /**
    * The name of the StatementWrapper class for this connection.
    *
@@ -99,18 +86,6 @@ abstract class Connection {
    */
   protected $transactionalDDLSupport = FALSE;
 
-  /**
-   * An index used to generate unique temporary table names.
-   *
-   * @var int
-   *
-   * @deprecated in drupal:9.3.0 and is removed from drupal:10.0.0. There is no
-   *   replacement.
-   *
-   * @see https://www.drupal.org/node/3211781
-   */
-  protected $temporaryNameIndex = 0;
-
   /**
    * The actual PDO connection.
    *
@@ -162,19 +137,6 @@ abstract class Connection {
    */
   protected $unprefixedTablesMap = [];
 
-  /**
-   * List of escaped database, table, and field names, keyed by unescaped names.
-   *
-   * @var array
-   *
-   * @deprecated in drupal:9.0.0 and is removed from drupal:10.0.0. This is no
-   *   longer used. Use \Drupal\Core\Database\Connection::$escapedTables or
-   *   \Drupal\Core\Database\Connection::$escapedFields instead.
-   *
-   * @see https://www.drupal.org/node/2986894
-   */
-  protected $escapedNames = [];
-
   /**
    * List of escaped table names, keyed by unescaped names.
    *
@@ -231,19 +193,8 @@ abstract class Connection {
    *   per-table prefixes, but it is meant for internal use only.
    */
   public function __construct(\PDO $connection, array $connection_options) {
-    if ($this->identifierQuotes === NULL) {
-      @trigger_error('In drupal:10.0.0 not setting the $identifierQuotes property in the concrete Connection class will result in an RuntimeException. See https://www.drupal.org/node/2986894', E_USER_DEPRECATED);
-      $this->identifierQuotes = ['', ''];
-    }
-
     assert(count($this->identifierQuotes) === 2 && Inspector::assertAllStrings($this->identifierQuotes), '\Drupal\Core\Database\Connection::$identifierQuotes must contain 2 string values');
 
-    // The 'transactions' option is deprecated.
-    if (isset($connection_options['transactions'])) {
-      @trigger_error('Passing a \'transactions\' connection option to ' . __METHOD__ . ' is deprecated in drupal:9.1.0 and is removed in drupal:10.0.0. All database drivers must support transactions. See https://www.drupal.org/node/2278745', E_USER_DEPRECATED);
-      unset($connection_options['transactions']);
-    }
-
     // Manage the table prefix.
     if (isset($connection_options['prefix']) && is_array($connection_options['prefix'])) {
       if (count($connection_options['prefix']) > 1) {
@@ -285,19 +236,6 @@ public function __construct(\PDO $connection, array $connection_options) {
       $connection_options['namespace'] = (new \ReflectionObject($this))->getNamespaceName();
     }
 
-    // The support for database drivers where the namespace that starts with
-    // Drupal\\Driver\\Database\\ is deprecated.
-    if (strpos($connection_options['namespace'], 'Drupal\Driver\Database') === 0) {
-      @trigger_error('Support for database drivers located in the "drivers/lib/Drupal/Driver/Database" directory is deprecated in drupal:9.1.0 and is removed in drupal:10.0.0. Contributed and custom database drivers should be provided by modules and use the namespace "Drupal\MODULE_NAME\Driver\Database\DRIVER_NAME". See https://www.drupal.org/node/3123251', E_USER_DEPRECATED);
-    }
-
-    // Set a Statement class, unless the driver opted out.
-    // @todo remove this in Drupal 10 https://www.drupal.org/node/3177490
-    if (!empty($this->statementClass)) {
-      @trigger_error('\Drupal\Core\Database\Connection::$statementClass is deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Database drivers should use or extend StatementWrapper instead, and encapsulate client-level statement objects. See https://www.drupal.org/node/3177488', E_USER_DEPRECATED);
-      $connection->setAttribute(\PDO::ATTR_STATEMENT_CLASS, [$this->statementClass, [$this]]);
-    }
-
     $this->connection = $connection;
     $this->connectionOptions = $connection_options;
   }
@@ -313,43 +251,10 @@ public function __construct(\PDO $connection, array $connection_options) {
    */
   public static function open(array &$connection_options = []) {}
 
-  /**
-   * Destroys this Connection object.
-   *
-   * PHP does not destruct an object if it is still referenced in other
-   * variables. In case of PDO database connection objects, PHP only closes the
-   * connection when the PDO object is destructed, so any references to this
-   * object may cause the number of maximum allowed connections to be exceeded.
-   *
-   * @deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Move custom
-   *   database destruction logic to __destruct().
-   *
-   * @see https://www.drupal.org/node/3142866
-   */
-  public function destroy() {
-    $backtrace = debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS, 2);
-    if ($backtrace[1]['class'] !== self::class && $backtrace[1]['function'] !== '__destruct') {
-      @trigger_error(__METHOD__ . '() is deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Move custom database destruction logic to __destruct(). See https://www.drupal.org/node/3142866', E_USER_DEPRECATED);
-      // Destroy all references to this connection by setting them to NULL.
-      // The Statement class attribute only accepts a new value that presents a
-      // proper callable, so we reset it to PDOStatement.
-      // @todo remove this in Drupal 10 https://www.drupal.org/node/3177490
-      if (!empty($this->statementClass)) {
-        $this->connection->setAttribute(\PDO::ATTR_STATEMENT_CLASS, ['PDOStatement', []]);
-      }
-      $this->schema = NULL;
-    }
-  }
-
   /**
    * Ensures that the PDO connection can be garbage collected.
    */
   public function __destruct() {
-    // Call the ::destroy method to provide a BC layer.
-    // @todo https://www.drupal.org/project/drupal/issues/3153864 Remove this
-    //   call in Drupal 10 as the logic in the destroy() method is no longer
-    //   required now we implement a proper destructor.
-    $this->destroy();
     // Ensure that the circular reference caused by Connection::__construct()
     // using $this in the call to set the statement class can be garbage
     // collected.
@@ -386,11 +291,6 @@ public function __destruct() {
    *   - Database::RETURN_NULL: Do not return anything, as there is no
    *     meaningful value to return. That is the case for INSERT queries on
    *     tables that do not contain a serial column.
-   * - throw_exception: (deprecated) By default, the database system will catch
-   *   any errors on a query as an Exception, log it, and then rethrow it so
-   *   that code further up the call chain can take an appropriate action. To
-   *   suppress that behavior and simply return NULL on failure, set this
-   *   option to FALSE.
    * - allow_delimiter_in_query: By default, queries which have the ; delimiter
    *   any place in them will cause an exception. This reduces the chance of SQL
    *   injection attacks that terminate the original query and add one or more
@@ -621,24 +521,11 @@ public function prepareStatement(string $query, array $options, bool $allow_row_
 
     try {
       $query = $this->preprocessStatement($query, $options);
-
-      // @todo in Drupal 10, only return the StatementWrapper.
-      // @see https://www.drupal.org/node/3177490
-      $statement = $this->statementWrapperClass ?
-        new $this->statementWrapperClass($this, $this->connection, $query, $options['pdo'] ?? [], $allow_row_count) :
-        $this->connection->prepare($query, $options['pdo'] ?? []);
+      return new $this->statementWrapperClass($this, $this->connection, $query, $options['pdo'] ?? [], $allow_row_count);
     }
     catch (\Exception $e) {
       $this->exceptionHandler()->handleStatementException($e, $query, $options);
     }
-    // BC layer: $options['throw_exception'] = FALSE or a \PDO::prepare() call
-    // returning false would lead to returning a value that fails the return
-    // typehint. Throw an exception in that case.
-    // @todo in Drupal 10, remove the check.
-    if (!isset($statement) || !$statement instanceof StatementInterface) {
-      throw new DatabaseExceptionWrapper("Statement preparation failure for query: $query");
-    }
-    return $statement;
   }
 
   /**
@@ -687,33 +574,6 @@ protected function preprocessStatement(string $query, array $options): string {
     return $query;
   }
 
-  /**
-   * Prepares a query string and returns the prepared statement.
-   *
-   * This method caches prepared statements, reusing them when possible. It also
-   * prefixes tables names enclosed in curly-braces and, optionally, quotes
-   * identifiers enclosed in square brackets.
-   *
-   * @param $query
-   *   The query string as SQL, with curly-braces surrounding the
-   *   table names.
-   * @param bool $quote_identifiers
-   *   (optional) Quote any identifiers enclosed in square brackets. Defaults to
-   *   TRUE.
-   *
-   * @return \Drupal\Core\Database\StatementInterface
-   *   A PDO prepared statement ready for its execute() method.
-   *
-   * @deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Use
-   *   ::prepareStatement instead.
-   *
-   * @see https://www.drupal.org/node/3137786
-   */
-  public function prepareQuery($query, $quote_identifiers = TRUE) {
-    @trigger_error('Connection::prepareQuery() is deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Use ::prepareStatement() instead. See https://www.drupal.org/node/3137786', E_USER_DEPRECATED);
-    return $this->prepareStatement($query, ['allow_square_brackets' => !$quote_identifiers]);
-  }
-
   /**
    * Tells this connection object what its target value is.
    *
@@ -877,16 +737,9 @@ protected function filterComment($comment = '') {
    * query. All queries executed by Drupal are executed as PDO prepared
    * statements.
    *
-   * @param string|\Drupal\Core\Database\StatementInterface|\PDOStatement $query
+   * @param string $query
    *   The query to execute. This is a string containing an SQL query with
    *   placeholders.
-   *   (deprecated) An already-prepared instance of StatementInterface or of
-   *   \PDOStatement may also be passed in order to allow calling code to
-   *   manually bind variables to a query. In such cases, the content of the
-   *   $args array will be ignored.
-   *   It is extremely rare that module code will need to pass a statement
-   *   object to this method. It is used primarily for database drivers for
-   *   databases that require special LOB field handling.
    * @param array $args
    *   The associative array of arguments for the prepared statement.
    * @param array $options
@@ -915,6 +768,8 @@ protected function filterComment($comment = '') {
    * @see \Drupal\Core\Database\Connection::defaultOptions()
    */
   public function query($query, array $args = [], $options = []) {
+    assert(is_string($query), 'The \'$query\' argument to ' . __METHOD__ . '() must be a string');
+
     // Use default values if not already set.
     $options += $this->defaultOptions();
 
@@ -924,32 +779,11 @@ public function query($query, array $args = [], $options = []) {
 
     assert(!isset($options['target']), 'Passing "target" option to query() has no effect. See https://www.drupal.org/node/2993033');
 
-    // We allow either a pre-bound statement object (deprecated) or a literal
-    // string. In either case, we want to end up with an executed statement
-    // object, which we pass to StatementInterface::execute.
-    if (is_string($query)) {
-      $this->expandArguments($query, $args);
-      $stmt = $this->prepareStatement($query, $options);
-    }
-    elseif ($query instanceof StatementInterface) {
-      @trigger_error('Passing a StatementInterface object as a $query argument to ' . __METHOD__ . ' is deprecated in drupal:9.2.0 and is removed in drupal:10.0.0. Call the execute method from the StatementInterface object directly instead. See https://www.drupal.org/node/3154439', E_USER_DEPRECATED);
-      $stmt = $query;
-    }
-    elseif ($query instanceof \PDOStatement) {
-      @trigger_error('Passing a \\PDOStatement object as a $query argument to ' . __METHOD__ . ' is deprecated in drupal:9.2.0 and is removed in drupal:10.0.0. Call the execute method from the StatementInterface object directly instead. See https://www.drupal.org/node/3154439', E_USER_DEPRECATED);
-      $stmt = $query;
-    }
+    $this->expandArguments($query, $args);
+    $stmt = $this->prepareStatement($query, $options);
 
     try {
-      if (is_string($query)) {
-        $stmt->execute($args, $options);
-      }
-      elseif ($query instanceof StatementInterface) {
-        $stmt->execute(NULL, $options);
-      }
-      elseif ($query instanceof \PDOStatement) {
-        $stmt->execute();
-      }
+      $stmt->execute($args, $options);
 
       // Depending on the type of query we may need to return a different value.
       // See DatabaseConnection::defaultOptions() for a description of each
@@ -978,74 +812,8 @@ public function query($query, array $args = [], $options = []) {
       }
     }
     catch (\Exception $e) {
-      // Most database drivers will return NULL here, but some of them
-      // (e.g. the SQLite driver) may need to re-run the query, so the return
-      // value will be the same as for static::query().
-      if (is_string($query)) {
-        return $this->exceptionHandler()->handleExecutionException($e, $stmt, $args, $options);
-      }
-      else {
-        return $this->handleQueryException($e, $query, $args, $options);
-      }
-    }
-  }
-
-  /**
-   * Wraps and re-throws any PDO exception thrown by static::query().
-   *
-   * @param \PDOException $e
-   *   The exception thrown by static::query().
-   * @param $query
-   *   The query executed by static::query().
-   * @param array $args
-   *   An array of arguments for the prepared statement.
-   * @param array $options
-   *   An associative array of options to control how the query is run.
-   *
-   * @return \Drupal\Core\Database\StatementInterface|int|null
-   *   Most database drivers will return NULL when a PDO exception is thrown for
-   *   a query, but some of them may need to re-run the query, so they can also
-   *   return a \Drupal\Core\Database\StatementInterface object or an integer.
-   *
-   * @throws \Drupal\Core\Database\DatabaseExceptionWrapper
-   * @throws \Drupal\Core\Database\IntegrityConstraintViolationException
-   *
-   * @deprecated in drupal:9.2.0 and is removed from drupal:10.0.0. Get a
-   *   handler through $this->exceptionHandler() instead, and use one of its
-   *   methods.
-   *
-   * @see https://www.drupal.org/node/3187222
-   */
-  protected function handleQueryException(\PDOException $e, $query, array $args = [], $options = []) {
-    @trigger_error('Connection::handleQueryException() is deprecated in drupal:9.2.0 and is removed in drupal:10.0.0. Get a handler through $this->exceptionHandler() instead, and use one of its methods. See https://www.drupal.org/node/3187222', E_USER_DEPRECATED);
-    if ($options['throw_exception'] ?? TRUE) {
-      // Wrap the exception in another exception, because PHP does not allow
-      // overriding Exception::getMessage(). Its message is the extra database
-      // debug information.
-      // @todo in Drupal 10, remove checking if $query is a statement object.
-      // @see https://www.drupal.org/node/3154439
-      if ($query instanceof StatementInterface) {
-        $query_string = $query->getQueryString();
-      }
-      elseif ($query instanceof \PDOStatement) {
-        $query_string = $query->queryString;
-      }
-      else {
-        $query_string = $query;
-      }
-      $message = $e->getMessage() . ": " . $query_string . "; " . print_r($args, TRUE);
-      // Match all SQLSTATE 23xxx errors.
-      if (substr($e->getCode(), -6, -3) == '23') {
-        $exception = new IntegrityConstraintViolationException($message, $e->getCode(), $e);
-      }
-      else {
-        $exception = new DatabaseExceptionWrapper($message, 0, $e);
-      }
-
-      throw $exception;
+      $this->exceptionHandler()->handleExecutionException($e, $stmt, $args, $options);
     }
-
-    return NULL;
   }
 
   /**
@@ -1214,9 +982,7 @@ public function exceptionHandler() {
    * @see \Drupal\Core\Database\Query\Select
    */
   public function select($table, $alias = NULL, array $options = []) {
-    if (!is_null($alias) && !is_string($alias)) {
-      @trigger_error('Passing a non-string \'alias\' argument to ' . __METHOD__ . '() is deprecated in drupal:9.3.0 and will be required in drupal:10.0.0. Refactor your calling code. See https://www.drupal.org/project/drupal/issues/3216552', E_USER_DEPRECATED);
-    }
+    assert(is_string($alias) || $alias === NULL, 'The \'$alias\' argument to ' . __METHOD__ . '() must be a string or NULL');
     $class = $this->getDriverClass('Select');
     return new $class($this, $table, $alias, $options);
   }
@@ -1404,7 +1170,7 @@ public function condition($conjunction) {
     // Creating an instance of the class Drupal\Core\Database\Query\Condition
     // should only be created from the database layer. This will allow database
     // drivers to override the default Condition class.
-    return new $class($conjunction, FALSE);
+    return new $class($conjunction);
   }
 
   /**
@@ -1776,54 +1542,6 @@ protected function doCommit() {
    */
   abstract public function queryRange($query, $from, $count, array $args = [], array $options = []);
 
-  /**
-   * Generates a temporary table name.
-   *
-   * @return string
-   *   A table name.
-   *
-   * @deprecated in drupal:9.3.0 and is removed from drupal:10.0.0. There is no
-   *   replacement.
-   *
-   * @see https://www.drupal.org/node/3211781
-   */
-  protected function generateTemporaryTableName() {
-    @trigger_error('Connection::generateTemporaryTableName() is deprecated in drupal:9.3.0 and is removed from drupal:10.0.0. There is no replacement. See https://www.drupal.org/node/3211781', E_USER_DEPRECATED);
-    return "db_temporary_" . $this->temporaryNameIndex++;
-  }
-
-  /**
-   * Runs a SELECT query and stores its results in a temporary table.
-   *
-   * Use this as a substitute for ->query() when the results need to stored
-   * in a temporary table. Temporary tables exist for the duration of the page
-   * request. User-supplied arguments to the query should be passed in as
-   * separate parameters so that they can be properly escaped to avoid SQL
-   * injection attacks.
-   *
-   * Note that if you need to know how many results were returned, you should do
-   * a SELECT COUNT(*) on the temporary table afterwards.
-   *
-   * @param string $query
-   *   A string containing a normal SELECT SQL query.
-   * @param array $args
-   *   (optional) An array of values to substitute into the query at placeholder
-   *   markers.
-   * @param array $options
-   *   (optional) An associative array of options to control how the query is
-   *   run. See the documentation for DatabaseConnection::defaultOptions() for
-   *   details.
-   *
-   * @return string
-   *   The name of the temporary table.
-   *
-   * @deprecated in drupal:9.3.0 and is removed from drupal:10.0.0. There is no
-   *   replacement.
-   *
-   * @see https://www.drupal.org/node/3211781
-   */
-  abstract public function queryTemporary($query, array $args = [], array $options = []);
-
   /**
    * Returns the type of database driver.
    *
@@ -1851,22 +1569,6 @@ public function clientVersion() {
     return $this->connection->getAttribute(\PDO::ATTR_CLIENT_VERSION);
   }
 
-  /**
-   * Determines if this driver supports transactions.
-   *
-   * @return bool
-   *   TRUE if this connection supports transactions, FALSE otherwise.
-   *
-   * @deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. All database
-   * drivers must support transactions.
-   *
-   * @see https://www.drupal.org/node/2278745
-   */
-  public function supportsTransactions() {
-    @trigger_error(__METHOD__ . ' is deprecated in drupal:9.1.0 and is removed in drupal:10.0.0. All database drivers must support transactions. See https://www.drupal.org/node/2278745', E_USER_DEPRECATED);
-    return TRUE;
-  }
-
   /**
    * Determines if this driver supports transactional DDL.
    *
@@ -1949,46 +1651,6 @@ public function commit() {
    */
   abstract public function nextId($existing_id = 0);
 
-  /**
-   * Prepares a statement for execution and returns a statement object.
-   *
-   * Emulated prepared statements do not communicate with the database server so
-   * this method does not check the statement.
-   *
-   * @param string $statement
-   *   This must be a valid SQL statement for the target database server.
-   * @param array $driver_options
-   *   (optional) This array holds one or more key=>value pairs to set
-   *   attribute values for the PDOStatement object that this method returns.
-   *   You would most commonly use this to set the \PDO::ATTR_CURSOR value to
-   *   \PDO::CURSOR_SCROLL to request a scrollable cursor. Some drivers have
-   *   driver specific options that may be set at prepare-time. Defaults to an
-   *   empty array.
-   *
-   * @return \PDOStatement|false
-   *   If the database server successfully prepares the statement, returns a
-   *   \PDOStatement object.
-   *   If the database server cannot successfully prepare the statement  returns
-   *   FALSE or emits \PDOException (depending on error handling).
-   *
-   * @throws \PDOException
-   *
-   * @see https://www.php.net/manual/en/pdo.prepare.php
-   *
-   * @deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Database
-   *   drivers should instantiate \PDOStatement objects by calling
-   *   \PDO::prepare in their Connection::prepareStatement method instead.
-   *   \PDO::prepare should not be called outside of driver code.
-   *
-   * @see https://www.drupal.org/node/3137786
-   */
-  public function prepare($statement, array $driver_options = []) {
-    @trigger_error('Connection::prepare() is deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Database drivers should instantiate \PDOStatement objects by calling \PDO::prepare in their Connection::prepareStatement method instead. \PDO::prepare should not be called outside of driver code. See https://www.drupal.org/node/3137786', E_USER_DEPRECATED);
-    return $this->statementWrapperClass ?
-      (new $this->statementWrapperClass($this, $this->connection, $statement, $driver_options))->getClientStatement() :
-      $this->connection->prepare($statement, $driver_options);
-  }
-
   /**
    * Quotes a string for use in a query.
    *
diff --git a/core/lib/Drupal/Core/Database/Database.php b/core/lib/Drupal/Core/Database/Database.php
index 79b681dc883a0f168ab4d7f39e4f5e1be05c59d4..bb24b63e63660e24e02bceafd1cab91b5568095b 100644
--- a/core/lib/Drupal/Core/Database/Database.php
+++ b/core/lib/Drupal/Core/Database/Database.php
@@ -644,31 +644,6 @@ public static function getConnectionInfoAsUrl($key = 'default') {
     return $connection_class::createUrlFromConnectionOptions($db_info['default']);
   }
 
-  /**
-   * Gets the PHP namespace of a database driver from the connection info.
-   *
-   * @param array $connection_info
-   *   The database connection information, as defined in settings.php. The
-   *   structure of this array depends on the database driver it is connecting
-   *   to.
-   *
-   * @return string
-   *   The PHP namespace of the driver's database.
-   *
-   * @deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. There is no
-   *   replacement as $connection_info['namespace'] is always set.
-   *
-   * @see https://www.drupal.org/node/3127769
-   */
-  protected static function getDatabaseDriverNamespace(array $connection_info) {
-    @trigger_error(__METHOD__ . " is deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. There is no replacement as \$connection_info['namespace'] is always set. See https://www.drupal.org/node/3127769.", E_USER_DEPRECATED);
-    if (isset($connection_info['namespace'])) {
-      return $connection_info['namespace'];
-    }
-    // Fallback for when the namespace is not provided in settings.php.
-    return 'Drupal\\' . $connection_info['driver'] . '\\Driver\\Database\\' . $connection_info['driver'];
-  }
-
   /**
    * Checks whether a namespace is within the namespace of a Drupal module.
    *
diff --git a/core/lib/Drupal/Core/Database/ExceptionHandler.php b/core/lib/Drupal/Core/Database/ExceptionHandler.php
index dcbbd5dd026b6edc10415b7929777415ae720f81..a35e0c0239ae8c9563d5ccd25537bb6b1c2b9932 100644
--- a/core/lib/Drupal/Core/Database/ExceptionHandler.php
+++ b/core/lib/Drupal/Core/Database/ExceptionHandler.php
@@ -25,13 +25,6 @@ class ExceptionHandler {
    * @throws \Drupal\Core\Database\DatabaseExceptionWrapper
    */
   public function handleStatementException(\Exception $exception, string $sql, array $options = []): void {
-    if (array_key_exists('throw_exception', $options)) {
-      @trigger_error('Passing a \'throw_exception\' option to ' . __METHOD__ . ' is deprecated in drupal:9.2.0 and is removed in drupal:10.0.0. Always catch exceptions. See https://www.drupal.org/node/3201187', E_USER_DEPRECATED);
-      if (!($options['throw_exception'])) {
-        return;
-      }
-    }
-
     if ($exception instanceof \PDOException) {
       // Wrap the exception in another exception, because PHP does not allow
       // overriding Exception::getMessage(). Its message is the extra database
@@ -60,13 +53,6 @@ public function handleStatementException(\Exception $exception, string $sql, arr
    * @throws \Drupal\Core\Database\IntegrityConstraintViolationException
    */
   public function handleExecutionException(\Exception $exception, StatementInterface $statement, array $arguments = [], array $options = []): void {
-    if (array_key_exists('throw_exception', $options)) {
-      @trigger_error('Passing a \'throw_exception\' option to ' . __METHOD__ . ' is deprecated in drupal:9.2.0 and is removed in drupal:10.0.0. Always catch exceptions. See https://www.drupal.org/node/3201187', E_USER_DEPRECATED);
-      if (!($options['throw_exception'])) {
-        return;
-      }
-    }
-
     if ($exception instanceof \PDOException) {
       // Wrap the exception in another exception, because PHP does not allow
       // overriding Exception::getMessage(). Its message is the extra database
diff --git a/core/lib/Drupal/Core/Database/Log.php b/core/lib/Drupal/Core/Database/Log.php
index 6499a2b6703f358a27bfec4d702cde7fdb282d16..5fd10b18ac30c2e9ecd20fd24d7be94831a2f261 100644
--- a/core/lib/Drupal/Core/Database/Log.php
+++ b/core/lib/Drupal/Core/Database/Log.php
@@ -116,13 +116,10 @@ public function end($logging_key) {
    */
   public function log(StatementInterface $statement, $args, $time, float $start = NULL) {
     foreach (array_keys($this->queryLog) as $key) {
-      // @todo Remove the method_exists check for getConnectionTarget in
-      //   Drupal 10.
-      // @see https://www.drupal.org/project/drupal/issues/3210310
       $this->queryLog[$key][] = [
         'query' => $statement->getQueryString(),
         'args' => $args,
-        'target' => method_exists($statement, 'getConnectionTarget') ? $statement->getConnectionTarget() : $statement->dbh->getTarget(),
+        'target' => $statement->getConnectionTarget(),
         'caller' => $this->findCaller(),
         'time' => $time,
         'start' => $start,
diff --git a/core/lib/Drupal/Core/Database/Query/Condition.php b/core/lib/Drupal/Core/Database/Query/Condition.php
index 80198371bb28eca15c834468b3a1e41a37dc4ba8..ad64622303ddc6aa1b930e9c57a4ca6b23f69169 100644
--- a/core/lib/Drupal/Core/Database/Query/Condition.php
+++ b/core/lib/Drupal/Core/Database/Query/Condition.php
@@ -78,19 +78,8 @@ class Condition implements ConditionInterface, \Countable {
    *
    * @param string $conjunction
    *   The operator to use to combine conditions: 'AND' or 'OR'.
-   * @param bool $trigger_deprecation
-   *   If TRUE then trigger the deprecation warning.
-   *
-   * @deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Creating an
-   *   instance of this class is deprecated.
-   *
-   * @see https://www.drupal.org/node/3159568
    */
-  public function __construct($conjunction, $trigger_deprecation = TRUE) {
-    if ($trigger_deprecation) {
-      @trigger_error('Creating an instance of this class is deprecated in drupal:9.1.0 and is removed in drupal:10.0.0. Use Database::getConnection()->condition() instead. See https://www.drupal.org/node/3159568', E_USER_DEPRECATED);
-    }
-
+  public function __construct($conjunction) {
     $this->conditions['#conjunction'] = $conjunction;
   }
 
diff --git a/core/lib/Drupal/Core/Database/Query/Merge.php b/core/lib/Drupal/Core/Database/Query/Merge.php
index fffca6ce9f74d8befb5a9611d5c346fba1683944..a6641820744a2c54a3da4aad13b5dd4e575e0fa0 100644
--- a/core/lib/Drupal/Core/Database/Query/Merge.php
+++ b/core/lib/Drupal/Core/Database/Query/Merge.php
@@ -362,63 +362,50 @@ public function __toString() {
    *     and an INSERT query is executed.
    *   - Merge::STATUS_UPDATE: If the entry already exists,
    *     and an UPDATE query is executed.
-   *   - NULL: (deprecated) If there is a problem and
-   *     queryOptions['throw_exception'] is FALSE.
    *
    * @throws \Drupal\Core\Database\Query\InvalidMergeQueryException
    *   When there are no conditions found to merge.
    */
   public function execute() {
+    if (!count($this->condition)) {
+      throw new InvalidMergeQueryException('Invalid merge query: no conditions');
+    }
 
-    try {
-      if (!count($this->condition)) {
-        throw new InvalidMergeQueryException('Invalid merge query: no conditions');
-      }
-      $select = $this->connection->select($this->conditionTable)
-        ->condition($this->condition);
-      $select->addExpression('1');
-      if (!$select->execute()->fetchField()) {
-        try {
-          $insert = $this->connection->insert($this->table)->fields($this->insertFields);
-          if ($this->defaultFields) {
-            $insert->useDefaults($this->defaultFields);
-          }
-          $insert->execute();
-          return self::STATUS_INSERT;
-        }
-        catch (IntegrityConstraintViolationException $e) {
-          // The insert query failed, maybe it's because a racing insert query
-          // beat us in inserting the same row. Retry the select query, if it
-          // returns a row, ignore the error and continue with the update
-          // query below.
-          if (!$select->execute()->fetchField()) {
-            throw $e;
-          }
+    $select = $this->connection->select($this->conditionTable)
+      ->condition($this->condition);
+    $select->addExpression('1');
+
+    if (!$select->execute()->fetchField()) {
+      try {
+        $insert = $this->connection->insert($this->table)->fields($this->insertFields);
+        if ($this->defaultFields) {
+          $insert->useDefaults($this->defaultFields);
         }
+        $insert->execute();
+        return self::STATUS_INSERT;
       }
-      if ($this->needsUpdate) {
-        $update = $this->connection->update($this->table)
-          ->fields($this->updateFields)
-          ->condition($this->condition);
-        if ($this->expressionFields) {
-          foreach ($this->expressionFields as $field => $data) {
-            $update->expression($field, $data['expression'], $data['arguments']);
-          }
+      catch (IntegrityConstraintViolationException $e) {
+        // The insert query failed, maybe it's because a racing insert query
+        // beat us in inserting the same row. Retry the select query, if it
+        // returns a row, ignore the error and continue with the update
+        // query below.
+        if (!$select->execute()->fetchField()) {
+          throw $e;
         }
-        $update->execute();
-        return self::STATUS_UPDATE;
       }
     }
-    catch (\Exception $e) {
-      // @todo 'throw_exception' option is deprecated. Remove in D10.
-      // @see https://www.drupal.org/project/drupal/issues/3210310
-      if (array_key_exists('throw_exception', $this->queryOptions)) {
-        @trigger_error('Passing a \'throw_exception\' option to ' . __METHOD__ . ' is deprecated in drupal:9.2.0 and is removed in drupal:10.0.0. Always catch exceptions. See https://www.drupal.org/node/3201187', E_USER_DEPRECATED);
-        if (!($this->queryOptions['throw_exception'])) {
-          return NULL;
+
+    if ($this->needsUpdate) {
+      $update = $this->connection->update($this->table)
+        ->fields($this->updateFields)
+        ->condition($this->condition);
+      if ($this->expressionFields) {
+        foreach ($this->expressionFields as $field => $data) {
+          $update->expression($field, $data['expression'], $data['arguments']);
         }
       }
-      throw $e;
+      $update->execute();
+      return self::STATUS_UPDATE;
     }
   }
 
diff --git a/core/lib/Drupal/Core/Database/Query/PagerSelectExtender.php b/core/lib/Drupal/Core/Database/Query/PagerSelectExtender.php
index 1a53ec860143e8b3836f4d08830da7b29aa3111b..3620d9b5d17c9d42b1e368db07be7fc52f9ac1fb 100644
--- a/core/lib/Drupal/Core/Database/Query/PagerSelectExtender.php
+++ b/core/lib/Drupal/Core/Database/Query/PagerSelectExtender.php
@@ -16,18 +16,6 @@
  */
 class PagerSelectExtender extends SelectExtender {
 
-  /**
-   * The highest element we've autogenerated so far.
-   *
-   * @var int
-   *
-   * @deprecated in drupal:9.2.0 and is removed from drupal:10.0.0. Use
-   *   \Drupal::service('pager.manager')->getMaxPagerElementId() instead.
-   *
-   * @see https://www.drupal.org/node/3194594
-   */
-  public static $maxElement = 0;
-
   /**
    * The number of elements per page to allow.
    *
diff --git a/core/lib/Drupal/Core/Database/Statement.php b/core/lib/Drupal/Core/Database/Statement.php
deleted file mode 100644
index fb9a1b8eb66bcf8eb040ec574b8bb87745ca68b6..0000000000000000000000000000000000000000
--- a/core/lib/Drupal/Core/Database/Statement.php
+++ /dev/null
@@ -1,204 +0,0 @@
-<?php
-
-namespace Drupal\Core\Database;
-
-@trigger_error('\Drupal\Core\Database\Statement is deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Database drivers should use or extend StatementWrapper instead, and encapsulate client-level statement objects. See https://www.drupal.org/node/3177488', E_USER_DEPRECATED);
-
-/**
- * Default implementation of StatementInterface.
- *
- * \PDO allows us to extend the \PDOStatement class to provide additional
- * functionality beyond that offered by default. We do need extra
- * functionality. By default, this class is not driver-specific. If a given
- * driver needs to set a custom statement class, it may do so in its
- * constructor.
- *
- * @see http://php.net/pdostatement
- *
- * @deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Database
- *   drivers should use or extend StatementWrapper instead, and encapsulate
- *   client-level statement objects.
- *
- * @see https://www.drupal.org/node/3177488
- */
-class Statement extends \PDOStatement implements StatementInterface {
-
-  /**
-   * Reference to the database connection object for this statement.
-   *
-   * The name $dbh is inherited from \PDOStatement.
-   *
-   * @var \Drupal\Core\Database\Connection
-   */
-  public $dbh;
-
-  /**
-   * Is rowCount() execution allowed.
-   *
-   * @var bool
-   */
-  public $allowRowCount = FALSE;
-
-  protected function __construct(Connection $dbh) {
-    $this->dbh = $dbh;
-    $this->setFetchMode(\PDO::FETCH_OBJ);
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getConnectionTarget(): string {
-    return $this->connection->getTarget();
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function execute($args = [], $options = []) {
-    if (isset($options['fetch'])) {
-      if (is_string($options['fetch'])) {
-        // \PDO::FETCH_PROPS_LATE tells __construct() to run before properties
-        // are added to the object.
-        $this->setFetchMode(\PDO::FETCH_CLASS | \PDO::FETCH_PROPS_LATE, $options['fetch']);
-      }
-      else {
-        $this->setFetchMode($options['fetch']);
-      }
-    }
-
-    $logger = $this->dbh->getLogger();
-    if (!empty($logger)) {
-      $query_start = microtime(TRUE);
-    }
-
-    $return = parent::execute($args);
-
-    if (!empty($logger)) {
-      $query_end = microtime(TRUE);
-      $logger->log($this, $args, $query_end - $query_start, $query_start);
-    }
-
-    return $return;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getQueryString() {
-    return $this->queryString;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function fetchCol($index = 0) {
-    return $this->fetchAll(\PDO::FETCH_COLUMN, $index);
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function fetchAllAssoc($key, $fetch = NULL) {
-    $return = [];
-    if (isset($fetch)) {
-      if (is_string($fetch)) {
-        $this->setFetchMode(\PDO::FETCH_CLASS, $fetch);
-      }
-      else {
-        $this->setFetchMode($fetch);
-      }
-    }
-
-    foreach ($this as $record) {
-      $record_key = is_object($record) ? $record->$key : $record[$key];
-      $return[$record_key] = $record;
-    }
-
-    return $return;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function fetchAllKeyed($key_index = 0, $value_index = 1) {
-    $return = [];
-    $this->setFetchMode(\PDO::FETCH_NUM);
-    foreach ($this as $record) {
-      $return[$record[$key_index]] = $record[$value_index];
-    }
-    return $return;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function fetchField($index = 0) {
-    // Call \PDOStatement::fetchColumn to fetch the field.
-    return $this->fetchColumn($index);
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function fetchAssoc() {
-    // Call \PDOStatement::fetch to fetch the row.
-    return $this->fetch(\PDO::FETCH_ASSOC);
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function rowCount() {
-    // SELECT query should not use the method.
-    if ($this->allowRowCount) {
-      return parent::rowCount();
-    }
-    else {
-      throw new RowCountException();
-    }
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function setFetchMode($mode, $a1 = NULL, $a2 = []) {
-    // Call \PDOStatement::setFetchMode to set fetch mode.
-    // \PDOStatement is picky about the number of arguments in some cases so we
-    // need to be pass the exact number of arguments we where given.
-    switch (func_num_args()) {
-      case 1:
-        return parent::setFetchMode($mode);
-
-      case 2:
-        return parent::setFetchMode($mode, $a1);
-
-      case 3:
-      default:
-        return parent::setFetchMode($mode, $a1, $a2);
-    }
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function fetchAll($mode = NULL, $column_index = NULL, $constructor_arguments = NULL) {
-    // Call \PDOStatement::fetchAll to fetch all rows.
-    // \PDOStatement is picky about the number of arguments in some cases so we
-    // need to be pass the exact number of arguments we where given.
-    switch (func_num_args()) {
-      case 0:
-        return parent::fetchAll();
-
-      case 1:
-        return parent::fetchAll($mode);
-
-      case 2:
-        return parent::fetchAll($mode, $column_index);
-
-      case 3:
-      default:
-        return parent::fetchAll($mode, $column_index, $constructor_arguments);
-    }
-  }
-
-}
diff --git a/core/lib/Drupal/Core/Database/StatementEmpty.php b/core/lib/Drupal/Core/Database/StatementEmpty.php
deleted file mode 100644
index fe173ecea3e599c6819746a313d8639de3f2140e..0000000000000000000000000000000000000000
--- a/core/lib/Drupal/Core/Database/StatementEmpty.php
+++ /dev/null
@@ -1,152 +0,0 @@
-<?php
-
-namespace Drupal\Core\Database;
-
-@trigger_error('\Drupal\Core\Database\StatementEmpty is deprecated in drupal:9.2.0 and is removed from drupal:10.0.0. There is no replacement. Use mocked StatementInterface classes in tests if needed. See https://www.drupal.org/node/3201283', E_USER_DEPRECATED);
-
-/**
- * Empty implementation of a database statement.
- *
- * This class satisfies the requirements of being a database statement/result
- * object, but does not actually contain data.  It is useful when developers
- * need to safely return an "empty" result set without connecting to an actual
- * database.  Calling code can then treat it the same as if it were an actual
- * result set that happens to contain no records.
- *
- * @see \Drupal\search\SearchQuery
- *
- * @deprecated in drupal:9.2.0 and is removed from drupal:10.0.0. There is no
- *   replacement. Use mocked StatementInterface classes in tests if needed.
- *
- * @see https://www.drupal.org/node/1234567
- */
-class StatementEmpty implements \Iterator, StatementInterface {
-
-  /**
-   * Is rowCount() execution allowed.
-   *
-   * @var bool
-   */
-  public $allowRowCount = FALSE;
-
-  /**
-   * {@inheritdoc}
-   */
-  public function execute($args = [], $options = []) {
-    return FALSE;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getQueryString() {
-    return '';
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function rowCount() {
-    if ($this->allowRowCount) {
-      return 0;
-    }
-    throw new RowCountException();
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function setFetchMode($mode, $a1 = NULL, $a2 = []) {}
-
-  /**
-   * {@inheritdoc}
-   */
-  public function fetch($mode = NULL, $cursor_orientation = NULL, $cursor_offset = NULL) {
-    return NULL;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function fetchField($index = 0) {
-    return NULL;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function fetchObject(string $class_name = NULL, array $constructor_arguments = NULL) {
-    return NULL;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function fetchAssoc() {
-    return NULL;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function fetchAll($mode = NULL, $column_index = NULL, $constructor_arguments = NULL) {
-    return [];
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function fetchCol($index = 0) {
-    return [];
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function fetchAllKeyed($key_index = 0, $value_index = 1) {
-    return [];
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function fetchAllAssoc($key, $fetch = NULL) {
-    return [];
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function current() {
-    return NULL;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function key() {
-    return NULL;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function rewind() {
-    // Nothing to do: our DatabaseStatement can't be rewound.
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function next() {
-    // Do nothing, since this is an always-empty implementation.
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function valid() {
-    return FALSE;
-  }
-
-}
diff --git a/core/lib/Drupal/Core/Database/StatementInterface.php b/core/lib/Drupal/Core/Database/StatementInterface.php
index 0bb030d0172ea4387a18bb6ca39aa778e0e47291..ad0f741da8271e86e0e2c54b96d737d488fbe0df 100644
--- a/core/lib/Drupal/Core/Database/StatementInterface.php
+++ b/core/lib/Drupal/Core/Database/StatementInterface.php
@@ -50,9 +50,7 @@ public function getQueryString();
    * @return string
    *   The target connection string of this statement.
    */
-  // @todo Include this method in the interface in Drupal 10.
-  // @see https://www.drupal.org/project/drupal/issues/3210310
-  // public function getConnectionTarget(): string;
+  public function getConnectionTarget(): string;
 
   /**
    * Returns the number of rows affected by the last SQL statement.
diff --git a/core/lib/Drupal/Core/Database/StatementPrefetch.php b/core/lib/Drupal/Core/Database/StatementPrefetch.php
index 267b6a85d55f72928b3df680379a105e0e761a61..899996f2222d873efb016a6c586ee2a2454fc34e 100644
--- a/core/lib/Drupal/Core/Database/StatementPrefetch.php
+++ b/core/lib/Drupal/Core/Database/StatementPrefetch.php
@@ -148,36 +148,6 @@ public function __construct(\PDO $pdo_connection, Connection $connection, $query
     $this->rowCountEnabled = $row_count_enabled;
   }
 
-  /**
-   * Implements the magic __get() method.
-   *
-   * @todo Remove the method before Drupal 10.
-   * @see https://www.drupal.org/i/3210310
-   */
-  public function __get($name) {
-    if ($name === 'dbh') {
-      @trigger_error(__CLASS__ . '::$dbh should not be accessed in drupal:9.3.0 and will error in drupal:10.0.0. Use $this->connection instead. See https://www.drupal.org/node/3186368', E_USER_DEPRECATED);
-      return $this->connection;
-    }
-    if ($name === 'allowRowCount') {
-      @trigger_error(__CLASS__ . '::$allowRowCount should not be accessed in drupal:9.3.0 and will error in drupal:10.0.0. Use $this->rowCountEnabled instead. See https://www.drupal.org/node/3186368', E_USER_DEPRECATED);
-      return $this->rowCountEnabled;
-    }
-  }
-
-  /**
-   * Implements the magic __set() method.
-   *
-   * @todo Remove the method before Drupal 10.
-   * @see https://www.drupal.org/i/3210310
-   */
-  public function __set($name, $value) {
-    if ($name === 'allowRowCount') {
-      @trigger_error(__CLASS__ . '::$allowRowCount should not be written in drupal:9.3.0 and will error in drupal:10.0.0. Enable row counting by passing the appropriate argument to the constructor instead. See https://www.drupal.org/node/3186368', E_USER_DEPRECATED);
-      $this->rowCountEnabled = $value;
-    }
-  }
-
   /**
    * {@inheritdoc}
    */
diff --git a/core/lib/Drupal/Core/Database/StatementWrapper.php b/core/lib/Drupal/Core/Database/StatementWrapper.php
index 2b3eb40f46368e73f0bc6d04c40cb5d0d82eb17d..8a3bc208c563a7623354170bd8b564cc2ac52b58 100644
--- a/core/lib/Drupal/Core/Database/StatementWrapper.php
+++ b/core/lib/Drupal/Core/Database/StatementWrapper.php
@@ -53,56 +53,6 @@ public function __construct(Connection $connection, $client_connection, string $
     $this->setFetchMode(\PDO::FETCH_OBJ);
   }
 
-  /**
-   * Implements the magic __get() method.
-   *
-   * @todo Remove the method before Drupal 10.
-   * @see https://www.drupal.org/i/3210310
-   */
-  public function __get($name) {
-    if ($name === 'queryString') {
-      @trigger_error("StatementWrapper::\$queryString should not be accessed in drupal:9.1.0 and will error in drupal:10.0.0. Access the client-level statement object via ::getClientStatement(). See https://www.drupal.org/node/3177488", E_USER_DEPRECATED);
-      return $this->getClientStatement()->queryString;
-    }
-    if ($name === 'dbh') {
-      @trigger_error(__CLASS__ . '::$dbh should not be accessed in drupal:9.3.0 and will error in drupal:10.0.0. Use $this->connection instead. See https://www.drupal.org/node/3186368', E_USER_DEPRECATED);
-      return $this->connection;
-    }
-    if ($name === 'allowRowCount') {
-      @trigger_error(__CLASS__ . '::$allowRowCount should not be accessed in drupal:9.3.0 and will error in drupal:10.0.0. Use $this->rowCountEnabled instead. See https://www.drupal.org/node/3186368', E_USER_DEPRECATED);
-      return $this->rowCountEnabled;
-    }
-  }
-
-  /**
-   * Implements the magic __set() method.
-   *
-   * @todo Remove the method before Drupal 10.
-   * @see https://www.drupal.org/i/3210310
-   */
-  public function __set($name, $value) {
-    if ($name === 'allowRowCount') {
-      @trigger_error(__CLASS__ . '::$allowRowCount should not be written in drupal:9.3.0 and will error in drupal:10.0.0. Enable row counting by passing the appropriate argument to the constructor instead. See https://www.drupal.org/node/3186368', E_USER_DEPRECATED);
-      $this->rowCountEnabled = $value;
-    }
-  }
-
-  /**
-   * Implements the magic __call() method.
-   *
-   * @deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Access the
-   *   client-level statement object via ::getClientStatement().
-   *
-   * @see https://www.drupal.org/node/3177488
-   */
-  public function __call($method, $arguments) {
-    if (is_callable([$this->getClientStatement(), $method])) {
-      @trigger_error("StatementWrapper::{$method} should not be called in drupal:9.1.0 and will error in drupal:10.0.0. Access the client-level statement object via ::getClientStatement(). See https://www.drupal.org/node/3177488", E_USER_DEPRECATED);
-      return call_user_func_array([$this->getClientStatement(), $method], $arguments);
-    }
-    throw new \BadMethodCallException($method);
-  }
-
   /**
    * Returns the client-level database statement object.
    *
@@ -313,95 +263,4 @@ public function getIterator() {
     return new \ArrayIterator($this->fetchAll());
   }
 
-  /**
-   * Bind a column to a PHP variable.
-   *
-   * @param mixed $column
-   *   Number of the column (1-indexed) or name of the column in the result set.
-   *   If using the column name, be aware that the name should match the case of
-   *   the column, as returned by the driver.
-   * @param mixed $param
-   *   Name of the PHP variable to which the column will be bound.
-   * @param int $type
-   *   (Optional) data type of the parameter, specified by the PDO::PARAM_*
-   *   constants.
-   * @param int $maxlen
-   *   (Optional) a hint for pre-allocation.
-   * @param mixed $driverdata
-   *   (Optional) optional parameter(s) for the driver.
-   *
-   * @return bool
-   *   Returns TRUE on success or FALSE on failure.
-   *
-   * @deprecated in drupal:9.1.0 and is removed from drupal:10.0.0.
-   *   StatementWrapper::bindColumn should not be called. Access the
-   *   client-level statement object via ::getClientStatement().
-   *
-   * @see https://www.drupal.org/node/3177488
-   */
-  public function bindColumn($column, &$param, int $type = 0, int $maxlen = 0, $driverdata = NULL): bool {
-    @trigger_error("StatementWrapper::bindColumn should not be called in drupal:9.1.0 and will error in drupal:10.0.0. Access the client-level statement object via ::getClientStatement(). See https://www.drupal.org/node/3177488", E_USER_DEPRECATED);
-    switch (func_num_args()) {
-      case 2:
-        return $this->clientStatement->bindColumn($column, $param);
-
-      case 3:
-        return $this->clientStatement->bindColumn($column, $param, $type);
-
-      case 4:
-        return $this->clientStatement->bindColumn($column, $param, $type, $maxlen);
-
-      case 5:
-        return $this->clientStatement->bindColumn($column, $param, $type, $maxlen, $driverdata);
-
-    }
-  }
-
-  /**
-   * Binds a parameter to the specified variable name.
-   *
-   * @param mixed $parameter
-   *   Parameter identifier. For a prepared statement using named placeholders,
-   *   this will be a parameter name of the form :name.
-   * @param mixed $variable
-   *   Name of the PHP variable to bind to the SQL statement parameter.
-   * @param int $data_type
-   *   (Optional) explicit data type for the parameter using the PDO::PARAM_*
-   *   constants. To return an INOUT parameter from a stored procedure, use the
-   *   bitwise OR operator to set the PDO::PARAM_INPUT_OUTPUT bits for the
-   *   data_type parameter.
-   * @param int $length
-   *   (Optional) length of the data type. To indicate that a parameter is an
-   *   OUT parameter from a stored procedure, you must explicitly set the
-   *   length.
-   * @param mixed $driver_options
-   *   (Optional) Driver options.
-   *
-   * @return bool
-   *   Returns TRUE on success or FALSE on failure.
-   *
-   * @deprecated in drupal:9.1.0 and is removed from drupal:10.0.0.
-   *   StatementWrapper::bindParam should not be called. Access the
-   *   client-level statement object via ::getClientStatement().
-   *
-   * @see https://www.drupal.org/node/3177488
-   */
-  public function bindParam($parameter, &$variable, int $data_type = \PDO::PARAM_STR, int $length = 0, $driver_options = NULL) : bool {
-    @trigger_error("StatementWrapper::bindParam should not be called in drupal:9.1.0 and will error in drupal:10.0.0. Access the client-level statement object via ::getClientStatement(). See https://www.drupal.org/node/3177488", E_USER_DEPRECATED);
-    switch (func_num_args()) {
-      case 2:
-        return $this->clientStatement->bindParam($parameter, $variable);
-
-      case 3:
-        return $this->clientStatement->bindParam($parameter, $variable, $data_type);
-
-      case 4:
-        return $this->clientStatement->bindParam($parameter, $variable, $data_type, $length);
-
-      case 5:
-        return $this->clientStatement->bindParam($parameter, $variable, $data_type, $length, $driver_options);
-
-    }
-  }
-
 }
diff --git a/core/lib/Drupal/Core/Pager/PagerManager.php b/core/lib/Drupal/Core/Pager/PagerManager.php
index 1fad49ce79c3744d4bd02433bcbea7297b4b0715..e60b5d51346b16f13768936ffd8a6065995cf9c2 100644
--- a/core/lib/Drupal/Core/Pager/PagerManager.php
+++ b/core/lib/Drupal/Core/Pager/PagerManager.php
@@ -2,7 +2,6 @@
 
 namespace Drupal\Core\Pager;
 
-use Drupal\Core\Database\Query\PagerSelectExtender;
 use Drupal\Core\DependencyInjection\DependencySerializationTrait;
 
 /**
@@ -109,9 +108,6 @@ public function getMaxPagerElementId() {
    */
   public function reservePagerElementId(int $element): void {
     $this->maxPagerElementId = max($element, $this->maxPagerElementId);
-    // BC for PagerSelectExtender::$maxElement.
-    // @todo remove the line below in D10.
-    PagerSelectExtender::$maxElement = $this->getMaxPagerElementId();
   }
 
   /**
diff --git a/core/modules/mysql/src/Driver/Database/mysql/Connection.php b/core/modules/mysql/src/Driver/Database/mysql/Connection.php
index b2de80be4f4a18d44c6d49b1bbf80d0074e07a84..e15575bcb06348b1c17a0b6c9bd594963906491d 100644
--- a/core/modules/mysql/src/Driver/Database/mysql/Connection.php
+++ b/core/modules/mysql/src/Driver/Database/mysql/Connection.php
@@ -3,7 +3,6 @@
 namespace Drupal\mysql\Driver\Database\mysql;
 
 use Drupal\Core\Database\DatabaseAccessDeniedException;
-use Drupal\Core\Database\IntegrityConstraintViolationException;
 use Drupal\Core\Database\DatabaseExceptionWrapper;
 use Drupal\Core\Database\StatementWrapper;
 use Drupal\Core\Database\Database;
@@ -47,11 +46,6 @@ class Connection extends DatabaseConnection {
    */
   const SQLSTATE_SYNTAX_ERROR = 42000;
 
-  /**
-   * {@inheritdoc}
-   */
-  protected $statementClass = NULL;
-
   /**
    * {@inheritdoc}
    */
@@ -117,25 +111,6 @@ public function __construct(\PDO $connection, array $connection_options) {
     parent::__construct($connection, $connection_options);
   }
 
-  /**
-   * {@inheritdoc}
-   */
-  protected function handleQueryException(\PDOException $e, $query, array $args = [], $options = []) {
-    // In case of attempted INSERT of a record with an undefined column and no
-    // default value indicated in schema, MySql returns a 1364 error code.
-    // Throw an IntegrityConstraintViolationException here like the other
-    // drivers do, to avoid the parent class to throw a generic
-    // DatabaseExceptionWrapper instead.
-    if (!empty($e->errorInfo[1]) && $e->errorInfo[1] === 1364) {
-      @trigger_error('Connection::handleQueryException() is deprecated in drupal:9.2.0 and is removed in drupal:10.0.0. Get a handler through $this->exceptionHandler() instead, and use one of its methods. See https://www.drupal.org/node/3187222', E_USER_DEPRECATED);
-      $query_string = ($query instanceof StatementInterface) ? $query->getQueryString() : $query;
-      $message = $e->getMessage() . ": " . $query_string . "; " . print_r($args, TRUE);
-      throw new IntegrityConstraintViolationException($message, is_int($e->getCode()) ? $e->getCode() : 0, $e);
-    }
-
-    parent::handleQueryException($e, $query, $args, $options);
-  }
-
   /**
    * {@inheritdoc}
    */
@@ -247,16 +222,6 @@ public function queryRange($query, $from, $count, array $args = [], array $optio
     return $this->query($query . ' LIMIT ' . (int) $from . ', ' . (int) $count, $args, $options);
   }
 
-  /**
-   * {@inheritdoc}
-   */
-  public function queryTemporary($query, array $args = [], array $options = []) {
-    @trigger_error('Connection::queryTemporary() is deprecated in drupal:9.3.0 and is removed from drupal:10.0.0. There is no replacement. See https://www.drupal.org/node/3211781', E_USER_DEPRECATED);
-    $tablename = $this->generateTemporaryTableName();
-    $this->query('CREATE TEMPORARY TABLE {' . $tablename . '} Engine=MEMORY ' . $query, $args, $options);
-    return $tablename;
-  }
-
   public function driver() {
     return 'mysql';
   }
diff --git a/core/modules/mysql/src/Driver/Database/mysql/ExceptionHandler.php b/core/modules/mysql/src/Driver/Database/mysql/ExceptionHandler.php
index 18ca77ab258e894a2cd4e0008887fc25da4d99ac..bef64b9c2eb83f38fa1bbfd1740a524fa275a42c 100644
--- a/core/modules/mysql/src/Driver/Database/mysql/ExceptionHandler.php
+++ b/core/modules/mysql/src/Driver/Database/mysql/ExceptionHandler.php
@@ -17,13 +17,6 @@ class ExceptionHandler extends BaseExceptionHandler {
    * {@inheritdoc}
    */
   public function handleExecutionException(\Exception $exception, StatementInterface $statement, array $arguments = [], array $options = []): void {
-    if (array_key_exists('throw_exception', $options)) {
-      @trigger_error('Passing a \'throw_exception\' option to ' . __METHOD__ . ' is deprecated in drupal:9.2.0 and is removed in drupal:10.0.0. Always catch exceptions. See https://www.drupal.org/node/3201187', E_USER_DEPRECATED);
-      if (!($options['throw_exception'])) {
-        return;
-      }
-    }
-
     if ($exception instanceof \PDOException) {
       // Wrap the exception in another exception, because PHP does not allow
       // overriding Exception::getMessage(). Its message is the extra database
diff --git a/core/modules/pgsql/src/Driver/Database/pgsql/Connection.php b/core/modules/pgsql/src/Driver/Database/pgsql/Connection.php
index fc2ab6508f8b1a2fef03887ad6c5b91dad6d5e6d..037e3860941b74f6d7d447ae8f719b93d80befe4 100644
--- a/core/modules/pgsql/src/Driver/Database/pgsql/Connection.php
+++ b/core/modules/pgsql/src/Driver/Database/pgsql/Connection.php
@@ -39,11 +39,6 @@ class Connection extends DatabaseConnection {
    */
   const CONNECTION_FAILURE = '08006';
 
-  /**
-   * {@inheritdoc}
-   */
-  protected $statementClass = NULL;
-
   /**
    * {@inheritdoc}
    */
@@ -214,16 +209,6 @@ public function queryRange($query, $from, $count, array $args = [], array $optio
     return $this->query($query . ' LIMIT ' . (int) $count . ' OFFSET ' . (int) $from, $args, $options);
   }
 
-  /**
-   * {@inheritdoc}
-   */
-  public function queryTemporary($query, array $args = [], array $options = []) {
-    @trigger_error('Connection::queryTemporary() is deprecated in drupal:9.3.0 and is removed from drupal:10.0.0. There is no replacement. See https://www.drupal.org/node/3211781', E_USER_DEPRECATED);
-    $tablename = $this->generateTemporaryTableName();
-    $this->query('CREATE TEMPORARY TABLE {' . $tablename . '} AS ' . $query, $args, $options);
-    return $tablename;
-  }
-
   public function driver() {
     return 'pgsql';
   }
diff --git a/core/modules/sqlite/src/Driver/Database/sqlite/Connection.php b/core/modules/sqlite/src/Driver/Database/sqlite/Connection.php
index 17fd2567d574f641076897ff8ce87f8ac31027a5..72fba8eb43c53c4cda856d4ef5d50d37394aa541 100644
--- a/core/modules/sqlite/src/Driver/Database/sqlite/Connection.php
+++ b/core/modules/sqlite/src/Driver/Database/sqlite/Connection.php
@@ -16,11 +16,6 @@ class Connection extends DatabaseConnection {
    */
   const DATABASE_NOT_FOUND = 14;
 
-  /**
-   * {@inheritdoc}
-   */
-  protected $statementClass = NULL;
-
   /**
    * {@inheritdoc}
    */
@@ -357,52 +352,10 @@ public static function sqlFunctionLikeBinary($pattern, $subject) {
     return preg_match('/^' . $pattern . '$/', $subject);
   }
 
-  /**
-   * {@inheritdoc}
-   */
-  public function prepare($statement, array $driver_options = []) {
-    @trigger_error('Connection::prepare() is deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Database drivers should instantiate \PDOStatement objects by calling \PDO::prepare in their Connection::prepareStatement method instead. \PDO::prepare should not be called outside of driver code. See https://www.drupal.org/node/3137786', E_USER_DEPRECATED);
-    return new Statement($this->connection, $this, $statement, $driver_options);
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function handleQueryException(\PDOException $e, $query, array $args = [], $options = []) {
-    // The database schema might be changed by another process in between the
-    // time that the statement was prepared and the time the statement was run
-    // (e.g. usually happens when running tests). In this case, we need to
-    // re-run the query.
-    // @see http://www.sqlite.org/faq.html#q15
-    // @see http://www.sqlite.org/rescode.html#schema
-    if (!empty($e->errorInfo[1]) && $e->errorInfo[1] === 17) {
-      @trigger_error('Connection::handleQueryException() is deprecated in drupal:9.2.0 and is removed in drupal:10.0.0. Get a handler through $this->exceptionHandler() instead, and use one of its methods. See https://www.drupal.org/node/3187222', E_USER_DEPRECATED);
-      return $this->query($query, $args, $options);
-    }
-
-    parent::handleQueryException($e, $query, $args, $options);
-  }
-
   public function queryRange($query, $from, $count, array $args = [], array $options = []) {
     return $this->query($query . ' LIMIT ' . (int) $from . ', ' . (int) $count, $args, $options);
   }
 
-  /**
-   * {@inheritdoc}
-   */
-  public function queryTemporary($query, array $args = [], array $options = []) {
-    @trigger_error('Connection::queryTemporary() is deprecated in drupal:9.3.0 and is removed from drupal:10.0.0. There is no replacement. See https://www.drupal.org/node/3211781', E_USER_DEPRECATED);
-    // Generate a new temporary table name and protect it from prefixing.
-    // SQLite requires that temporary tables to be non-qualified.
-    $tablename = $this->generateTemporaryTableName();
-    $prefixes = $this->prefixes;
-    $prefixes[$tablename] = '';
-    $this->setPrefix($prefixes);
-
-    $this->query('CREATE TEMPORARY TABLE ' . $tablename . ' AS ' . $query, $args, $options);
-    return $tablename;
-  }
-
   public function driver() {
     return 'sqlite';
   }
diff --git a/core/modules/system/tests/modules/database_test/database_test.routing.yml b/core/modules/system/tests/modules/database_test/database_test.routing.yml
index 51b365e29ae9e4f2213edb0e15f9220bb5c8d2a1..1bc0751b4a6b21f95bee420db1f8f0d9967146dc 100644
--- a/core/modules/system/tests/modules/database_test/database_test.routing.yml
+++ b/core/modules/system/tests/modules/database_test/database_test.routing.yml
@@ -1,10 +1,3 @@
-database_test.db_query_temporary:
-  path: '/database_test/db_query_temporary'
-  defaults:
-    _controller: '\Drupal\database_test\Controller\DatabaseTestController::dbQueryTemporary'
-  requirements:
-    _access: 'TRUE'
-
 database_test.pager_query_even:
   path: '/database_test/pager_query_even/{limit}'
   defaults:
diff --git a/core/modules/system/tests/modules/database_test/src/Controller/DatabaseTestController.php b/core/modules/system/tests/modules/database_test/src/Controller/DatabaseTestController.php
index 2364e39fc7df5bf62f695ae5359aba124d0b6c81..ddc5f7577b26529d90450587dfe2a65cdcbe2b42 100644
--- a/core/modules/system/tests/modules/database_test/src/Controller/DatabaseTestController.php
+++ b/core/modules/system/tests/modules/database_test/src/Controller/DatabaseTestController.php
@@ -40,23 +40,6 @@ public static function create(ContainerInterface $container) {
     );
   }
 
-  /**
-   * Creates temporary table and outputs the table name and its number of rows.
-   *
-   * We need to test that the table created is temporary, so we run it here, in a
-   * separate menu callback request; After this request is done, the temporary
-   * table should automatically dropped.
-   *
-   * @return \Symfony\Component\HttpFoundation\JsonResponse
-   */
-  public function dbQueryTemporary() {
-    $table_name = $this->connection->queryTemporary('SELECT [age] FROM {test}', []);
-    return new JsonResponse([
-      'table_name' => $table_name,
-      'row_count' => $this->connection->select($table_name)->countQuery()->execute()->fetchField(),
-    ]);
-  }
-
   /**
    * Runs a pager query and returns the results.
    *
diff --git a/core/modules/system/tests/src/Functional/Database/SelectPagerDefaultTest.php b/core/modules/system/tests/src/Functional/Database/SelectPagerDefaultTest.php
index e261f6e9a3fb7841d6921e0c5216d98ee10c9230..41aa5c3a830994e61b1aa48cce236b5809149cdd 100644
--- a/core/modules/system/tests/src/Functional/Database/SelectPagerDefaultTest.php
+++ b/core/modules/system/tests/src/Functional/Database/SelectPagerDefaultTest.php
@@ -151,9 +151,6 @@ public function testElementNumbers() {
       ->orderBy('age')
       ->limit(1);
     $this->assertSame(2, $query->getElement());
-    // BC for PagerSelectExtender::$maxElement.
-    // @todo remove the assertion below in D10.
-    $this->assertSame(2, PagerSelectExtender::$maxElement);
     $name = $query->execute()
       ->fetchField();
     $this->assertEquals('Paul', $name, 'Pager query #1 with a specified element ID returned the correct results.');
@@ -167,9 +164,6 @@ public function testElementNumbers() {
       ->orderBy('age')
       ->limit(1);
     $this->assertSame(1, $query->getElement());
-    // BC for PagerSelectExtender::$maxElement.
-    // @todo remove the assertion below in D10.
-    $this->assertSame(2, PagerSelectExtender::$maxElement);
     $name = $query->execute()
       ->fetchField();
     $this->assertEquals('George', $name, 'Pager query #2 with a specified element ID returned the correct results.');
@@ -180,9 +174,6 @@ public function testElementNumbers() {
       ->orderBy('age')
       ->limit(1);
     $this->assertSame(3, $query->getElement());
-    // BC for PagerSelectExtender::$maxElement.
-    // @todo remove the assertion below in D10.
-    $this->assertSame(3, PagerSelectExtender::$maxElement);
     $name = $query->execute()
       ->fetchField();
     $this->assertEquals('John', $name, 'Pager query #3 with a generated element ID returned the correct results.');
diff --git a/core/modules/system/tests/src/Functional/Database/TemporaryQueryTest.php b/core/modules/system/tests/src/Functional/Database/TemporaryQueryTest.php
deleted file mode 100644
index 419363a2cb9fa583a2a7c37830347c5c21bfc746..0000000000000000000000000000000000000000
--- a/core/modules/system/tests/src/Functional/Database/TemporaryQueryTest.php
+++ /dev/null
@@ -1,67 +0,0 @@
-<?php
-
-namespace Drupal\Tests\system\Functional\Database;
-
-use Drupal\Core\Database\Database;
-
-/**
- * Tests the temporary query functionality.
- *
- * @group Database
- * @group legacy
- */
-class TemporaryQueryTest extends DatabaseTestBase {
-
-  /**
-   * {@inheritdoc}
-   */
-  protected static $modules = ['database_test'];
-
-  /**
-   * {@inheritdoc}
-   */
-  protected $defaultTheme = 'stark';
-
-  /**
-   * Returns the number of rows of a table.
-   */
-  public function countTableRows($table_name) {
-    return Database::getConnection()->select($table_name)->countQuery()->execute()->fetchField();
-  }
-
-  /**
-   * Confirms that temporary tables work and are limited to one request.
-   */
-  public function testTemporaryQuery() {
-    $this->expectDeprecation('Connection::queryTemporary() is deprecated in drupal:9.3.0 and is removed from drupal:10.0.0. There is no replacement. See https://www.drupal.org/node/3211781');
-    $this->expectDeprecation('Connection::generateTemporaryTableName() is deprecated in drupal:9.3.0 and is removed from drupal:10.0.0. There is no replacement. See https://www.drupal.org/node/3211781');
-
-    $connection = Database::getConnection();
-    $this->drupalGet('database_test/db_query_temporary');
-    $data = json_decode($this->getSession()->getPage()->getContent());
-    if ($data) {
-      $this->assertEquals($this->countTableRows('test'), $data->row_count, 'The temporary table contains the correct amount of rows.');
-      $this->assertFalse($connection->schema()->tableExists($data->table_name), 'The temporary table is, indeed, temporary.');
-    }
-    else {
-      $this->fail('The creation of the temporary table failed.');
-    }
-
-    // Now try to run two temporary queries in the same request.
-    $table_name_test = $connection->queryTemporary('SELECT [name] FROM {test}', []);
-    $table_name_task = $connection->queryTemporary('SELECT [pid] FROM {test_task}', []);
-
-    $this->assertEquals($this->countTableRows('test'), $this->countTableRows($table_name_test), 'A temporary table was created successfully in this request.');
-    $this->assertEquals($this->countTableRows('test_task'), $this->countTableRows($table_name_task), 'A second temporary table was created successfully in this request.');
-
-    // Check that leading whitespace and comments do not cause problems
-    // in the modified query.
-    $sql = "
-      -- Let's select some rows into a temporary table
-      SELECT [name] FROM {test}
-    ";
-    $table_name_test = $connection->queryTemporary($sql, []);
-    $this->assertEquals($this->countTableRows('test'), $this->countTableRows($table_name_test), 'Leading white space and comments do not interfere with temporary table creation.');
-  }
-
-}
diff --git a/core/tests/Drupal/KernelTests/Core/Database/ConnectionTest.php b/core/tests/Drupal/KernelTests/Core/Database/ConnectionTest.php
index 570146fc10da67be1efeb025711b504c83de093b..b7da5e5c536e9c44e74d3ad6279555d902ab0d50 100644
--- a/core/tests/Drupal/KernelTests/Core/Database/ConnectionTest.php
+++ b/core/tests/Drupal/KernelTests/Core/Database/ConnectionTest.php
@@ -6,7 +6,6 @@
 use Drupal\Core\Database\Database;
 use Drupal\Core\Database\DatabaseExceptionWrapper;
 use Drupal\Core\Database\Query\Condition;
-use Drupal\Core\Database\StatementWrapper;
 
 /**
  * Tests of the core database system.
@@ -119,52 +118,6 @@ public function testConnectionOptions() {
     $this->assertNotEquals($connection_info['default']['database'], $connectionOptions['database'], 'The test connection info database does not match the current connection options database.');
   }
 
-  /**
-   * Tests the deprecation of the 'transactions' connection option.
-   *
-   * @group legacy
-   */
-  public function testTransactionsOptionDeprecation() {
-    $this->expectDeprecation('Passing a \'transactions\' connection option to Drupal\Core\Database\Connection::__construct is deprecated in drupal:9.1.0 and is removed in drupal:10.0.0. All database drivers must support transactions. See https://www.drupal.org/node/2278745');
-    $this->expectDeprecation('Drupal\Core\Database\Connection::supportsTransactions is deprecated in drupal:9.1.0 and is removed in drupal:10.0.0. All database drivers must support transactions. See https://www.drupal.org/node/2278745');
-    $connection_info = Database::getConnectionInfo('default');
-    $connection_info['default']['transactions'] = FALSE;
-    Database::addConnectionInfo('default', 'foo', $connection_info['default']);
-    $foo_connection = Database::getConnection('foo', 'default');
-    $this->assertInstanceOf(Connection::class, $foo_connection);
-    $this->assertTrue($foo_connection->supportsTransactions());
-  }
-
-  /**
-   * Tests the deprecation of passing a statement object to ::query.
-   *
-   * @group legacy
-   */
-  public function testStatementQueryDeprecation(): void {
-    $this->expectDeprecation('Passing a StatementInterface object as a $query argument to Drupal\Core\Database\Connection::query is deprecated in drupal:9.2.0 and is removed in drupal:10.0.0. Call the execute method from the StatementInterface object directly instead. See https://www.drupal.org/node/3154439');
-    $db = Database::getConnection();
-    $stmt = $db->prepareStatement('SELECT * FROM {test}', []);
-    $this->assertNotNull($db->query($stmt));
-  }
-
-  /**
-   * Tests the deprecation of passing a PDOStatement object to ::query.
-   *
-   * @group legacy
-   */
-  public function testPDOStatementQueryDeprecation(): void {
-    $db = Database::getConnection();
-    $stmt = $db->prepareStatement('SELECT * FROM {test}', []);
-    if (!$stmt instanceof StatementWrapper) {
-      $this->markTestSkipped("This test only runs for db drivers using StatementWrapper.");
-    }
-    if (!$stmt->getClientStatement() instanceof \PDOStatement) {
-      $this->markTestSkipped("This test only runs for PDO-based db drivers.");
-    }
-    $this->expectDeprecation('Passing a \\PDOStatement object as a $query argument to Drupal\Core\Database\Connection::query is deprecated in drupal:9.2.0 and is removed in drupal:10.0.0. Call the execute method from the StatementInterface object directly instead. See https://www.drupal.org/node/3154439');
-    $this->assertNotNull($db->query($stmt->getClientStatement()));
-  }
-
   /**
    * Tests per-table prefix connection option.
    */
diff --git a/core/tests/Drupal/KernelTests/Core/Database/DatabaseExceptionWrapperTest.php b/core/tests/Drupal/KernelTests/Core/Database/DatabaseExceptionWrapperTest.php
index d7af19f1c919ef53f3f867dff7b7169c373488f5..0fd2770fe52696f0535201e75f82df271731d62b 100644
--- a/core/tests/Drupal/KernelTests/Core/Database/DatabaseExceptionWrapperTest.php
+++ b/core/tests/Drupal/KernelTests/Core/Database/DatabaseExceptionWrapperTest.php
@@ -13,55 +13,6 @@
  */
 class DatabaseExceptionWrapperTest extends KernelTestBase {
 
-  /**
-   * Tests deprecation of Connection::prepare.
-   *
-   * @group legacy
-   */
-  public function testPrepare() {
-    $this->expectDeprecation('Connection::prepare() is deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Database drivers should instantiate \PDOStatement objects by calling \PDO::prepare in their Connection::prepareStatement method instead. \PDO::prepare should not be called outside of driver code. See https://www.drupal.org/node/3137786');
-    $connection = Database::getConnection();
-    try {
-      // SQLite validates the syntax upon preparing a statement already.
-      // @throws \PDOException
-      $query = $connection->prepare('bananas');
-
-      // MySQL only validates the syntax upon trying to execute a query.
-      // @throws \Drupal\Core\Database\DatabaseExceptionWrapper
-      $connection->query($query);
-
-      $this->fail('Expected PDOException or DatabaseExceptionWrapper, none was thrown.');
-    }
-    catch (\Exception $e) {
-      $this->assertTrue($e instanceof \PDOException || $e instanceof DatabaseExceptionWrapper, 'Exception should be an instance of \PDOException or DatabaseExceptionWrapper, thrown ' . get_class($e));
-    }
-  }
-
-  /**
-   * Tests deprecation of Connection::prepareQuery.
-   *
-   * @group legacy
-   */
-  public function testPrepareQuery() {
-    $this->expectDeprecation('Connection::prepareQuery() is deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Use ::prepareStatement() instead. See https://www.drupal.org/node/3137786');
-    $this->expectException(\PDOException::class);
-    $stmt = Database::getConnection()->prepareQuery('bananas');
-    $stmt->execute();
-  }
-
-  /**
-   * Tests deprecation of Connection::handleQueryException.
-   *
-   * @group legacy
-   */
-  public function testHandleQueryExceptionDeprecation(): void {
-    $this->expectDeprecation('Passing a StatementInterface object as a $query argument to Drupal\Core\Database\Connection::query is deprecated in drupal:9.2.0 and is removed in drupal:10.0.0. Call the execute method from the StatementInterface object directly instead. See https://www.drupal.org/node/3154439');
-    $this->expectDeprecation('Connection::handleQueryException() is deprecated in drupal:9.2.0 and is removed in drupal:10.0.0. Get a handler through $this->exceptionHandler() instead, and use one of its methods. See https://www.drupal.org/node/3187222');
-    $this->expectException(DatabaseExceptionWrapper::class);
-    $stmt = Database::getConnection()->prepareStatement('SELECT * FROM {does_not_exist}', []);
-    Database::getConnection()->query($stmt);
-  }
-
   /**
    * Tests Connection::prepareStatement exceptions on execution.
    *
@@ -97,26 +48,6 @@ public function testPrepareStatementFailOnPreparation() {
     $stmt = $foo_connection->prepareStatement('bananas', []);
   }
 
-  /**
-   * Tests Connection::prepareStatement with throw_exception option set.
-   *
-   * @group legacy
-   */
-  public function testPrepareStatementFailOnPreparationWithThrowExceptionOption(): void {
-    $driver = Database::getConnection()->driver();
-    if ($driver !== 'mysql') {
-      $this->markTestSkipped("MySql tests can not run for driver '$driver'.");
-    }
-
-    $connection_info = Database::getConnectionInfo('default');
-    $connection_info['default']['pdo'][\PDO::ATTR_EMULATE_PREPARES] = FALSE;
-    Database::addConnectionInfo('default', 'foo', $connection_info['default']);
-    $foo_connection = Database::getConnection('foo', 'default');
-    $this->expectException(DatabaseExceptionWrapper::class);
-    $this->expectDeprecation('Passing a \'throw_exception\' option to %AExceptionHandler::handleStatementException is deprecated in drupal:9.2.0 and is removed in drupal:10.0.0. Always catch exceptions. See https://www.drupal.org/node/3201187');
-    $stmt = $foo_connection->prepareStatement('bananas', ['throw_exception' => FALSE]);
-  }
-
   /**
    * Tests the expected database exception thrown for inexistent tables.
    */
diff --git a/core/tests/Drupal/KernelTests/Core/Database/MergeTest.php b/core/tests/Drupal/KernelTests/Core/Database/MergeTest.php
index b0ad41d455fed698e0ffc463cbe1b589206932a1..11b2da2786051e2225b6e89aec24425a87ed20f3 100644
--- a/core/tests/Drupal/KernelTests/Core/Database/MergeTest.php
+++ b/core/tests/Drupal/KernelTests/Core/Database/MergeTest.php
@@ -204,21 +204,6 @@ public function testInvalidMerge() {
       ->execute();
   }
 
-  /**
-   * Tests deprecation of the 'throw_exception' option.
-   *
-   * @group legacy
-   */
-  public function testLegacyThrowExceptionOption(): void {
-    $this->expectDeprecation("Passing a 'throw_exception' option to %AMerge::execute is deprecated in drupal:9.2.0 and is removed in drupal:10.0.0. Always catch exceptions. See https://www.drupal.org/node/3201187");
-    // This merge will fail because there is no key field specified.
-    $this->assertNull($this->connection
-      ->merge('test_people', ['throw_exception' => FALSE])
-      ->fields(['age' => 31, 'name' => 'Tiffany'])
-      ->execute()
-    );
-  }
-
   /**
    * Tests that we can merge-insert with reserved keywords.
    */
diff --git a/core/tests/Drupal/KernelTests/Core/Database/SelectTest.php b/core/tests/Drupal/KernelTests/Core/Database/SelectTest.php
index 89e98bf4a20415574445da9e178c43b651380184..fe85345d9b9419206dd41853d4042e37c8c03637 100644
--- a/core/tests/Drupal/KernelTests/Core/Database/SelectTest.php
+++ b/core/tests/Drupal/KernelTests/Core/Database/SelectTest.php
@@ -550,21 +550,6 @@ public function testSelectDuplicateAlias() {
     $this->assertNotSame($alias1, $alias2, 'Duplicate aliases are renamed.');
   }
 
-  /**
-   * Tests deprecation of the 'throw_exception' option.
-   *
-   * @group legacy
-   */
-  public function testLegacyThrowExceptionOption(): void {
-    $this->expectDeprecation("Passing a 'throw_exception' option to %AExceptionHandler::handleExecutionException is deprecated in drupal:9.2.0 and is removed in drupal:10.0.0. Always catch exceptions. See https://www.drupal.org/node/3201187");
-    // This query will fail because the table does not exist.
-    $this->assertNull($this->connection->select('some_table_that_does_not_exist', 't', ['throw_exception' => FALSE])
-      ->fields('t')
-      ->countQuery()
-      ->execute()
-    );
-  }
-
   /**
    * Tests that an invalid count query throws an exception.
    */
diff --git a/core/tests/Drupal/KernelTests/Core/Database/StatementTest.php b/core/tests/Drupal/KernelTests/Core/Database/StatementTest.php
index f1c0cc806023c319641b54b0e01ae98fa1b3ae96..eeb23046638543be90bc4596bb94b7af2755f9d2 100644
--- a/core/tests/Drupal/KernelTests/Core/Database/StatementTest.php
+++ b/core/tests/Drupal/KernelTests/Core/Database/StatementTest.php
@@ -43,30 +43,4 @@ public function testRepeatedInsertStatementReuse() {
     $this->assertSame('31', $this->connection->query('SELECT [age] FROM {test} WHERE [name] = :name', [':name' => 'Curly'])->fetchField());
   }
 
-  /**
-   * Tests accessing deprecated properties.
-   *
-   * @group legacy
-   */
-  public function testGetDeprecatedProperties(): void {
-    $statement = $this->connection->prepareStatement('SELECT * FROM {test}', []);
-    $this->expectDeprecation('%s$dbh should not be accessed in drupal:9.3.0 and will error in drupal:10.0.0. Use $this->connection instead. See https://www.drupal.org/node/3186368');
-    $this->assertNotNull($statement->dbh);
-    $this->expectDeprecation('%s$allowRowCount should not be accessed in drupal:9.3.0 and will error in drupal:10.0.0. Use $this->rowCountEnabled instead. See https://www.drupal.org/node/3186368');
-    $this->assertFalse($statement->allowRowCount);
-  }
-
-  /**
-   * Tests writing deprecated properties.
-   *
-   * @group legacy
-   */
-  public function testSetDeprecatedProperties(): void {
-    $statement = $this->connection->prepareStatement('UPDATE {test} SET [age] = :age', []);
-    $this->expectDeprecation('%s$allowRowCount should not be written in drupal:9.3.0 and will error in drupal:10.0.0. Enable row counting by passing the appropriate argument to the constructor instead. See https://www.drupal.org/node/3186368');
-    $statement->allowRowCount = TRUE;
-    $statement->execute([':age' => 12]);
-    $this->assertEquals(4, $statement->rowCount());
-  }
-
 }
diff --git a/core/tests/Drupal/KernelTests/Core/Database/StatementWrapperLegacyTest.php b/core/tests/Drupal/KernelTests/Core/Database/StatementWrapperLegacyTest.php
deleted file mode 100644
index 9c7457a58aabdc66667cad6870845d72d28910dd..0000000000000000000000000000000000000000
--- a/core/tests/Drupal/KernelTests/Core/Database/StatementWrapperLegacyTest.php
+++ /dev/null
@@ -1,73 +0,0 @@
-<?php
-
-namespace Drupal\KernelTests\Core\Database;
-
-use Drupal\Core\Database\StatementWrapper;
-
-/**
- * Tests the deprecations of the StatementWrapper class.
- *
- * @coversDefaultClass \Drupal\Core\Database\StatementWrapper
- * @group legacy
- * @group Database
- */
-class StatementWrapperLegacyTest extends DatabaseTestBase {
-
-  protected $statement;
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function setUp(): void {
-    parent::setUp();
-    $this->statement = $this->connection->prepareStatement('SELECT id FROM {test}', []);
-    if (!$this->statement instanceof StatementWrapper) {
-      $this->markTestSkipped('This test only works for drivers implementing Drupal\Core\Database\StatementWrapper.');
-    }
-  }
-
-  /**
-   * @covers ::getQueryString
-   */
-  public function testQueryString() {
-    $this->expectDeprecation('StatementWrapper::$queryString should not be accessed in drupal:9.1.0 and will error in drupal:10.0.0. Access the client-level statement object via ::getClientStatement(). See https://www.drupal.org/node/3177488');
-    $this->assertStringContainsString('SELECT id FROM ', $this->statement->queryString);
-    $this->assertStringContainsString('SELECT id FROM ', $this->statement->getQueryString());
-  }
-
-  /**
-   * Tests calling a non existing \PDOStatement method.
-   */
-  public function testMissingMethod() {
-    $this->expectException('\BadMethodCallException');
-    $this->statement->boo();
-  }
-
-  /**
-   * Tests calling an existing \PDOStatement method.
-   */
-  public function testClientStatementMethod() {
-    $this->expectDeprecation('StatementWrapper::columnCount should not be called in drupal:9.1.0 and will error in drupal:10.0.0. Access the client-level statement object via ::getClientStatement(). See https://www.drupal.org/node/3177488');
-    $this->statement->execute();
-    $this->assertEquals(1, $this->statement->columnCount());
-  }
-
-  /**
-   * @covers ::bindParam
-   */
-  public function testBindParam() {
-    $this->expectDeprecation('StatementWrapper::bindParam should not be called in drupal:9.1.0 and will error in drupal:10.0.0. Access the client-level statement object via ::getClientStatement(). See https://www.drupal.org/node/3177488');
-    $test = NULL;
-    $this->assertTrue($this->statement->bindParam(':name', $test));
-  }
-
-  /**
-   * @covers ::bindColumn
-   */
-  public function testBindColumn() {
-    $this->expectDeprecation('StatementWrapper::bindColumn should not be called in drupal:9.1.0 and will error in drupal:10.0.0. Access the client-level statement object via ::getClientStatement(). See https://www.drupal.org/node/3177488');
-    $test = NULL;
-    $this->assertTrue($this->statement->bindColumn(1, $test));
-  }
-
-}
diff --git a/core/tests/Drupal/Tests/Core/Database/ConditionTest.php b/core/tests/Drupal/Tests/Core/Database/ConditionTest.php
index 05b65b5f7d67961a857d28a32c1c78f12dd1752a..6ef0cc7adb5877d3c17f670b1348b9e3031c25cc 100644
--- a/core/tests/Drupal/Tests/Core/Database/ConditionTest.php
+++ b/core/tests/Drupal/Tests/Core/Database/ConditionTest.php
@@ -203,15 +203,4 @@ class_alias('MockCondition', $mocked_namespace);
     $this->assertSame('MockCondition', get_class($condition));
   }
 
-  /**
-   * Tests the deprecation of the class Condition.
-   *
-   * @group legacy
-   */
-  public function testConditionClassDeprecation() {
-    $this->expectDeprecation('Creating an instance of this class is deprecated in drupal:9.1.0 and is removed in drupal:10.0.0. Use Database::getConnection()->condition() instead. See https://www.drupal.org/node/3159568');
-    $condition = new Condition('OR');
-    $this->assertSame('Drupal\Core\Database\Query\Condition', get_class($condition));
-  }
-
 }
diff --git a/core/tests/Drupal/Tests/Core/Database/ConnectionTest.php b/core/tests/Drupal/Tests/Core/Database/ConnectionTest.php
index 09b2f1a9ab08008406372f57c4e0b0e05603c45d..412ad324e6b3c5c13da15ad88f19caa661bd71a6 100644
--- a/core/tests/Drupal/Tests/Core/Database/ConnectionTest.php
+++ b/core/tests/Drupal/Tests/Core/Database/ConnectionTest.php
@@ -3,7 +3,6 @@
 namespace Drupal\Tests\Core\Database;
 
 use Composer\Autoload\ClassLoader;
-use Drupal\Core\Database\Statement;
 use Drupal\Tests\Core\Database\Stub\StubConnection;
 use Drupal\Tests\Core\Database\Stub\StubPDO;
 use Drupal\Tests\UnitTestCase;
@@ -359,51 +358,6 @@ public function testSchema($expected, $driver, $namespace) {
     $this->assertInstanceOf($expected, $connection->schema());
   }
 
-  /**
-   * Tests Connection::destroy().
-   *
-   * @group legacy
-   */
-  public function testDestroy() {
-    $this->expectDeprecation('Drupal\Core\Database\Connection::destroy() is deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Move custom database destruction logic to __destruct(). See https://www.drupal.org/node/3142866');
-    $mock_pdo = $this->createMock('Drupal\Tests\Core\Database\Stub\StubPDO');
-    // Mocking StubConnection gives us access to the $schema attribute.
-    $connection = new StubConnection($mock_pdo, ['namespace' => 'Drupal\\Tests\\Core\\Database\\Stub\\Driver']);
-    // Generate a schema object in order to verify that we've NULLed it later.
-    $this->assertInstanceOf(
-      'Drupal\\Tests\\Core\\Database\\Stub\\Driver\\Schema',
-      $connection->schema()
-    );
-    $connection->destroy();
-
-    $reflected_schema = (new \ReflectionObject($connection))->getProperty('schema');
-    $reflected_schema->setAccessible(TRUE);
-    $this->assertNull($reflected_schema->getValue($connection));
-  }
-
-  /**
-   * Tests Connection::__destruct().
-   *
-   * @group legacy
-   */
-  public function testDestructBcLayer() {
-    $this->expectDeprecation('Drupal\Core\Database\Connection::destroy() is deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Move custom database destruction logic to __destruct(). See https://www.drupal.org/node/3142866');
-    $mock_pdo = $this->createMock(StubPDO::class);
-    $fake_connection = new class($mock_pdo, ['namespace' => Driver::class]) extends StubConnection {
-
-      public function destroy() {
-        parent::destroy();
-      }
-
-    };
-    // Destroy the object which will result in the Connection::__destruct()
-    // calling Connection::destroy() and a deprecation error being triggered.
-    // @see \Drupal\KernelTests\Core\Database\ConnectionUnitTest for tests that
-    // connection object destruction does not trigger deprecations unless
-    // Connection::destroy() is overridden.
-    $fake_connection = NULL;
-  }
-
   /**
    * Data provider for testMakeComments().
    *
@@ -605,16 +559,6 @@ public function testEscapeDatabase($expected, $name, array $identifier_quote = [
     $this->assertEquals($expected, $connection->escapeDatabase($name));
   }
 
-  /**
-   * @covers ::__construct
-   * @group legacy
-   */
-  public function testIdentifierQuotesDeprecation() {
-    $this->expectDeprecation('In drupal:10.0.0 not setting the $identifierQuotes property in the concrete Connection class will result in an RuntimeException. See https://www.drupal.org/node/2986894');
-    $mock_pdo = $this->createMock(StubPDO::class);
-    new StubConnection($mock_pdo, [], NULL);
-  }
-
   /**
    * @covers ::__construct
    */
@@ -644,21 +588,6 @@ public function testNamespaceDefault() {
     $this->assertSame('Drupal\Tests\Core\Database\Stub', $connection->getConnectionOptions()['namespace']);
   }
 
-  /**
-   * Tests deprecation of the Statement class.
-   *
-   * @group legacy
-   */
-  public function testStatementDeprecation() {
-    if (PHP_VERSION_ID >= 80000) {
-      $this->markTestSkipped('Drupal\Core\Database\Statement is incompatible with PHP 8.0. Remove in https://www.drupal.org/node/3177490');
-    }
-    $this->expectDeprecation('\Drupal\Core\Database\Statement is deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Database drivers should use or extend StatementWrapper instead, and encapsulate client-level statement objects. See https://www.drupal.org/node/3177488');
-    $mock_statement = $this->getMockBuilder(Statement::class)
-      ->disableOriginalConstructor()
-      ->getMock();
-  }
-
   /**
    * Test rtrim() of query strings.
    *
@@ -717,31 +646,4 @@ public function provideQueriesToTrim() {
     ];
   }
 
-  /**
-   * Tests the deprecation of Drupal 8 style database drivers.
-   *
-   * @group legacy
-   */
-  public function testLegacyDatabaseDriverInRootDriversDirectory() {
-    $this->expectDeprecation('Support for database drivers located in the "drivers/lib/Drupal/Driver/Database" directory is deprecated in drupal:9.1.0 and is removed in drupal:10.0.0. Contributed and custom database drivers should be provided by modules and use the namespace "Drupal\MODULE_NAME\Driver\Database\DRIVER_NAME". See https://www.drupal.org/node/3123251');
-    $namespace = 'Drupal\\Driver\\Database\\Stub';
-    $mock_pdo = $this->createMock(StubPDO::class);
-    $connection = new StubConnection($mock_pdo, ['namespace' => $namespace], ['"', '"']);
-    $this->assertEquals($namespace, $connection->getConnectionOptions()['namespace']);
-  }
-
-  /**
-   * Tests the deprecation of \Drupal\Core\Database\Connection::$statementClass.
-   *
-   * @group legacy
-   */
-  public function testPdoStatementClass() {
-    if (PHP_VERSION_ID >= 80000) {
-      $this->markTestSkipped('Drupal\Core\Database\Statement is incompatible with PHP 8.0. Remove in https://www.drupal.org/node/3177490');
-    }
-    $this->expectDeprecation('\Drupal\Core\Database\Connection::$statementClass is deprecated in drupal:9.1.0 and is removed from drupal:10.0.0. Database drivers should use or extend StatementWrapper instead, and encapsulate client-level statement objects. See https://www.drupal.org/node/3177488');
-    $mock_pdo = $this->createMock(StubPDO::class);
-    new StubConnection($mock_pdo, ['namespace' => 'Drupal\\Tests\\Core\\Database\\Stub\\Driver'], ['"', '"'], Statement::class);
-  }
-
 }
diff --git a/core/tests/Drupal/Tests/Core/Database/EmptyStatementTest.php b/core/tests/Drupal/Tests/Core/Database/EmptyStatementTest.php
deleted file mode 100644
index f17c968ab905670e4a98adb75da9c4ea27eb5fa8..0000000000000000000000000000000000000000
--- a/core/tests/Drupal/Tests/Core/Database/EmptyStatementTest.php
+++ /dev/null
@@ -1,46 +0,0 @@
-<?php
-
-namespace Drupal\Tests\Core\Database;
-
-use Drupal\Core\Database\StatementEmpty;
-use Drupal\Core\Database\StatementInterface;
-use Drupal\Tests\UnitTestCase;
-
-/**
- * Tests the empty pseudo-statement class.
- *
- * @group Database
- */
-class EmptyStatementTest extends UnitTestCase {
-
-  /**
-   * Tests that the empty result set behaves as empty.
-   *
-   * @group legacy
-   */
-  public function testEmpty() {
-    $this->expectDeprecation('\Drupal\Core\Database\StatementEmpty is deprecated in drupal:9.2.0 and is removed from drupal:10.0.0. There is no replacement. Use mocked StatementInterface classes in tests if needed. See https://www.drupal.org/node/3201283');
-    $result = new StatementEmpty();
-
-    $this->assertInstanceOf(StatementInterface::class, $result);
-    $this->assertNull($result->fetchObject(), 'Null result returned.');
-  }
-
-  /**
-   * Tests that the empty result set iterates safely.
-   */
-  public function testEmptyIteration() {
-    $result = new StatementEmpty();
-    $this->assertSame(0, iterator_count($result), 'Empty result set should not iterate.');
-  }
-
-  /**
-   * Tests that the empty result set mass-fetches in an expected way.
-   */
-  public function testEmptyFetchAll() {
-    $result = new StatementEmpty();
-
-    $this->assertEquals([], $result->fetchAll(), 'Empty array returned from empty result set.');
-  }
-
-}
diff --git a/core/tests/Drupal/Tests/Core/Database/Stub/StubConnection.php b/core/tests/Drupal/Tests/Core/Database/Stub/StubConnection.php
index b553dcce0dc54868b4ac27b067d817360186d2c4..97658062d06c0ca82636fbf2b5e25c33954c5255 100644
--- a/core/tests/Drupal/Tests/Core/Database/Stub/StubConnection.php
+++ b/core/tests/Drupal/Tests/Core/Database/Stub/StubConnection.php
@@ -13,11 +13,6 @@
  */
 class StubConnection extends Connection {
 
-  /**
-   * {@inheritdoc}
-   */
-  protected $statementClass = NULL;
-
   /**
    * {@inheritdoc}
    */
@@ -40,15 +35,9 @@ class StubConnection extends Connection {
    *   An array of options for the connection.
    * @param string[]|null $identifier_quotes
    *   The identifier quote characters. Defaults to an empty strings.
-   * @param string|null $statement_class
-   *   A class to use as a statement class for deprecation testing.
    */
-  public function __construct(\PDO $connection, array $connection_options, $identifier_quotes = ['', ''], $statement_class = NULL) {
+  public function __construct(\PDO $connection, array $connection_options, $identifier_quotes = ['', '']) {
     $this->identifierQuotes = $identifier_quotes;
-    if ($statement_class) {
-      $this->statementClass = $statement_class;
-      $this->statementWrapperClass = NULL;
-    }
     parent::__construct($connection, $connection_options);
   }
 
@@ -59,14 +48,6 @@ public function queryRange($query, $from, $count, array $args = [], array $optio
     return NULL;
   }
 
-  /**
-   * {@inheritdoc}
-   */
-  public function queryTemporary($query, array $args = [], array $options = []) {
-    @trigger_error('Connection::queryTemporary() is deprecated in drupal:9.3.0 and is removed from drupal:10.0.0. There is no replacement. See https://www.drupal.org/node/3211781', E_USER_DEPRECATED);
-    return '';
-  }
-
   /**
    * {@inheritdoc}
    */
diff --git a/core/tests/fixtures/database_drivers/custom/fake/Connection.php b/core/tests/fixtures/database_drivers/custom/fake/Connection.php
index ec0c9cf2dc90728badbd9bfa19bedac89994a1d3..5d3312650dc3ee6f3ac85684315c506b4be8c860 100644
--- a/core/tests/fixtures/database_drivers/custom/fake/Connection.php
+++ b/core/tests/fixtures/database_drivers/custom/fake/Connection.php
@@ -24,14 +24,6 @@ public function queryRange($query, $from, $count, array $args = [], array $optio
     return NULL;
   }
 
-  /**
-   * {@inheritdoc}
-   */
-  public function queryTemporary($query, array $args = [], array $options = []) {
-    @trigger_error('Connection::queryTemporary() is deprecated in drupal:9.3.0 and is removed from drupal:10.0.0. There is no replacement. See https://www.drupal.org/node/3211781', E_USER_DEPRECATED);
-    return '';
-  }
-
   /**
    * {@inheritdoc}
    */