2017-03-02 03:47:51 +00:00
< ? php
2017-11-17 01:23:18 +00:00
/** @ license MIT
* Copyright 2017 J . King , Dustin Wilson et al .
* See LICENSE and AUTHORS files for details */
2017-03-02 03:47:51 +00:00
declare ( strict_types = 1 );
2021-04-14 15:17:01 +00:00
2018-11-09 19:58:10 +00:00
namespace JKingWeb\Arsse\TestCase\Db ;
2017-03-02 03:47:51 +00:00
2017-08-29 14:50:31 +00:00
use JKingWeb\Arsse\Db\Statement ;
2017-03-02 03:47:51 +00:00
2018-11-23 04:18:20 +00:00
abstract class BaseStatement extends \JKingWeb\Arsse\Test\AbstractTest {
2018-11-27 19:26:33 +00:00
protected static $interface ;
2018-11-23 04:18:20 +00:00
protected $statementClass ;
abstract protected function makeStatement ( string $q , array $types = []) : array ;
2024-12-28 00:58:07 +00:00
abstract protected static function decorateTypeSyntax ( string $value , string $type ) : string ;
2018-11-23 04:18:20 +00:00
2019-10-16 18:42:43 +00:00
public static function setUpBeforeClass () : void {
2018-11-27 19:26:33 +00:00
// establish a clean baseline
static :: clearData ();
static :: setConf ();
2019-01-12 17:43:06 +00:00
static :: $interface = static :: dbInterface ();
2018-11-27 19:26:33 +00:00
}
2020-03-01 20:16:50 +00:00
2019-10-16 18:42:43 +00:00
public function setUp () : void {
2021-02-27 20:24:02 +00:00
parent :: setUp ();
2018-11-23 04:18:20 +00:00
self :: setConf ();
2018-11-27 19:26:33 +00:00
if ( ! static :: $interface ) {
$this -> markTestSkipped ( static :: $implementation . " database driver not available " );
2018-11-21 18:06:01 +00:00
}
2018-11-27 19:26:33 +00:00
// completely clear the database
2019-01-12 17:43:06 +00:00
static :: dbRaze ( static :: $interface );
$this -> statementClass = static :: $dbStatementClass ;
2018-11-23 04:18:20 +00:00
}
2019-10-16 18:42:43 +00:00
public static function tearDownAfterClass () : void {
2018-12-10 18:17:04 +00:00
if ( static :: $interface ) {
// completely clear the database
2019-01-12 17:43:06 +00:00
static :: dbRaze ( static :: $interface );
2018-12-10 18:17:04 +00:00
}
static :: $interface = null ;
2021-02-27 20:24:02 +00:00
self :: clearData ( true );
2017-03-02 03:47:51 +00:00
}
2020-01-20 18:52:48 +00:00
public function testConstructStatement () : void {
2018-11-23 04:18:20 +00:00
$this -> assertInstanceOf ( Statement :: class , new $this -> statementClass ( ... $this -> makeStatement ( " SELECT ? as value " )));
2017-03-02 03:47:51 +00:00
}
2017-12-30 20:59:45 +00:00
/** @dataProvider provideBindings */
2020-01-20 18:52:48 +00:00
public function testBindATypedValue ( $value , string $type , string $exp ) : void {
2019-01-11 15:38:06 +00:00
if ( $exp === " null " ) {
2018-11-29 18:45:37 +00:00
$query = " SELECT (? is null) as pass " ;
2018-11-20 20:45:20 +00:00
} else {
$query = " SELECT ( $exp = ?) as pass " ;
}
2018-11-23 04:18:20 +00:00
$s = new $this -> statementClass ( ... $this -> makeStatement ( $query ));
2018-11-20 20:45:20 +00:00
$s -> retype ( ... [ $type ]);
$act = $s -> run ( ... [ $value ]) -> getValue ();
$this -> assertTrue (( bool ) $act );
2017-12-30 20:59:45 +00:00
}
2018-11-20 21:32:18 +00:00
/** @dataProvider provideBinaryBindings */
2020-01-20 18:52:48 +00:00
public function testHandleBinaryData ( $value , string $type , string $exp ) : void {
2019-01-11 15:38:06 +00:00
if ( $exp === " null " ) {
2018-11-29 18:45:37 +00:00
$query = " SELECT (? is null) as pass " ;
2018-11-20 21:32:18 +00:00
} else {
$query = " SELECT ( $exp = ?) as pass " ;
}
2018-11-23 04:18:20 +00:00
$s = new $this -> statementClass ( ... $this -> makeStatement ( $query ));
2018-11-20 21:32:18 +00:00
$s -> retype ( ... [ $type ]);
$act = $s -> run ( ... [ $value ]) -> getValue ();
$this -> assertTrue (( bool ) $act );
}
2020-01-20 18:52:48 +00:00
public function testBindMissingValue () : void {
2018-11-23 04:18:20 +00:00
$s = new $this -> statementClass ( ... $this -> makeStatement ( " SELECT ? as value " , [ " int " ]));
2017-04-07 01:41:21 +00:00
$val = $s -> runArray () -> getRow ()[ 'value' ];
$this -> assertSame ( null , $val );
}
2017-03-03 18:20:26 +00:00
2020-01-20 18:52:48 +00:00
public function testBindMultipleValues () : void {
2017-03-03 18:20:26 +00:00
$exp = [
2019-01-14 14:51:00 +00:00
'one' => " A " ,
'two' => " B " ,
2017-03-03 18:20:26 +00:00
];
2019-01-14 14:51:00 +00:00
$s = new $this -> statementClass ( ... $this -> makeStatement ( " SELECT ? as one, ? as two " , [ " str " , " str " ]));
$val = $s -> runArray ([ " A " , " B " ]) -> getRow ();
2017-04-07 01:41:21 +00:00
$this -> assertSame ( $exp , $val );
2017-03-03 18:20:26 +00:00
}
2020-01-20 18:52:48 +00:00
public function testBindRecursively () : void {
2017-04-07 01:24:26 +00:00
$exp = [
2019-01-14 14:51:00 +00:00
'one' => " A " ,
'two' => " B " ,
'three' => " C " ,
'four' => " D " ,
2017-04-07 01:24:26 +00:00
];
2019-01-14 14:51:00 +00:00
$s = new $this -> statementClass ( ... $this -> makeStatement ( " SELECT ? as one, ? as two, ? as three, ? as four " , [ " str " , [ " str " , " str " ], " str " ]));
$val = $s -> runArray ([ " A " , [ " B " , " C " ], " D " ]) -> getRow ();
2017-04-07 01:41:21 +00:00
$this -> assertSame ( $exp , $val );
}
2017-04-07 01:24:26 +00:00
2020-01-20 18:52:48 +00:00
public function testBindWithoutType () : void {
2017-04-07 01:41:21 +00:00
$this -> assertException ( " paramTypeMissing " , " Db " );
2018-11-23 04:18:20 +00:00
$s = new $this -> statementClass ( ... $this -> makeStatement ( " SELECT ? as value " , []));
2017-04-07 01:41:21 +00:00
$s -> runArray ([ 1 ]);
2017-03-03 18:20:26 +00:00
}
2017-03-08 18:59:20 +00:00
2020-01-20 18:52:48 +00:00
public function testViolateConstraint () : void {
2018-12-20 23:06:28 +00:00
( new $this -> statementClass ( ... $this -> makeStatement ( " CREATE TABLE if not exists arsse_meta( \" key \" varchar(255) primary key not null, value text) " ))) -> run ();
$s = new $this -> statementClass ( ... $this -> makeStatement ( " INSERT INTO arsse_meta( \" key \" ) values(?) " , [ " str " ]));
2017-04-07 01:41:21 +00:00
$this -> assertException ( " constraintViolation " , " Db " , " ExceptionInput " );
$s -> runArray ([ null ]);
}
2017-03-09 14:44:50 +00:00
2020-01-20 18:52:48 +00:00
public function testMismatchTypes () : void {
2018-11-23 04:18:20 +00:00
( new $this -> statementClass ( ... $this -> makeStatement ( " CREATE TABLE if not exists arsse_feeds(id integer primary key not null, url text not null) " ))) -> run ();
$s = new $this -> statementClass ( ... $this -> makeStatement ( " INSERT INTO arsse_feeds(id,url) values(?,?) " , [ " str " , " str " ]));
2017-04-07 01:41:21 +00:00
$this -> assertException ( " typeViolation " , " Db " , " ExceptionInput " );
2018-11-09 19:58:10 +00:00
$s -> runArray ([ 'ook' , 'eek' ]);
}
2024-12-28 00:58:07 +00:00
public static function provideBindings () : iterable {
2018-11-09 19:58:10 +00:00
$dateMutable = new \DateTime ( " Noon Today " , new \DateTimezone ( " America/Toronto " ));
$dateImmutable = new \DateTimeImmutable ( " Noon Today " , new \DateTimezone ( " America/Toronto " ));
$dateUTC = new \DateTime ( " @ " . $dateMutable -> getTimestamp (), new \DateTimezone ( " UTC " ));
$tests = [
2020-03-01 20:16:50 +00:00
'Null as integer' => [ null , " integer " , " null " ],
'Null as float' => [ null , " float " , " null " ],
'Null as string' => [ null , " string " , " null " ],
'Null as datetime' => [ null , " datetime " , " null " ],
'Null as boolean' => [ null , " boolean " , " null " ],
'Null as strict integer' => [ null , " strict integer " , " 0 " ],
'Null as strict float' => [ null , " strict float " , " 0.0 " ],
'Null as strict string' => [ null , " strict string " , " '' " ],
'Null as strict datetime' => [ null , " strict datetime " , " '0001-01-01 00:00:00' " ],
'Null as strict boolean' => [ null , " strict boolean " , " 0 " ],
'True as integer' => [ true , " integer " , " 1 " ],
'True as float' => [ true , " float " , " 1.0 " ],
'True as string' => [ true , " string " , " '1' " ],
'True as datetime' => [ true , " datetime " , " null " ],
'True as boolean' => [ true , " boolean " , " 1 " ],
'True as strict integer' => [ true , " strict integer " , " 1 " ],
'True as strict float' => [ true , " strict float " , " 1.0 " ],
'True as strict string' => [ true , " strict string " , " '1' " ],
'True as strict datetime' => [ true , " strict datetime " , " '0001-01-01 00:00:00' " ],
'True as strict boolean' => [ true , " strict boolean " , " 1 " ],
'False as integer' => [ false , " integer " , " 0 " ],
'False as float' => [ false , " float " , " 0.0 " ],
'False as string' => [ false , " string " , " '' " ],
'False as datetime' => [ false , " datetime " , " null " ],
'False as boolean' => [ false , " boolean " , " 0 " ],
'False as strict integer' => [ false , " strict integer " , " 0 " ],
'False as strict float' => [ false , " strict float " , " 0.0 " ],
'False as strict string' => [ false , " strict string " , " '' " ],
'False as strict datetime' => [ false , " strict datetime " , " '0001-01-01 00:00:00' " ],
'False as strict boolean' => [ false , " strict boolean " , " 0 " ],
'Integer as integer' => [ 2112 , " integer " , " 2112 " ],
'Integer as float' => [ 2112 , " float " , " 2112.0 " ],
'Integer as string' => [ 2112 , " string " , " '2112' " ],
'Integer as datetime' => [ 2112 , " datetime " , " '1970-01-01 00:35:12' " ],
'Integer as boolean' => [ 2112 , " boolean " , " 1 " ],
'Integer as strict integer' => [ 2112 , " strict integer " , " 2112 " ],
'Integer as strict float' => [ 2112 , " strict float " , " 2112.0 " ],
'Integer as strict string' => [ 2112 , " strict string " , " '2112' " ],
'Integer as strict datetime' => [ 2112 , " strict datetime " , " '1970-01-01 00:35:12' " ],
'Integer as strict boolean' => [ 2112 , " strict boolean " , " 1 " ],
'Integer zero as integer' => [ 0 , " integer " , " 0 " ],
'Integer zero as float' => [ 0 , " float " , " 0.0 " ],
'Integer zero as string' => [ 0 , " string " , " '0' " ],
'Integer zero as datetime' => [ 0 , " datetime " , " '1970-01-01 00:00:00' " ],
'Integer zero as boolean' => [ 0 , " boolean " , " 0 " ],
'Integer zero as strict integer' => [ 0 , " strict integer " , " 0 " ],
'Integer zero as strict float' => [ 0 , " strict float " , " 0.0 " ],
'Integer zero as strict string' => [ 0 , " strict string " , " '0' " ],
'Integer zero as strict datetime' => [ 0 , " strict datetime " , " '1970-01-01 00:00:00' " ],
'Integer zero as strict boolean' => [ 0 , " strict boolean " , " 0 " ],
'Float as integer' => [ 2112.5 , " integer " , " 2112 " ],
'Float as float' => [ 2112.5 , " float " , " 2112.5 " ],
'Float as string' => [ 2112.5 , " string " , " '2112.5' " ],
'Float as datetime' => [ 2112.5 , " datetime " , " '1970-01-01 00:35:12' " ],
'Float as boolean' => [ 2112.5 , " boolean " , " 1 " ],
'Float as strict integer' => [ 2112.5 , " strict integer " , " 2112 " ],
'Float as strict float' => [ 2112.5 , " strict float " , " 2112.5 " ],
'Float as strict string' => [ 2112.5 , " strict string " , " '2112.5' " ],
'Float as strict datetime' => [ 2112.5 , " strict datetime " , " '1970-01-01 00:35:12' " ],
'Float as strict boolean' => [ 2112.5 , " strict boolean " , " 1 " ],
'Float zero as integer' => [ 0.0 , " integer " , " 0 " ],
'Float zero as float' => [ 0.0 , " float " , " 0.0 " ],
'Float zero as string' => [ 0.0 , " string " , " '0' " ],
'Float zero as datetime' => [ 0.0 , " datetime " , " '1970-01-01 00:00:00' " ],
'Float zero as boolean' => [ 0.0 , " boolean " , " 0 " ],
'Float zero as strict integer' => [ 0.0 , " strict integer " , " 0 " ],
'Float zero as strict float' => [ 0.0 , " strict float " , " 0.0 " ],
'Float zero as strict string' => [ 0.0 , " strict string " , " '0' " ],
'Float zero as strict datetime' => [ 0.0 , " strict datetime " , " '1970-01-01 00:00:00' " ],
'Float zero as strict boolean' => [ 0.0 , " strict boolean " , " 0 " ],
'ASCII string as integer' => [ " Random string " , " integer " , " 0 " ],
'ASCII string as float' => [ " Random string " , " float " , " 0.0 " ],
'ASCII string as string' => [ " Random string " , " string " , " 'Random string' " ],
'ASCII string as datetime' => [ " Random string " , " datetime " , " null " ],
'ASCII string as boolean' => [ " Random string " , " boolean " , " 1 " ],
'ASCII string as strict integer' => [ " Random string " , " strict integer " , " 0 " ],
'ASCII string as strict float' => [ " Random string " , " strict float " , " 0.0 " ],
'ASCII string as strict string' => [ " Random string " , " strict string " , " 'Random string' " ],
'ASCII string as strict datetime' => [ " Random string " , " strict datetime " , " '0001-01-01 00:00:00' " ],
'ASCII string as strict boolean' => [ " Random string " , " strict boolean " , " 1 " ],
'UTF-8 string as integer' => [ " \ u { e9} " , " integer " , " 0 " ],
'UTF-8 string as float' => [ " \ u { e9} " , " float " , " 0.0 " ],
'UTF-8 string as string' => [ " \ u { e9} " , " string " , " char(233) " ],
'UTF-8 string as datetime' => [ " \ u { e9} " , " datetime " , " null " ],
'UTF-8 string as boolean' => [ " \ u { e9} " , " boolean " , " 1 " ],
'UTF-8 string as strict integer' => [ " \ u { e9} " , " strict integer " , " 0 " ],
'UTF-8 string as strict float' => [ " \ u { e9} " , " strict float " , " 0.0 " ],
'UTF-8 string as strict string' => [ " \ u { e9} " , " strict string " , " char(233) " ],
'UTF-8 string as strict datetime' => [ " \ u { e9} " , " strict datetime " , " '0001-01-01 00:00:00' " ],
'UTF-8 string as strict boolean' => [ " \ u { e9} " , " strict boolean " , " 1 " ],
'ISO 8601 string as integer' => [ " 2017-01-09T13:11:17 " , " integer " , " 0 " ],
'ISO 8601 string as float' => [ " 2017-01-09T13:11:17 " , " float " , " 0.0 " ],
'ISO 8601 string as string' => [ " 2017-01-09T13:11:17 " , " string " , " '2017-01-09T13:11:17' " ],
'ISO 8601 string as datetime' => [ " 2017-01-09T13:11:17 " , " datetime " , " '2017-01-09 13:11:17' " ],
'ISO 8601 string as boolean' => [ " 2017-01-09T13:11:17 " , " boolean " , " 1 " ],
'ISO 8601 string as strict integer' => [ " 2017-01-09T13:11:17 " , " strict integer " , " 0 " ],
'ISO 8601 string as strict float' => [ " 2017-01-09T13:11:17 " , " strict float " , " 0.0 " ],
'ISO 8601 string as strict string' => [ " 2017-01-09T13:11:17 " , " strict string " , " '2017-01-09T13:11:17' " ],
'ISO 8601 string as strict datetime' => [ " 2017-01-09T13:11:17 " , " strict datetime " , " '2017-01-09 13:11:17' " ],
'ISO 8601 string as strict boolean' => [ " 2017-01-09T13:11:17 " , " strict boolean " , " 1 " ],
'Arbitrary date string as integer' => [ " Today " , " integer " , " 0 " ],
'Arbitrary date string as float' => [ " Today " , " float " , " 0.0 " ],
'Arbitrary date string as string' => [ " Today " , " string " , " 'Today' " ],
'Arbitrary date string as datetime' => [ " Today " , " datetime " , " ' " . date_create ( " Today " , new \DateTimezone ( " UTC " )) -> format ( " Y-m-d H:i:s " ) . " ' " ],
'Arbitrary date string as boolean' => [ " Today " , " boolean " , " 1 " ],
'Arbitrary date string as strict integer' => [ " Today " , " strict integer " , " 0 " ],
'Arbitrary date string as strict float' => [ " Today " , " strict float " , " 0.0 " ],
'Arbitrary date string as strict string' => [ " Today " , " strict string " , " 'Today' " ],
2018-11-09 19:58:10 +00:00
'Arbitrary date string as strict datetime' => [ " Today " , " strict datetime " , " ' " . date_create ( " Today " , new \DateTimezone ( " UTC " )) -> format ( " Y-m-d H:i:s " ) . " ' " ],
2020-03-01 20:16:50 +00:00
'Arbitrary date string as strict boolean' => [ " Today " , " strict boolean " , " 1 " ],
'DateTime as integer' => [ $dateMutable , " integer " , ( string ) $dateUTC -> getTimestamp ()],
'DateTime as float' => [ $dateMutable , " float " , $dateUTC -> getTimestamp () . " .0 " ],
'DateTime as string' => [ $dateMutable , " string " , " ' " . $dateUTC -> format ( " Y-m-d H:i:s " ) . " ' " ],
'DateTime as datetime' => [ $dateMutable , " datetime " , " ' " . $dateUTC -> format ( " Y-m-d H:i:s " ) . " ' " ],
'DateTime as boolean' => [ $dateMutable , " boolean " , " 1 " ],
'DateTime as strict integer' => [ $dateMutable , " strict integer " , ( string ) $dateUTC -> getTimestamp ()],
'DateTime as strict float' => [ $dateMutable , " strict float " , $dateUTC -> getTimestamp () . " .0 " ],
'DateTime as strict string' => [ $dateMutable , " strict string " , " ' " . $dateUTC -> format ( " Y-m-d H:i:s " ) . " ' " ],
'DateTime as strict datetime' => [ $dateMutable , " strict datetime " , " ' " . $dateUTC -> format ( " Y-m-d H:i:s " ) . " ' " ],
'DateTime as strict boolean' => [ $dateMutable , " strict boolean " , " 1 " ],
'DateTimeImmutable as integer' => [ $dateImmutable , " integer " , ( string ) $dateUTC -> getTimestamp ()],
'DateTimeImmutable as float' => [ $dateImmutable , " float " , $dateUTC -> getTimestamp () . " .0 " ],
'DateTimeImmutable as string' => [ $dateImmutable , " string " , " ' " . $dateUTC -> format ( " Y-m-d H:i:s " ) . " ' " ],
'DateTimeImmutable as datetime' => [ $dateImmutable , " datetime " , " ' " . $dateUTC -> format ( " Y-m-d H:i:s " ) . " ' " ],
'DateTimeImmutable as boolean' => [ $dateImmutable , " boolean " , " 1 " ],
'DateTimeImmutable as strict integer' => [ $dateImmutable , " strict integer " , ( string ) $dateUTC -> getTimestamp ()],
'DateTimeImmutable as strict float' => [ $dateImmutable , " strict float " , $dateUTC -> getTimestamp () . " .0 " ],
'DateTimeImmutable as strict string' => [ $dateImmutable , " strict string " , " ' " . $dateUTC -> format ( " Y-m-d H:i:s " ) . " ' " ],
'DateTimeImmutable as strict datetime' => [ $dateImmutable , " strict datetime " , " ' " . $dateUTC -> format ( " Y-m-d H:i:s " ) . " ' " ],
'DateTimeImmutable as strict boolean' => [ $dateImmutable , " strict boolean " , " 1 " ],
2018-11-09 19:58:10 +00:00
];
2020-03-01 20:16:50 +00:00
foreach ( $tests as $index => [ $value , $type , $exp ]) {
2018-11-23 04:18:20 +00:00
$t = preg_replace ( " <^strict > " , " " , $type );
2024-12-28 00:58:07 +00:00
$exp = ( $exp === " null " ) ? $exp : static :: decorateTypeSyntax ( $exp , $t );
2018-11-23 04:18:20 +00:00
yield $index => [ $value , $type , $exp ];
2018-11-20 21:32:18 +00:00
}
}
2024-12-28 00:58:07 +00:00
public static function provideBinaryBindings () : iterable {
2018-11-20 21:32:18 +00:00
$dateMutable = new \DateTime ( " Noon Today " , new \DateTimezone ( " America/Toronto " ));
$dateImmutable = new \DateTimeImmutable ( " Noon Today " , new \DateTimezone ( " America/Toronto " ));
$dateUTC = new \DateTime ( " @ " . $dateMutable -> getTimestamp (), new \DateTimezone ( " UTC " ));
$tests = [
2020-03-01 20:16:50 +00:00
'Null as binary' => [ null , " binary " , " null " ],
'Null as strict binary' => [ null , " strict binary " , " x'' " ],
'True as binary' => [ true , " binary " , " x'31' " ],
'True as strict binary' => [ true , " strict binary " , " x'31' " ],
'False as binary' => [ false , " binary " , " x'' " ],
'False as strict binary' => [ false , " strict binary " , " x'' " ],
'Integer as binary' => [ 2112 , " binary " , " x'32313132' " ],
'Integer as strict binary' => [ 2112 , " strict binary " , " x'32313132' " ],
'Integer zero as binary' => [ 0 , " binary " , " x'30' " ],
'Integer zero as strict binary' => [ 0 , " strict binary " , " x'30' " ],
'Float as binary' => [ 2112.5 , " binary " , " x'323131322e35' " ],
'Float as strict binary' => [ 2112.5 , " strict binary " , " x'323131322e35' " ],
'Float zero as binary' => [ 0.0 , " binary " , " x'30' " ],
'Float zero as strict binary' => [ 0.0 , " strict binary " , " x'30' " ],
'ASCII string as binary' => [ " Random string " , " binary " , " x'52616e646f6d20737472696e67' " ],
'ASCII string as strict binary' => [ " Random string " , " strict binary " , " x'52616e646f6d20737472696e67' " ],
'UTF-8 string as binary' => [ " \ u { e9} " , " binary " , " x'c3a9' " ],
'UTF-8 string as strict binary' => [ " \ u { e9} " , " strict binary " , " x'c3a9' " ],
'Binary string as integer' => [ chr ( 233 ) . chr ( 233 ), " integer " , " 0 " ],
'Binary string as float' => [ chr ( 233 ) . chr ( 233 ), " float " , " 0.0 " ],
'Binary string as binary' => [ chr ( 233 ) . chr ( 233 ), " binary " , " x'e9e9' " ],
'Binary string as datetime' => [ chr ( 233 ) . chr ( 233 ), " datetime " , " null " ],
'Binary string as boolean' => [ chr ( 233 ) . chr ( 233 ), " boolean " , " 1 " ],
'Binary string as strict integer' => [ chr ( 233 ) . chr ( 233 ), " strict integer " , " 0 " ],
'Binary string as strict float' => [ chr ( 233 ) . chr ( 233 ), " strict float " , " 0.0 " ],
'Binary string as strict binary' => [ chr ( 233 ) . chr ( 233 ), " strict binary " , " x'e9e9' " ],
'Binary string as strict datetime' => [ chr ( 233 ) . chr ( 233 ), " strict datetime " , " '0001-01-01 00:00:00' " ],
'Binary string as strict boolean' => [ chr ( 233 ) . chr ( 233 ), " strict boolean " , " 1 " ],
'ISO 8601 string as binary' => [ " 2017-01-09T13:11:17 " , " binary " , " x'323031372d30312d30395431333a31313a3137' " ],
'ISO 8601 string as strict binary' => [ " 2017-01-09T13:11:17 " , " strict binary " , " x'323031372d30312d30395431333a31313a3137' " ],
'Arbitrary date string as binary' => [ " Today " , " binary " , " x'546f646179' " ],
2018-11-20 21:32:18 +00:00
'Arbitrary date string as strict binary' => [ " Today " , " strict binary " , " x'546f646179' " ],
2020-03-01 20:16:50 +00:00
'DateTime as binary' => [ $dateMutable , " binary " , " x' " . bin2hex ( $dateUTC -> format ( " Y-m-d H:i:s " )) . " ' " ],
'DateTime as strict binary' => [ $dateMutable , " strict binary " , " x' " . bin2hex ( $dateUTC -> format ( " Y-m-d H:i:s " )) . " ' " ],
'DateTimeImmutable as binary' => [ $dateImmutable , " binary " , " x' " . bin2hex ( $dateUTC -> format ( " Y-m-d H:i:s " )) . " ' " ],
'DateTimeImmutable as strict binary' => [ $dateImmutable , " strict binary " , " x' " . bin2hex ( $dateUTC -> format ( " Y-m-d H:i:s " )) . " ' " ],
2018-11-20 21:32:18 +00:00
];
2020-03-01 20:16:50 +00:00
foreach ( $tests as $index => [ $value , $type , $exp ]) {
2018-11-23 04:18:20 +00:00
$t = preg_replace ( " <^strict > " , " " , $type );
2024-12-28 00:58:07 +00:00
$exp = ( $exp === " null " ) ? $exp : static :: decorateTypeSyntax ( $exp , $t );
2018-11-23 04:18:20 +00:00
yield $index => [ $value , $type , $exp ];
2018-11-09 19:58:10 +00:00
}
2017-04-07 01:41:21 +00:00
}
2017-08-29 14:50:31 +00:00
}