| | 1 | <?php |
| | 2 | |
| | 3 | require_once ABSPATH . 'wp-admin/includes/plugin-install.php'; |
| | 4 | |
| | 5 | /** |
| | 6 | * |
| | 7 | * @group plugins-api |
| | 8 | */ |
| | 9 | class Tests_Plugins_API extends WP_UnitTestCase { |
| | 10 | |
| | 11 | public $fields = array( |
| | 12 | 'short_description' => true, |
| | 13 | 'description' => true, |
| | 14 | 'sections' => true, |
| | 15 | 'tested' => true, |
| | 16 | 'requires' => true, |
| | 17 | 'rating' => true, |
| | 18 | 'ratings' => true, |
| | 19 | 'downloaded' => true, |
| | 20 | 'downloadlink' => true, |
| | 21 | 'last_updated' => true, |
| | 22 | 'added' => true, |
| | 23 | 'tags' => true, |
| | 24 | 'compatibility' => true, |
| | 25 | 'homepage' => true, |
| | 26 | 'versions' => true, |
| | 27 | 'stable_tag' => true, |
| | 28 | 'donate_link' => true, |
| | 29 | 'reviews' => true, |
| | 30 | 'banners' => true, |
| | 31 | 'icons' => true, |
| | 32 | 'active_installs' => true, |
| | 33 | 'group' => true, |
| | 34 | 'contributors' => true, |
| | 35 | ); |
| | 36 | |
| | 37 | function test_wporg_plugin_api_serialize_php() { |
| | 38 | $response = wp_remote_post( 'http://api.wordpress.org/plugins/info/1.0/', array( |
| | 39 | 'timeout' => 15, |
| | 40 | 'body' => array( |
| | 41 | 'action' => 'plugin_information', |
| | 42 | 'request' => serialize( (object) array( 'slug' => 'jetpack', 'fields' => $this->fields ) ), |
| | 43 | ), |
| | 44 | ) ); |
| | 45 | $plugins = maybe_unserialize( wp_remote_retrieve_body( $response ) ); |
| | 46 | |
| | 47 | $this->_check_response_attributes( $plugins ); |
| | 48 | } |
| | 49 | |
| | 50 | function test_wporg_plugin_api_serialize_php_get() { |
| | 51 | $url = add_query_arg( 'fields', implode( ',' , array_keys( $this->fields ) ), 'http://api.wordpress.org/plugins/info/1.0/jetpack.php' ); |
| | 52 | $response = wp_remote_get( $url ); |
| | 53 | $plugins = maybe_unserialize( wp_remote_retrieve_body( $response ) ); |
| | 54 | |
| | 55 | $this->_check_response_attributes( $plugins ); |
| | 56 | } |
| | 57 | |
| | 58 | function test_wporg_plugin_api_xml() { |
| | 59 | $url = add_query_arg( 'fields', implode( ',' , array_keys( $this->fields ) ), 'http://api.wordpress.org/plugins/info/1.0/jetpack.xml' ); |
| | 60 | $response = wp_remote_get( $url ); |
| | 61 | $plugins = wp_remote_retrieve_body( $response ); |
| | 62 | |
| | 63 | // TODO: validate XML response. |
| | 64 | $this->markTestSkipped( 'Needs XML parsing.' ); |
| | 65 | $this->_check_response_attributes( $plugins ); |
| | 66 | } |
| | 67 | |
| | 68 | function test_wporg_plugin_api_json() { |
| | 69 | $url = add_query_arg( 'fields', implode( ',' , array_keys( $this->fields ) ), 'http://api.wordpress.org/plugins/info/1.0/jetpack.json' ); |
| | 70 | $response = wp_remote_get( $url ); |
| | 71 | $plugins = (object) json_decode( wp_remote_retrieve_body( $response ), true ); |
| | 72 | |
| | 73 | $this->_check_response_attributes( $plugins ); |
| | 74 | } |
| | 75 | |
| | 76 | function test_wporg_plugin_api_1_1_json() { |
| | 77 | $response = wp_remote_post( 'http://api.wordpress.org/plugins/info/1.1/', array( |
| | 78 | 'timeout' => 15, |
| | 79 | 'body' => array( |
| | 80 | 'action' => 'plugin_information', |
| | 81 | 'request' => (object) array( 'slug' => 'jetpack', 'fields' => $this->fields ), |
| | 82 | ), |
| | 83 | ) ); |
| | 84 | $plugins = (object) json_decode( wp_remote_retrieve_body( $response ), true ); |
| | 85 | |
| | 86 | $this->_check_response_attributes( $plugins ); |
| | 87 | } |
| | 88 | |
| | 89 | function test_plugins_api_function_action_plugin_information() { |
| | 90 | $plugins = plugins_api( 'plugin_information', array( 'slug' => 'jetpack', 'fields' => $this->fields ) ); |
| | 91 | $this->_check_response_attributes( $plugins ); |
| | 92 | } |
| | 93 | |
| | 94 | function test_plugins_api_function_action_query_plugins_search() { |
| | 95 | $slug = 'hello-dolly'; |
| | 96 | $per_page = 4; |
| | 97 | $page = 2; |
| | 98 | $plugins = plugins_api( 'query_plugins', array( |
| | 99 | 'search' => $slug, |
| | 100 | 'per_page' => $per_page, |
| | 101 | 'page' => $page, |
| | 102 | 'fields' => $this->fields, |
| | 103 | ) ); |
| | 104 | |
| | 105 | $this->_check_response_plugin_query( $plugins, $per_page, $page ); |
| | 106 | |
| | 107 | // If search term exactly matches a slug, it should be returned first. |
| | 108 | $plugins = plugins_api( 'query_plugins', array( 'search' => $slug ) ); |
| | 109 | $this->assertEquals( $plugins->plugins[0]->slug, $slug ); |
| | 110 | } |
| | 111 | |
| | 112 | // Plugins with a specific tag. |
| | 113 | function test_plugins_api_function_action_query_plugins_tag() { |
| | 114 | $tag = 'widget'; |
| | 115 | $plugins = plugins_api( 'query_plugins', array( 'tag' => $tag, 'fields' => $this->fields ) ); |
| | 116 | |
| | 117 | foreach ( $plugins->plugins as $plugin ) { |
| | 118 | $this->assertArrayHasKey( $tag, $plugin->tags, "Contains tag $tag" ); |
| | 119 | } |
| | 120 | $this->_check_response_plugin_query( $plugins ); |
| | 121 | } |
| | 122 | |
| | 123 | // Plugins written by a specific author. |
| | 124 | function test_plugins_api_function_action_query_plugins_author() { |
| | 125 | $author = 'wordpressdotorg'; |
| | 126 | $plugins = plugins_api( 'query_plugins', array( 'author' => $author, 'fields' => $this->fields ) ); |
| | 127 | |
| | 128 | foreach ( $plugins->plugins as $plugin ) { |
| | 129 | $this->assertArrayHasKey( $author, $plugin->contributors, "Contains author $author" ); |
| | 130 | } |
| | 131 | $this->_check_response_plugin_query( $plugins, 1 ); |
| | 132 | } |
| | 133 | |
| | 134 | // Favorites. |
| | 135 | function test_plugins_api_function_action_query_plugins_user() { |
| | 136 | $plugins = plugins_api( 'query_plugins', array( 'user' => 'markjaquith', 'fields' => $this->fields ) ); |
| | 137 | $this->_check_response_plugin_query( $plugins, 1 ); |
| | 138 | } |
| | 139 | |
| | 140 | function test_plugins_api_function_action_query_plugins_browse() { |
| | 141 | $plugins = plugins_api( 'query_plugins', array( 'browse' => 'popular', 'fields' => $this->fields ) ); |
| | 142 | $this->_check_response_plugin_query( $plugins ); |
| | 143 | } |
| | 144 | |
| | 145 | function test_plugins_api_function_action_query_plugins_installed_plugins() { |
| | 146 | $plugins = plugins_api( 'query_plugins', array( 'installed_plugins' => array( 'jetpack' ), 'fields' => $this->fields ) ); |
| | 147 | $this->_check_response_plugin_query( $plugins, 1 ); |
| | 148 | } |
| | 149 | |
| | 150 | function test_plugins_api_function_action_query_plugins_local() { |
| | 151 | // Not yet implemented. Shouldn't change the structure of the response though. |
| | 152 | $plugins = plugins_api( 'query_plugins', array( 'local' => 'hello', 'fields' => $this->fields ) ); |
| | 153 | $this->_check_response_plugin_query( $plugins, 1 ); |
| | 154 | } |
| | 155 | |
| | 156 | function test_plugins_api_function_action_hot_categories() { |
| | 157 | $number = 2; |
| | 158 | $plugins = plugins_api( 'hot_categories', array( 'number' => $number ) ); |
| | 159 | |
| | 160 | $this->assertEquals( $number, count( $plugins ) ); |
| | 161 | $this->assertInternalType( 'array', $plugins, 'Response array is array' ); |
| | 162 | |
| | 163 | foreach ( $plugins as $hot_category => $category_array ) { |
| | 164 | $this->assertInternalType( 'array', $category_array, 'Category array is array' ); |
| | 165 | $this->assertArrayHasKey( 'name', $category_array, 'Name exists' ); |
| | 166 | $this->assertArrayHasKey( 'slug', $category_array, 'Slug exists' ); |
| | 167 | |
| | 168 | // Only check the first result. |
| | 169 | break; |
| | 170 | } |
| | 171 | } |
| | 172 | |
| | 173 | function test_plugins_api_function_action_hot_tags() { |
| | 174 | $number = 3; |
| | 175 | $plugins = plugins_api( 'hot_tags', array( 'number' => $number ) ); |
| | 176 | |
| | 177 | $this->assertEquals( $number, count( $plugins ) ); |
| | 178 | $this->assertInternalType( 'array', $plugins, 'Response array is array' ); |
| | 179 | |
| | 180 | foreach ( $plugins as $hot_tag => $tag_array ) { |
| | 181 | $this->assertInternalType( 'array', $tag_array, 'Tag array is array' ); |
| | 182 | $this->assertArrayHasKey( 'name', $tag_array, 'Name exists' ); |
| | 183 | $this->assertArrayHasKey( 'slug', $tag_array, 'Slug exists' ); |
| | 184 | $this->assertArrayHasKey( 'count', $tag_array, 'Count exists' ); |
| | 185 | |
| | 186 | // Only check the first result. |
| | 187 | break; |
| | 188 | } |
| | 189 | } |
| | 190 | |
| | 191 | function _check_response_plugin_query( $plugin_query, $per_page = 24, $page = 1 ) { |
| | 192 | $this->assertObjectHasAttribute( 'info', $plugin_query, 'Info exists' ); |
| | 193 | $info = $plugin_query->info; |
| | 194 | |
| | 195 | $this->assertArrayHasKey( 'page', $info, 'Page exists' ); |
| | 196 | $this->assertEquals( $info['page'], $page, 'Page equals to ' . $page ); |
| | 197 | |
| | 198 | $this->assertArrayHasKey( 'pages', $info, 'Pages exists' ); |
| | 199 | $this->assertArrayHasKey( 'results', $info, 'Results exists' ); |
| | 200 | |
| | 201 | // Plugins. |
| | 202 | $this->assertObjectHasAttribute( 'plugins', $plugin_query, 'Plugins exists' ); |
| | 203 | $this->assertAttributeInternalType( 'array', 'plugins', $plugin_query, 'Plugins should be an array' ); |
| | 204 | |
| | 205 | $this->greaterThanOrEqual( count( $plugin_query->plugins ), $per_page ); |
| | 206 | |
| | 207 | $this->_check_response_attributes( $plugin_query->plugins[0] ); |
| | 208 | } |
| | 209 | |
| | 210 | function _check_response_attributes( $plugin_info ) { |
| | 211 | $this->assertObjectHasAttribute( 'name', $plugin_info, 'Name exists' ); |
| | 212 | $this->assertObjectHasAttribute( 'slug', $plugin_info, 'Slug exits' ); |
| | 213 | $this->assertObjectHasAttribute( 'version', $plugin_info, 'Version exists' ); |
| | 214 | $this->assertObjectHasAttribute( 'author', $plugin_info, 'Author exists' ); |
| | 215 | $this->assertObjectHasAttribute( 'author_profile', $plugin_info, 'Author Profile exists' ); |
| | 216 | $this->assertObjectHasAttribute( 'contributors', $plugin_info, 'Contributors exists' ); |
| | 217 | $this->assertAttributeInternalType( 'array', 'contributors', $plugin_info, 'Contributors should be an array' ); |
| | 218 | $this->assertObjectHasAttribute( 'requires', $plugin_info, 'Requires exists' ); |
| | 219 | $this->assertObjectHasAttribute( 'tested', $plugin_info, 'Tested exists' ); |
| | 220 | $this->assertAttributeInternalType( 'string', 'tested', $plugin_info, 'Tested should be a string' ); |
| | 221 | $this->assertObjectHasAttribute( 'compatibility', $plugin_info, 'Compatibility exists' ); |
| | 222 | $this->assertAttributeInternalType( 'array', 'compatibility', $plugin_info, 'Compatibility should be an array' ); |
| | 223 | |
| | 224 | // Ratings. |
| | 225 | $this->assertObjectHasAttribute( 'rating', $plugin_info, 'Rating exists' ); |
| | 226 | $this->assertObjectHasAttribute( 'ratings', $plugin_info, 'Ratings exists' ); |
| | 227 | $this->assertAttributeInternalType( 'array', 'ratings', $plugin_info, 'Ratings should be an array' ); |
| | 228 | $this->assertArrayHasKey( '1', $plugin_info->ratings, 'Rating should have an attribute of 1' ); |
| | 229 | $this->assertArrayHasKey( '2', $plugin_info->ratings, 'Rating should have an attribute of 2' ); |
| | 230 | $this->assertArrayHasKey( '3', $plugin_info->ratings, 'Rating should have an attribute of 3' ); |
| | 231 | $this->assertArrayHasKey( '4', $plugin_info->ratings, 'Rating should have an attribute of 4' ); |
| | 232 | $this->assertArrayHasKey( '5', $plugin_info->ratings, 'Rating should have an attribute of 5' ); |
| | 233 | $this->assertObjectHasAttribute( 'num_ratings', $plugin_info, 'Num ratings exists' ); |
| | 234 | $this->assertTrue( is_numeric( $plugin_info->num_ratings ), 'Num ratings are numeric' ); |
| | 235 | |
| | 236 | $this->assertObjectHasAttribute( 'active_installs', $plugin_info, 'Active Installs exists' ); |
| | 237 | $this->assertAttributeInternalType( 'integer', 'active_installs', $plugin_info, 'Active Installs should be an integer' ); |
| | 238 | $this->assertObjectHasAttribute( 'downloaded', $plugin_info, 'Active Installs exists' ); |
| | 239 | $this->assertAttributeInternalType( 'integer', 'downloaded', $plugin_info, 'Downloaded should be an integer' ); |
| | 240 | |
| | 241 | $this->assertObjectHasAttribute( 'last_updated', $plugin_info, 'Last Updated exists' ); |
| | 242 | $this->assertAttributeInternalType( 'string', 'last_updated', $plugin_info, 'Last Updated should be a string' ); |
| | 243 | $this->assertObjectHasAttribute( 'added', $plugin_info, 'Added exists' ); |
| | 244 | $this->assertAttributeInternalType( 'string', 'short_description', $plugin_info, 'Added should be a string' ); |
| | 245 | |
| | 246 | if ( function_exists( 'date_create_from_format' ) ) { |
| | 247 | $last_updated_date = DateTime::createFromFormat( 'Y-m-d g:ia \G\M\T', $plugin_info->last_updated ); |
| | 248 | $date_time_errors = DateTime::getLastErrors(); |
| | 249 | $this->assertTrue( $last_updated_date && 0 == $date_time_errors['warning_count'] && 0 == $date_time_errors['error_count'], 'Last updated has a valid format' ); |
| | 250 | |
| | 251 | $added_date = DateTime::createFromFormat( 'Y-m-d', $plugin_info->added ); |
| | 252 | $date_time_errors = DateTime::getLastErrors(); |
| | 253 | $this->assertTrue( $added_date && 0 == $date_time_errors['warning_count'] && 0 == $date_time_errors['error_count'], 'Added has a valid format' ); |
| | 254 | } |
| | 255 | |
| | 256 | $this->assertObjectHasAttribute( 'homepage', $plugin_info, 'Homepage exists' ); |
| | 257 | $this->assertAttributeInternalType( 'string', 'homepage', $plugin_info, 'Homepage should be a string' ); |
| | 258 | $this->assertObjectHasAttribute( 'sections', $plugin_info, 'Sections exists' ); |
| | 259 | $this->assertAttributeInternalType( 'array', 'sections', $plugin_info, 'Sections should be an array' ); |
| | 260 | |
| | 261 | $this->assertObjectHasAttribute( 'description', $plugin_info, 'Description exists' ); |
| | 262 | $this->assertAttributeInternalType( 'string', 'description', $plugin_info, 'Description should be a string' ); |
| | 263 | $this->assertObjectHasAttribute( 'short_description', $plugin_info, 'Short Description exists' ); |
| | 264 | $this->assertAttributeInternalType( 'string', 'short_description', $plugin_info, 'Short Description should be a string' ); |
| | 265 | |
| | 266 | $this->assertObjectHasAttribute( 'download_link', $plugin_info, 'Download link exists' ); |
| | 267 | $this->assertAttributeInternalType( 'string', 'download_link', $plugin_info, 'Download link should be a string' ); |
| | 268 | $this->assertFalse( empty( $plugin_info->download_link ), 'Download link should have a value' ); |
| | 269 | |
| | 270 | $this->assertObjectHasAttribute( 'tags', $plugin_info, 'Tags exists' ); |
| | 271 | $this->assertAttributeInternalType( 'array', 'tags', $plugin_info, 'Tags should be an array' ); |
| | 272 | |
| | 273 | $this->assertObjectHasAttribute( 'stable_tag', $plugin_info, 'Stable tag exists' ); |
| | 274 | $this->assertAttributeInternalType( 'string', 'stable_tag', $plugin_info, 'Stable tag should be a string' ); |
| | 275 | $this->assertFalse( empty( $plugin_info->stable_tag ), 'Stable tag should have a value' ); |
| | 276 | |
| | 277 | $this->assertObjectHasAttribute( 'versions', $plugin_info, 'Versions exists' ); |
| | 278 | $this->assertAttributeInternalType( 'array', 'versions', $plugin_info, 'Versions should be an array' ); |
| | 279 | |
| | 280 | $this->assertObjectHasAttribute( 'donate_link', $plugin_info, 'Donate link exists' ); |
| | 281 | $this->assertAttributeInternalType( 'string', 'donate_link', $plugin_info, 'Donate link should be a string' ); |
| | 282 | |
| | 283 | $this->assertObjectHasAttribute( 'banners', $plugin_info, 'Banners exists' ); |
| | 284 | $this->assertAttributeInternalType( 'array', 'banners', $plugin_info, 'Banners should be an array' ); |
| | 285 | |
| | 286 | $this->assertObjectHasAttribute( 'icons', $plugin_info, 'Icons exists' ); |
| | 287 | $this->assertAttributeInternalType( 'array', 'icons', $plugin_info, 'Icons should be an array' ); |
| | 288 | } |
| | 289 | } |