Duplicate code is one of the most pungent code smells. A rule that is often used is to re-structure code once it is duplicated in three or more places.
Common duplication problems, and corresponding solutions are:
| 1 | <?php // phpcs:ignore WordPress.Files.FileName.NotHyphenatedLowercase | ||
| 18 | class ManagerTest extends TestCase { | ||
| 19 | |||
| 20 | /** | ||
| 21 | * Temporary stack for `wp_redirect`. | ||
| 22 | * | ||
| 23 | * @var array | ||
| 24 | */ | ||
| 25 | protected $arguments_stack = array(); | ||
| 26 | |||
| 27 | /** | ||
| 28 | * User ID added for the test. | ||
| 29 | * | ||
| 30 | * @var int | ||
| 31 | */ | ||
| 32 | protected $user_id; | ||
| 33 | |||
| 34 | const DEFAULT_TEST_CAPS = array( 'default_test_caps' ); | ||
| 35 | |||
| 36 | /** | ||
| 37 | * Initialize the object before running the test method. | ||
| 38 | * | ||
| 39 | * @before | ||
| 40 | */ | ||
| 41 | 	public function set_up() { | ||
| 60 | |||
| 61 | /** | ||
| 62 | * Clean up the testing environment. | ||
| 63 | * | ||
| 64 | * @after | ||
| 65 | */ | ||
| 66 | 	public function tear_down() { | ||
| 73 | |||
| 74 | /** | ||
| 75 | * Test the `is_active` functionality when connected. | ||
| 76 | * | ||
| 77 | * @covers Automattic\Jetpack\Connection\Manager::is_active | ||
| 78 | */ | ||
| 79 | View Code Duplication | 	public function test_is_active_when_connected() { | |
| 90 | |||
| 91 | /** | ||
| 92 | * Test the `is_active` functionality when not connected. | ||
| 93 | * | ||
| 94 | * @covers Automattic\Jetpack\Connection\Manager::is_active | ||
| 95 | */ | ||
| 96 | 	public function test_is_active_when_not_connected() { | ||
| 103 | |||
| 104 | /** | ||
| 105 | * Test the `api_url` generation. | ||
| 106 | * | ||
| 107 | * @covers Automattic\Jetpack\Connection\Manager::api_url | ||
| 108 | */ | ||
| 109 | 	public function test_api_url_defaults() { | ||
| 123 | |||
| 124 | /** | ||
| 125 | * Testing the ability of the api_url method to follow set constants and filters. | ||
| 126 | * | ||
| 127 | * @covers Automattic\Jetpack\Connection\Manager::api_url | ||
| 128 | */ | ||
| 129 | 	public function test_api_url_uses_constants_and_filters() { | ||
| 170 | |||
| 171 | /** | ||
| 172 | * Test the `is_user_connected` functionality. | ||
| 173 | * | ||
| 174 | * @covers Automattic\Jetpack\Connection\Manager::is_user_connected | ||
| 175 | */ | ||
| 176 | 	public function test_is_user_connected_with_default_user_id_logged_out() { | ||
| 179 | |||
| 180 | /** | ||
| 181 | * Test the `is_user_connected` functionality. | ||
| 182 | * | ||
| 183 | * @covers Automattic\Jetpack\Connection\Manager::is_user_connected | ||
| 184 | */ | ||
| 185 | 	public function test_is_user_connected_with_false_user_id_logged_out() { | ||
| 188 | |||
| 189 | /** | ||
| 190 | * Test the `is_user_connected` functionality | ||
| 191 | * | ||
| 192 | * @covers Automattic\Jetpack\Connection\Manager::is_user_connected | ||
| 193 | */ | ||
| 194 | 	public function test_is_user_connected_with_user_id_logged_out_not_connected() { | ||
| 201 | |||
| 202 | /** | ||
| 203 | * Test the `is_user_connected` functionality. | ||
| 204 | * | ||
| 205 | * @covers Automattic\Jetpack\Connection\Manager::is_user_connected | ||
| 206 | */ | ||
| 207 | View Code Duplication | 	public function test_is_user_connected_with_default_user_id_logged_in() { | |
| 220 | |||
| 221 | /** | ||
| 222 | * Test the `is_user_connected` functionality. | ||
| 223 | * | ||
| 224 | * @covers Automattic\Jetpack\Connection\Manager::is_user_connected | ||
| 225 | */ | ||
| 226 | View Code Duplication | 	public function test_is_user_connected_with_user_id_logged_in() { | |
| 237 | |||
| 238 | /** | ||
| 239 | * Unit test for the "Delete all tokens" functionality. | ||
| 240 | * | ||
| 241 | * @covers Automattic\Jetpack\Connection\Manager::delete_all_connection_tokens | ||
| 242 | */ | ||
| 243 | View Code Duplication | 	public function test_delete_all_connection_tokens() { | |
| 255 | |||
| 256 | /** | ||
| 257 | * Unit test for the "Disconnect from WP" functionality. | ||
| 258 | * | ||
| 259 | * @covers Automattic\Jetpack\Connection\Manager::disconnect_site_wpcom | ||
| 260 | */ | ||
| 261 | View Code Duplication | 	public function test_disconnect_site_wpcom() { | |
| 273 | |||
| 274 | /** | ||
| 275 | * Test the `jetpack_connection_custom_caps' method. | ||
| 276 | * | ||
| 277 | * @covers Automattic\Jetpack\Connection\Manager::jetpack_connection_custom_caps | ||
| 278 | * @dataProvider jetpack_connection_custom_caps_data_provider | ||
| 279 | * | ||
| 280 | * @param bool $in_offline_mode Whether offline mode is active. | ||
| 281 | * @param bool $owner_exists Whether a connection owner exists. | ||
| 282 | * @param string $custom_cap The custom capability that is being tested. | ||
| 283 | * @param array $expected_caps The expected output. | ||
| 284 | */ | ||
| 285 | 	public function test_jetpack_connection_custom_caps( $in_offline_mode, $owner_exists, $custom_cap, $expected_caps ) { | ||
| 301 | |||
| 302 | /** | ||
| 303 | * Data provider test_jetpack_connection_custom_caps. | ||
| 304 | * | ||
| 305 | * Structure of the test data arrays: | ||
| 306 | * [0] => 'in_offline_mode' boolean Whether offline mode is active. | ||
| 307 | * [1] => 'owner_exists' boolean Whether a connection owner exists. | ||
| 308 | * [2] => 'custom_cap' string The custom capability that is being tested. | ||
| 309 | * [3] => 'expected_caps' array The expected output of the call to jetpack_connection_custom_caps. | ||
| 310 | */ | ||
| 311 | 	public function jetpack_connection_custom_caps_data_provider() { | ||
| 328 | |||
| 329 | /** | ||
| 330 | * Test the `get_signed_token` functionality. | ||
| 331 | * | ||
| 332 | * @covers Automattic\Jetpack\Connection\Manager::get_signed_token | ||
| 333 | */ | ||
| 334 | View Code Duplication | 	public function test_get_signed_token() { | |
| 357 | |||
| 358 | /** | ||
| 359 | * Test disconnecting a user from WordPress.com. | ||
| 360 | * | ||
| 361 | * @covers Automattic\Jetpack\Connection\Manager::disconnect_user | ||
| 362 | * @dataProvider get_disconnect_user_scenarios | ||
| 363 | * | ||
| 364 | * @param bool $remote Was the remote disconnection successful. | ||
| 365 | * @param bool $local Was the remote disconnection successful. | ||
| 366 | * @param bool $expected Expected outcome. | ||
| 367 | */ | ||
| 368 | 	public function test_disconnect_user( $remote, $local, $expected ) { | ||
| 389 | |||
| 390 | /** | ||
| 391 | * Test data for test_disconnect_user | ||
| 392 | * | ||
| 393 | * @return array | ||
| 394 | */ | ||
| 395 | View Code Duplication | 	public function get_disconnect_user_scenarios() { | |
| 414 | |||
| 415 | /** | ||
| 416 | * Filter to set the default constant values. | ||
| 417 | * | ||
| 418 | * @param string $value Existing value (empty and ignored). | ||
| 419 | * @param string $name Constant name. | ||
| 420 | * | ||
| 421 | * @see Utils::DEFAULT_JETPACK__API_BASE | ||
| 422 | * @see Utils::DEFAULT_JETPACK__API_VERSION | ||
| 423 | * | ||
| 424 | * @return string | ||
| 425 | */ | ||
| 426 | 	public function filter_api_constant( $value, $name ) { | ||
| 429 | |||
| 430 | } | ||
| 431 | 
This check compares calls to functions or methods with their respective definitions. If the call has more arguments than are defined, it raises an issue.
If a function is defined several times with a different number of parameters, the check may pick up the wrong definition and report false positives. One codebase where this has been known to happen is Wordpress.
In this case you can add the
@ignorePhpDoc annotation to the duplicate definition and it will be ignored.