2021-08-08 10:39:54 +02:00
< ? php
2021-12-15 04:58:32 +01:00
2021-09-23 19:29:15 +02:00
namespace Psalm\Tests\Internal\Codebase ;
2021-08-08 10:39:54 +02:00
2022-06-22 12:59:47 +02:00
use InvalidArgumentException ;
use PHPUnit\Framework\ExpectationFailedException ;
2022-06-22 15:05:24 +02:00
use PHPUnit\Framework\SkippedTestError ;
2022-06-15 10:56:15 +02:00
use Psalm\Codebase ;
2022-06-13 15:10:23 +02:00
use Psalm\Internal\Analyzer\ProjectAnalyzer ;
2021-08-08 10:39:54 +02:00
use Psalm\Internal\Codebase\InternalCallMapHandler ;
2022-06-15 09:27:40 +02:00
use Psalm\Internal\Codebase\Reflection ;
2022-06-13 15:10:23 +02:00
use Psalm\Internal\Provider\FakeFileProvider ;
use Psalm\Internal\Provider\Providers ;
2022-06-15 09:27:40 +02:00
use Psalm\Internal\Type\Comparator\UnionTypeComparator ;
2022-06-13 15:10:23 +02:00
use Psalm\Tests\Internal\Provider\FakeParserCacheProvider ;
2021-12-03 20:11:20 +01:00
use Psalm\Tests\TestCase ;
2022-06-13 15:10:23 +02:00
use Psalm\Tests\TestConfig ;
2022-06-15 09:27:40 +02:00
use Psalm\Type ;
2022-06-13 15:10:23 +02:00
use ReflectionFunction ;
use ReflectionParameter ;
2022-06-15 10:56:15 +02:00
use ReflectionType ;
2022-06-15 13:28:09 +02:00
2022-06-22 15:05:24 +02:00
use function class_exists ;
2022-06-15 13:28:09 +02:00
use function count ;
use function explode ;
use function function_exists ;
use function in_array ;
2022-06-22 15:37:51 +02:00
use function is_array ;
2022-06-22 15:32:35 +02:00
use function is_int ;
2022-06-15 13:28:09 +02:00
use function json_encode ;
use function preg_match ;
use function print_r ;
2022-06-22 15:32:35 +02:00
use function strcmp ;
2022-06-15 13:28:09 +02:00
use function strncmp ;
use function strpos ;
use function substr ;
2021-08-08 10:39:54 +02:00
2022-06-22 15:32:35 +02:00
use const PHP_MAJOR_VERSION ;
use const PHP_MINOR_VERSION ;
2021-12-03 20:11:20 +01:00
class InternalCallMapHandlerTest extends TestCase
2021-08-08 10:39:54 +02:00
{
2022-06-15 13:16:24 +02:00
/**
2022-06-22 15:32:35 +02:00
* Specify a function name as value , or a function name as key and
* an array containing the PHP versions in which to ignore this function as values .
* @ var array < int | string , string | list < string >>
2022-06-15 13:16:24 +02:00
*/
2022-06-15 10:56:15 +02:00
private static $ignoredFunctions = [
2022-06-22 14:57:40 +02:00
'apcu_entry' ,
2022-06-22 13:28:17 +02:00
'array_multisort' ,
'bcdiv' ,
2022-06-22 12:59:47 +02:00
'bcmod' ,
2022-06-22 13:28:17 +02:00
'bcpowmod' ,
'bzdecompress' ,
'count' ,
'crypt' ,
'date_isodate_set' ,
'debug_zval_dump' ,
'deflate_add' ,
'dns_get_mx' ,
'easter_date' ,
2022-06-22 14:57:40 +02:00
'enchant_broker_describe' ,
'enchant_broker_dict_exists' ,
'enchant_broker_free' ,
'enchant_broker_free_dict' ,
'enchant_broker_get_dict_path' ,
'enchant_broker_get_error' ,
'enchant_broker_list_dicts' ,
'enchant_broker_request_dict' ,
'enchant_broker_request_pwl_dict' ,
'enchant_broker_set_dict_path' ,
'enchant_broker_set_ordering' ,
'enchant_dict_add_to_personal' ,
'enchant_dict_add_to_session' ,
'enchant_dict_check' ,
'enchant_dict_describe' ,
'enchant_dict_get_error' ,
'enchant_dict_is_in_session' ,
'enchant_dict_quick_check' ,
'enchant_dict_store_replacement' ,
'enchant_dict_suggest' ,
2022-06-22 13:28:17 +02:00
'enum_exists' ,
'extract' ,
2022-06-22 15:32:35 +02:00
// https://www.php.net/manual/en/function.fputcsv.php
'fputcsv' => [ '8.1' ],
2022-06-22 13:28:17 +02:00
'get_class_methods' ,
'get_headers' ,
'get_parent_class' ,
2022-06-22 14:57:40 +02:00
'gmp_clrbit' ,
'gmp_div' ,
'gmp_setbit' ,
'gnupg_adddecryptkey' ,
'gnupg_addencryptkey' ,
'gnupg_addsignkey' ,
'gnupg_cleardecryptkeys' ,
'gnupg_clearencryptkeys' ,
'gnupg_clearsignkeys' ,
'gnupg_decrypt' ,
'gnupg_decryptverify' ,
'gnupg_encrypt' ,
'gnupg_encryptsign' ,
'gnupg_export' ,
'gnupg_geterror' ,
'gnupg_getprotocol' ,
'gnupg_import' ,
'gnupg_init' ,
'gnupg_keyinfo' ,
'gnupg_setarmor' ,
'gnupg_seterrormode' ,
'gnupg_setsignmode' ,
'gnupg_sign' ,
'gnupg_verify' ,
2022-06-22 13:28:17 +02:00
'hash_hmac_file' ,
'igbinary_unserialize' ,
2022-06-22 14:57:40 +02:00
'imagefilledpolygon' ,
2022-06-22 13:28:17 +02:00
'imagefilter' ,
'imagegd' ,
'imagegd2' ,
'imageinterlace' ,
'imageopenpolygon' ,
'imagepolygon' ,
'imagerotate' ,
'imagesetinterpolation' ,
'imagettfbbox' ,
2022-06-22 12:59:47 +02:00
'imagettftext' ,
'imagexbm' ,
2022-06-22 13:28:17 +02:00
'imap_delete' ,
'imap_open' ,
'imap_rfc822_write_address' ,
'imap_sort' ,
2022-06-22 12:59:47 +02:00
'imap_undelete' ,
2022-06-22 13:28:17 +02:00
'inflate_add' ,
'inflate_get_read_len' ,
'inflate_get_status' ,
'inotify_rm_watch' ,
2022-06-22 14:57:40 +02:00
'intlcal_from_date_time' ,
2022-06-22 13:28:17 +02:00
'intlcal_get_weekend_transition' ,
2022-06-22 12:59:47 +02:00
'intlgregcal_create_instance' ,
'intlgregcal_is_leap_year' ,
'intltz_create_enumeration' ,
2022-06-22 13:28:17 +02:00
'intltz_get_canonical_id' ,
'intltz_get_display_name' ,
'long2ip' ,
'lzf_compress' ,
'lzf_decompress' ,
'mail' ,
2022-06-22 14:57:40 +02:00
'mailparse_msg_extract_part' ,
'mailparse_msg_extract_part_file' ,
'mailparse_msg_extract_whole_part_file' ,
'mailparse_msg_free' ,
'mailparse_msg_get_part' ,
'mailparse_msg_get_part_data' ,
'mailparse_msg_get_structure' ,
'mailparse_msg_parse' ,
'mailparse_stream_encode' ,
'memcache_add' ,
'memcache_add_server' ,
'memcache_append' ,
'memcache_cas' ,
'memcache_close' ,
'memcache_connect' ,
'memcache_decrement' ,
'memcache_delete' ,
'memcache_flush' ,
'memcache_get_extended_stats' ,
'memcache_get_server_status' ,
'memcache_get_stats' ,
'memcache_get_version' ,
'memcache_increment' ,
'memcache_pconnect' ,
'memcache_prepend' ,
'memcache_replace' ,
'memcache_set' ,
'memcache_set_compress_threshold' ,
'memcache_set_failure_callback' ,
'memcache_set_server_params' ,
2022-06-22 13:28:17 +02:00
'mongodb\bson\tophp' ,
'msg_receive' ,
'msg_remove_queue' ,
'msg_send' ,
'msg_set_queue' ,
'msg_stat_queue' ,
'mysqli_poll' ,
'mysqli_real_connect' ,
'mysqli_stmt_bind_param' ,
'normalizer_get_raw_decomposition' ,
2022-06-22 14:57:40 +02:00
'oauth_get_sbs' ,
'oci_collection_append' ,
'oci_collection_assign' ,
'oci_collection_element_assign' ,
'oci_collection_element_get' ,
'oci_collection_max' ,
'oci_collection_size' ,
'oci_collection_trim' ,
'oci_fetch_object' ,
'oci_field_is_null' ,
'oci_field_name' ,
'oci_field_precision' ,
'oci_field_scale' ,
'oci_field_size' ,
'oci_field_type' ,
'oci_field_type_raw' ,
'oci_free_collection' ,
'oci_free_descriptor' ,
'oci_lob_append' ,
'oci_lob_eof' ,
'oci_lob_erase' ,
'oci_lob_export' ,
'oci_lob_flush' ,
'oci_lob_import' ,
'oci_lob_load' ,
'oci_lob_read' ,
'oci_lob_rewind' ,
'oci_lob_save' ,
'oci_lob_seek' ,
'oci_lob_size' ,
'oci_lob_tell' ,
'oci_lob_truncate' ,
'oci_lob_write' ,
'oci_register_taf_callback' ,
'oci_result' ,
'ocigetbufferinglob' ,
'ocisetbufferinglob' ,
'odbc_procedurecolumns' ,
'odbc_procedures' ,
'odbc_result' ,
2022-06-22 13:28:17 +02:00
'openssl_pkcs7_read' ,
'pg_exec' ,
'pg_fetch_all' ,
'pg_get_notify' ,
'pg_get_result' ,
'pg_pconnect' ,
'pg_select' ,
'pg_send_execute' ,
'preg_filter' ,
'preg_replace_callback_array' ,
'sapi_windows_cp_get' ,
'sem_acquire' ,
'sem_get' ,
'sem_release' ,
'sem_remove' ,
'shm_detach' ,
'shm_get_var' ,
'shm_has_var' ,
'shm_put_var' ,
'shm_remove' ,
'shm_remove_var' ,
'shmop_close' ,
'shmop_delete' ,
'shmop_read' ,
'shmop_size' ,
'shmop_write' ,
2022-06-22 14:57:40 +02:00
'snmp_set_enum_print' ,
'snmp_set_valueretrieval' ,
2022-06-22 13:28:17 +02:00
'snmpset' ,
2022-06-22 12:59:47 +02:00
'socket_addrinfo_lookup' ,
2022-06-22 13:28:17 +02:00
'socket_bind' ,
'socket_cmsg_space' ,
'socket_connect' ,
'socket_create_pair' ,
'socket_get_option' ,
'socket_getopt' ,
'socket_getpeername' ,
'socket_getsockname' ,
'socket_read' ,
2022-06-22 12:59:47 +02:00
'socket_recv' ,
2022-06-22 13:28:17 +02:00
'socket_recvfrom' ,
'socket_recvmsg' ,
'socket_select' ,
2022-06-22 12:59:47 +02:00
'socket_send' ,
2022-06-22 13:28:17 +02:00
'socket_sendmsg' ,
'socket_sendto' ,
'socket_set_blocking' ,
'socket_set_option' ,
'socket_setopt' ,
'socket_shutdown' ,
'socket_strerror' ,
'sodium_crypto_generichash' ,
'sodium_crypto_generichash_final' ,
'sodium_crypto_generichash_init' ,
'sodium_crypto_generichash_update' ,
'sodium_crypto_kx_client_session_keys' ,
'sodium_crypto_secretstream_xchacha20poly1305_rekey' ,
2022-06-22 14:57:40 +02:00
'sqlsrv_connect' ,
'sqlsrv_errors' ,
'sqlsrv_fetch_array' ,
'sqlsrv_fetch_object' ,
'sqlsrv_get_field' ,
'sqlsrv_prepare' ,
'sqlsrv_query' ,
'sqlsrv_server_info' ,
'stomp_abort' ,
'stomp_ack' ,
'stomp_begin' ,
'stomp_commit' ,
'stomp_read_frame' ,
'stomp_send' ,
'stomp_set_read_timeout' ,
'stomp_subscribe' ,
'stomp_unsubscribe' ,
2022-06-22 15:32:35 +02:00
'stream_select' => [ '8.0' ],
2022-06-22 13:28:17 +02:00
'substr_replace' ,
2022-06-22 14:57:40 +02:00
'tidy_getopt' ,
'uopz_allow_exit' ,
'uopz_get_mock' ,
'uopz_get_property' ,
'uopz_get_return' ,
'uopz_get_static' ,
'uopz_set_mock' ,
'uopz_set_property' ,
'uopz_set_static' ,
'uopz_unset_mock' ,
'xdiff_file_bdiff' ,
'xdiff_file_bdiff_size' ,
'xdiff_file_diff' ,
'xdiff_file_diff_binary' ,
'xdiff_file_merge3' ,
'xdiff_file_rabdiff' ,
'xdiff_string_bdiff' ,
'xdiff_string_bdiff_size' ,
'xdiff_string_bpatch' ,
'xdiff_string_diff' ,
'xdiff_string_diff_binary' ,
'xdiff_string_merge3' ,
'xdiff_string_patch' ,
'xdiff_string_patch_binary' ,
'xdiff_string_rabdiff' ,
'xmlrpc_server_add_introspection_data' ,
'xmlrpc_server_call_method' ,
'xmlrpc_server_destroy' ,
'xmlrpc_server_register_introspection_callback' ,
'xmlrpc_server_register_method' ,
'yaml_emit' ,
'yaml_emit_file' ,
2022-06-22 13:28:17 +02:00
'zip_entry_close' ,
'zlib_encode' ,
2022-06-14 15:16:13 +02:00
];
2022-06-14 15:43:13 +02:00
2022-06-15 13:16:24 +02:00
/**
*
* @ var Codebase
*/
2022-06-15 10:56:15 +02:00
private static $codebase ;
2022-06-15 13:13:24 +02:00
2022-06-15 10:56:15 +02:00
public static function setUpBeforeClass () : void
{
$project_analyzer = new ProjectAnalyzer (
new TestConfig (),
new Providers (
new FakeFileProvider (),
new FakeParserCacheProvider ()
)
);
self :: $codebase = $project_analyzer -> getCodebase ();
}
2022-06-22 14:57:40 +02:00
2022-06-22 15:32:35 +02:00
public function testIgnoresAreSortedAndUnique () : void
2022-06-22 14:57:40 +02:00
{
2022-06-22 15:32:35 +02:00
$previousFunction = " " ;
foreach ( self :: $ignoredFunctions as $key => $value ) {
2022-06-22 15:37:51 +02:00
/** @var string */
2022-06-22 15:32:35 +02:00
$function = is_int ( $key ) ? $value : $key ;
$this -> assertGreaterThan ( 0 , strcmp ( $function , $previousFunction ));
$previousFunction = $function ;
}
2022-06-22 14:57:40 +02:00
}
2022-06-15 10:56:15 +02:00
public static function tearDownAfterClass () : void
{
self :: $codebase = null ;
}
2022-06-14 15:16:13 +02:00
2021-08-08 10:39:54 +02:00
/**
2022-01-11 16:45:29 +01:00
* @ covers \Psalm\Internal\Codebase\InternalCallMapHandler :: getCallMap
2021-08-08 10:39:54 +02:00
*/
public function testGetcallmapReturnsAValidCallmap () : void
{
$callMap = InternalCallMapHandler :: getCallMap ();
self :: assertArrayKeysAreStrings ( $callMap , " Returned CallMap has non-string keys " );
self :: assertArrayValuesAreArrays ( $callMap , " Returned CallMap has non-array values " );
foreach ( $callMap as $function => $signature ) {
self :: assertArrayKeysAreZeroOrString ( $signature , " Function " . $function . " in returned CallMap has invalid keys " );
self :: assertArrayValuesAreStrings ( $signature , " Function " . $function . " in returned CallMap has non-string values " );
foreach ( $signature as $type ) {
self :: assertStringIsParsableType ( $type , " Function " . $function . " in returned CallMap contains invalid type declaration " . $type );
}
}
}
2022-06-13 15:10:23 +02:00
2022-06-15 13:23:32 +02:00
/**
*
2022-06-22 13:28:17 +02:00
* @ return iterable < string , array { 0 : callable - string , 1 : array < int | string > } >
2022-06-15 13:23:32 +02:00
*/
2022-06-13 15:10:23 +02:00
public function callMapEntryProvider () : iterable
{
2022-06-15 14:46:50 +02:00
/**
* This call is needed since InternalCallMapHandler uses the singleton that is initialized by it .
**/
new ProjectAnalyzer (
2022-06-13 15:10:23 +02:00
new TestConfig (),
new Providers (
new FakeFileProvider (),
new FakeParserCacheProvider ()
)
);
$callMap = InternalCallMapHandler :: getCallMap ();
2022-06-15 13:28:09 +02:00
foreach ( $callMap as $function => $entry ) {
2022-06-13 15:10:23 +02:00
// Skip class methods
2022-06-15 09:11:14 +02:00
if ( strpos ( $function , '::' ) !== false || ! function_exists ( $function )) {
continue ;
}
2022-06-13 15:29:21 +02:00
// Skip functions with alternate signatures
2022-06-14 15:16:13 +02:00
if ( isset ( $callMap [ " $function '1 " ]) || preg_match ( " / \ ' \ d $ / " , $function )) {
2022-06-13 15:10:23 +02:00
continue ;
}
2022-06-15 10:56:15 +02:00
// if ($function != 'fprintf') continue;
2022-06-13 15:10:23 +02:00
yield " $function : " . json_encode ( $entry ) => [ $function , $entry ];
}
}
2022-06-22 12:59:47 +02:00
/**
*/
2022-06-22 15:05:24 +02:00
private function isIgnored ( string $functionName ) : bool
2022-06-22 12:59:47 +02:00
{
if ( in_array ( $functionName , self :: $ignoredFunctions )) {
return true ;
}
2022-06-22 15:32:35 +02:00
2022-06-22 15:37:51 +02:00
if ( isset ( self :: $ignoredFunctions [ $functionName ])
&& is_array ( self :: $ignoredFunctions [ $functionName ])
&& in_array ( PHP_MAJOR_VERSION . '.' . PHP_MINOR_VERSION , self :: $ignoredFunctions [ $functionName ])) {
2022-06-22 15:32:35 +02:00
return true ;
}
2022-06-22 12:59:47 +02:00
return false ;
}
/**
2022-06-22 15:32:35 +02:00
* @ depends testIgnoresAreSortedAndUnique
2022-06-22 13:28:17 +02:00
* @ depends testGetcallmapReturnsAValidCallmap
2022-06-22 12:59:47 +02:00
* @ dataProvider callMapEntryProvider
2022-06-22 13:28:17 +02:00
* @ coversNothing
* @ psalm - param callable - string $functionName
2022-06-22 12:59:47 +02:00
*/
public function testIgnoredFunctionsStillFail ( string $functionName , array $callMapEntry ) : void
{
if ( ! $this -> isIgnored ( $functionName )) {
// Dummy assertion to mark it as passed
$this -> assertTrue ( true );
return ;
}
$this -> expectException ( ExpectationFailedException :: class );
try {
unset ( $callMapEntry [ 0 ]);
/** @var array<string, string> $callMapEntry */
$this -> assertEntryIsCorrect ( $callMapEntry , $functionName );
2022-06-22 15:05:24 +02:00
} catch ( InvalidArgumentException $t ) {
2022-06-22 12:59:47 +02:00
// Silence this one for now.
$this -> markTestSkipped ( 'IA' );
2022-06-22 15:05:24 +02:00
} catch ( SkippedTestError $t ) {
2022-06-22 12:59:47 +02:00
die ( 'this should not happen' );
2022-06-22 15:05:24 +02:00
} catch ( ExpectationFailedException $e ) {
2022-06-22 12:59:47 +02:00
// This is good!
throw $e ;
2022-06-22 15:05:24 +02:00
} catch ( InvalidArgumentException $e ) {
2022-06-22 12:59:47 +02:00
// This can happen if a class does not exist, we handle the message to check for this case.
if ( preg_match ( '/^Could not get class storage for (.*)$/' , $e -> getMessage (), $matches )
&& ! class_exists ( $matches [ 1 ])
) {
die ( " Class mentioned in callmap does not exist: " . $matches [ 1 ]);
}
}
$this -> markTestIncomplete ( " Remove function ' { $functionName } ' from your ignores " );
}
2022-06-13 15:10:23 +02:00
/**
* This function will test functions that are in the callmap AND currently defined
* @ coversNothing
* @ depends testGetcallmapReturnsAValidCallmap
2022-06-22 15:32:35 +02:00
* @ depends testIgnoresAreSortedAndUnique
2022-06-13 15:10:23 +02:00
* @ dataProvider callMapEntryProvider
2022-06-22 13:28:17 +02:00
* @ psalm - param callable - string $functionName
2022-06-15 15:05:06 +02:00
* @ param array $callMapEntry
2022-06-13 15:10:23 +02:00
*/
public function testCallMapCompliesWithReflection ( string $functionName , array $callMapEntry ) : void
{
2022-06-22 12:59:47 +02:00
if ( $this -> isIgnored ( $functionName )) {
2022-06-15 10:56:15 +02:00
$this -> markTestSkipped ( " Function $functionName is ignored in config " );
}
2022-06-15 15:05:06 +02:00
unset ( $callMapEntry [ 0 ]);
/** @var array<string, string> $callMapEntry */
2022-06-13 15:10:23 +02:00
$this -> assertEntryIsCorrect ( $callMapEntry , $functionName );
}
2022-06-15 14:46:50 +02:00
/**
*
2022-06-15 15:05:06 +02:00
* @ param array < string , string > $callMapEntryWithoutReturn
* @ psalm - param callable - string $functionName
2022-06-15 14:46:50 +02:00
*/
2022-06-15 15:05:06 +02:00
private function assertEntryIsCorrect ( array $callMapEntryWithoutReturn , string $functionName ) : void
2022-06-13 15:10:23 +02:00
{
$rF = new ReflectionFunction ( $functionName );
/**
* Parse the parameter names from the map .
2022-06-15 15:05:06 +02:00
* @ var array < string , array { byRef : bool , refMode : 'rw' | 'w' , variadic : bool , optional : bool , type : string } >
2022-06-13 15:10:23 +02:00
*/
$normalizedEntries = [];
2022-06-15 15:05:06 +02:00
foreach ( $callMapEntryWithoutReturn as $key => $entry ) {
2022-06-13 15:10:23 +02:00
$normalizedKey = $key ;
2022-06-15 15:05:06 +02:00
/**
*
* @ var array { byRef : bool , refMode : 'rw' | 'w' , variadic : bool , optional : bool , type : string } $normalizedEntry
*/
2022-06-13 15:10:23 +02:00
$normalizedEntry = [
'variadic' => false ,
'byRef' => false ,
'optional' => false ,
'type' => $entry ,
];
2022-06-15 10:56:15 +02:00
if ( strncmp ( $normalizedKey , '&' , 1 ) === 0 ) {
2022-06-13 15:10:23 +02:00
$normalizedEntry [ 'byRef' ] = true ;
2022-06-15 10:56:15 +02:00
$normalizedKey = substr ( $normalizedKey , 1 );
2022-06-13 15:10:23 +02:00
}
2022-06-15 10:56:15 +02:00
2022-06-13 15:10:23 +02:00
if ( strncmp ( $normalizedKey , '...' , 3 ) === 0 ) {
$normalizedEntry [ 'variadic' ] = true ;
$normalizedKey = substr ( $normalizedKey , 3 );
}
2022-06-15 10:56:15 +02:00
// Read the reference mode
if ( $normalizedEntry [ 'byRef' ]) {
$parts = explode ( '_' , $normalizedKey , 2 );
if ( count ( $parts ) === 2 ) {
$normalizedEntry [ 'refMode' ] = $parts [ 0 ];
$normalizedKey = $parts [ 1 ];
} else {
$normalizedEntry [ 'refMode' ] = 'rw' ;
}
}
// Strip prefixes.
2022-06-13 15:10:23 +02:00
if ( substr ( $normalizedKey , - 1 , 1 ) === " = " ) {
$normalizedEntry [ 'optional' ] = true ;
$normalizedKey = substr ( $normalizedKey , 0 , - 1 );
}
2022-06-15 10:56:15 +02:00
$normalizedEntry [ 'name' ] = $normalizedKey ;
2022-06-13 15:10:23 +02:00
$normalizedEntries [ $normalizedKey ] = $normalizedEntry ;
}
2022-06-15 13:28:09 +02:00
foreach ( $rF -> getParameters () as $parameter ) {
2022-06-22 14:57:40 +02:00
$this -> assertArrayHasKey ( $parameter -> getName (), $normalizedEntries , " Callmap is missing entry for param { $parameter -> getName () } in $functionName : " . print_r ( $normalizedEntries , true ));
2022-06-15 15:05:06 +02:00
$this -> assertParameter ( $normalizedEntries [ $parameter -> getName ()], $parameter );
2022-06-13 15:10:23 +02:00
}
}
/**
*
* @ param array { byRef : bool , refMode : 'rw' | 'w' , variadic : bool , optional : bool , type : string } $normalizedEntry
*/
2022-06-15 15:05:06 +02:00
private function assertParameter ( array $normalizedEntry , ReflectionParameter $param ) : void
2022-06-13 15:10:23 +02:00
{
$name = $param -> getName ();
2022-06-22 12:59:47 +02:00
$this -> assertSame ( $param -> isOptional (), $normalizedEntry [ 'optional' ], " Expected param ' { $name } ' to " . ( $param -> isOptional () ? " be " : " not be " ) . " optional " );
$this -> assertSame ( $param -> isVariadic (), $normalizedEntry [ 'variadic' ], " Expected param ' { $name } ' to " . ( $param -> isVariadic () ? " be " : " not be " ) . " variadic " );
$this -> assertSame ( $param -> isPassedByReference (), $normalizedEntry [ 'byRef' ], " Expected param ' { $name } ' to " . ( $param -> isPassedByReference () ? " be " : " not be " ) . " by reference " );
2022-06-13 15:10:23 +02:00
$expectedType = $param -> getType ();
2022-06-14 15:16:13 +02:00
2022-06-22 13:28:17 +02:00
if ( isset ( $expectedType ) && ! empty ( $normalizedEntry [ 'type' ])) {
2022-06-13 15:10:23 +02:00
$this -> assertTypeValidity ( $expectedType , $normalizedEntry [ 'type' ], " Param ' { $name } ' has incorrect type " );
}
}
/**
* Since string equality is too strict , we do some extra checking here
*/
2022-06-15 10:56:15 +02:00
private function assertTypeValidity ( ReflectionType $reflected , string $specified , string $message ) : void
2022-06-13 15:10:23 +02:00
{
2022-06-15 09:27:40 +02:00
$expectedType = Reflection :: getPsalmTypeFromReflectionType ( $reflected );
2022-06-22 12:59:47 +02:00
2022-06-22 15:08:50 +02:00
$parsedType = Type :: parseString ( $specified );
2022-06-15 09:27:40 +02:00
2022-06-22 12:59:47 +02:00
try {
2022-06-22 15:05:24 +02:00
$this -> assertTrue ( UnionTypeComparator :: isContainedBy ( self :: $codebase , $parsedType , $expectedType ), $message );
} catch ( InvalidArgumentException $e ) {
2022-06-22 12:59:47 +02:00
if ( preg_match ( '/^Could not get class storage for (.*)$/' , $e -> getMessage (), $matches )
&& ! class_exists ( $matches [ 1 ])
) {
die ( " Class mentioned in callmap does not exist: " . $matches [ 1 ]);
}
}
2022-06-13 15:10:23 +02:00
}
2021-08-08 10:39:54 +02:00
}