6 namespace Friendica\Test\legacy;
9 use Friendica\Core\Config\Capability\IManageConfigValues;
11 use Friendica\Module\Api\ApiResponse;
12 use Friendica\Module\BaseApi;
13 use Friendica\Security\BasicAuth;
14 use Friendica\Test\FixtureTest;
15 use Friendica\Util\Arrays;
16 use Friendica\Util\DateTimeFormat;
17 use Monolog\Handler\TestHandler;
19 require_once __DIR__ . '/../../include/api.php';
22 * Tests for the API functions.
24 * Functions that use header() need to be tested in a separate process.
25 * @see https://phpunit.de/manual/5.7/en/appendixes.annotations.html#appendixes.annotations.runTestsInSeparateProcesses
27 * @backupGlobals enabled
29 class ApiTest extends FixtureTest
32 * @var TestHandler Can handle log-outputs
39 protected $friendUser;
43 protected $wrongUserId;
48 /** @var IManageConfigValues */
52 * Create variables used by tests.
54 protected function setUp() : void
56 global $API, $called_api;
62 /** @var IManageConfigValues $config */
63 $this->config = $this->dice->create(IManageConfigValues::class);
65 $this->config->set('system', 'url', 'http://localhost');
66 $this->config->set('system', 'hostname', 'localhost');
67 $this->config->set('system', 'worker_dont_fork', true);
70 $this->config->set('config', 'hostname', 'localhost');
71 $this->config->set('system', 'throttle_limit_day', 100);
72 $this->config->set('system', 'throttle_limit_week', 100);
73 $this->config->set('system', 'throttle_limit_month', 100);
74 $this->config->set('system', 'theme', 'system_theme');
78 $this->app = DI::app();
80 DI::args()->setArgc(1);
82 // User data that the test database is populated with
85 'name' => 'Self contact',
86 'nick' => 'selfcontact',
87 'nurl' => 'http://localhost/profile/selfcontact'
91 'name' => 'Friend contact',
92 'nick' => 'friendcontact',
93 'nurl' => 'http://localhost/profile/friendcontact'
97 'name' => 'othercontact',
98 'nick' => 'othercontact',
99 'nurl' => 'http://localhost/profile/othercontact'
102 // User ID that we know is not in the database
103 $this->wrongUserId = 666;
105 DI::session()->start();
107 // Most API require login so we force the session
109 'authenticated' => true,
110 'uid' => $this->selfUser['id']
112 BasicAuth::setCurrentUserID($this->selfUser['id']);
116 * Assert that a list array contains expected keys.
118 * @param array $list List array
122 private function assertList(array $list = [])
124 self::assertIsString($list['name']);
125 self::assertIsInt($list['id']);
126 self::assertIsString('string', $list['id_str']);
127 self::assertContains($list['mode'], ['public', 'private']);
128 // We could probably do more checks here.
132 * Assert that the string is XML and contain the root element.
134 * @param string $result XML string
135 * @param string $root_element Root element name
139 private function assertXml($result = '', $root_element = '')
141 self::assertStringStartsWith('<?xml version="1.0"?>', $result);
142 self::assertStringContainsString('<' . $root_element, $result);
143 // We could probably do more checks here.
147 * Test the api_user() function.
151 public function testApiUser()
153 self::assertEquals($this->selfUser['id'], BaseApi::getCurrentUserID());
157 * Test the api_user() function with an unallowed user.
161 public function testApiUserWithUnallowedUser()
163 // self::assertEquals(false, api_user());
167 * Test the api_source() function.
171 public function testApiSource()
173 self::assertEquals('api', BasicAuth::getCurrentApplicationToken()['name']);
177 * Test the api_source() function with a Twidere user agent.
181 public function testApiSourceWithTwidere()
183 $_SERVER['HTTP_USER_AGENT'] = 'Twidere';
184 self::assertEquals('Twidere', BasicAuth::getCurrentApplicationToken()['name']);
188 * Test the api_source() function with a GET parameter.
192 public function testApiSourceWithGet()
194 $_REQUEST['source'] = 'source_name';
195 self::assertEquals('source_name', BasicAuth::getCurrentApplicationToken()['name']);
199 * Test the api_date() function.
203 public function testApiDate()
205 self::assertEquals('Wed Oct 10 00:00:00 +0000 1990', DateTimeFormat::utc('1990-10-10', DateTimeFormat::API));
209 * Test the api_register_func() function.
213 public function testApiRegisterFunc()
225 self::assertTrue(is_callable($API['api_path']['func']));
229 * Test the BasicAuth::getCurrentUserID() function without any login.
231 * @runInSeparateProcess
232 * @preserveGlobalState disabled
233 * @preserveGlobalState disabled
235 public function testApiLoginWithoutLogin()
237 BasicAuth::setCurrentUserID();
238 $this->expectException(\Friendica\Network\HTTPException\UnauthorizedException::class);
239 BasicAuth::getCurrentUserID(true);
243 * Test the BasicAuth::getCurrentUserID() function with a bad login.
245 * @runInSeparateProcess
246 * @preserveGlobalState disabled
247 * @preserveGlobalState disabled
249 public function testApiLoginWithBadLogin()
251 BasicAuth::setCurrentUserID();
252 $this->expectException(\Friendica\Network\HTTPException\UnauthorizedException::class);
253 $_SERVER['PHP_AUTH_USER'] = 'user@server';
254 BasicAuth::getCurrentUserID(true);
258 * Test the BasicAuth::getCurrentUserID() function with oAuth.
262 public function testApiLoginWithOauth()
264 $this->markTestIncomplete('Can we test this easily?');
268 * Test the BasicAuth::getCurrentUserID() function with authentication provided by an addon.
272 public function testApiLoginWithAddonAuth()
274 $this->markTestIncomplete('Can we test this easily?');
278 * Test the BasicAuth::getCurrentUserID() function with a correct login.
280 * @runInSeparateProcess
281 * @preserveGlobalState disabled
282 * @doesNotPerformAssertions
284 public function testApiLoginWithCorrectLogin()
286 BasicAuth::setCurrentUserID();
287 $_SERVER['PHP_AUTH_USER'] = 'Test user';
288 $_SERVER['PHP_AUTH_PW'] = 'password';
289 BasicAuth::getCurrentUserID(true);
293 * Test the BasicAuth::getCurrentUserID() function with a remote user.
295 * @runInSeparateProcess
296 * @preserveGlobalState disabled
298 public function testApiLoginWithRemoteUser()
300 BasicAuth::setCurrentUserID();
301 $this->expectException(\Friendica\Network\HTTPException\UnauthorizedException::class);
302 $_SERVER['REDIRECT_REMOTE_USER'] = '123456dXNlcjpwYXNzd29yZA==';
303 BasicAuth::getCurrentUserID(true);
307 * Test the api_call() function.
309 * @runInSeparateProcess
310 * @preserveGlobalState disabled
312 public function testApiCall()
316 'method' => 'method',
317 'func' => function () {
318 return ['data' => ['some_data']];
321 $_SERVER['REQUEST_METHOD'] = 'method';
322 $_SERVER['QUERY_STRING'] = 'pagename=api_path';
323 $_GET['callback'] = 'callback_name';
326 'callback_name(["some_data"])',
327 api_call('api_path', 'json')
332 * Test the api_call() function with the profiled enabled.
334 * @runInSeparateProcess
335 * @preserveGlobalState disabled
337 public function testApiCallWithProfiler()
341 'method' => 'method',
342 'func' => function () {
343 return ['data' => ['some_data']];
347 $_SERVER['REQUEST_METHOD'] = 'method';
348 $_SERVER['QUERY_STRING'] = 'pagename=api_path';
350 $this->config->set('system', 'profiler', true);
351 $this->config->set('rendertime', 'callstack', true);
352 $this->app->callstack = [
353 'database' => ['some_function' => 200],
354 'database_write' => ['some_function' => 200],
355 'cache' => ['some_function' => 200],
356 'cache_write' => ['some_function' => 200],
357 'network' => ['some_function' => 200]
362 api_call('api_path', 'json')
367 * Test the api_call() function with a JSON result.
369 * @runInSeparateProcess
370 * @preserveGlobalState disabled
372 public function testApiCallWithJson()
376 'method' => 'method',
377 'func' => function () {
378 return ['data' => ['some_data']];
381 $_SERVER['REQUEST_METHOD'] = 'method';
382 $_SERVER['QUERY_STRING'] = 'pagename=api_path.json';
386 api_call('api_path.json', 'json')
391 * Test the api_call() function with an XML result.
393 * @runInSeparateProcess
394 * @preserveGlobalState disabled
396 public function testApiCallWithXml()
400 'method' => 'method',
401 'func' => function () {
405 $_SERVER['REQUEST_METHOD'] = 'method';
406 $_SERVER['QUERY_STRING'] = 'pagename=api_path.xml';
408 $args = DI::args()->determine($_SERVER, $_GET);
412 api_call('api_path.xml', 'xml')
417 * Test the api_call() function with an RSS result.
419 * @runInSeparateProcess
420 * @preserveGlobalState disabled
422 public function testApiCallWithRss()
426 'method' => 'method',
427 'func' => function () {
431 $_SERVER['REQUEST_METHOD'] = 'method';
432 $_SERVER['QUERY_STRING'] = 'pagename=api_path.rss';
435 '<?xml version="1.0" encoding="UTF-8"?>' . "\n" .
437 api_call('api_path.rss', 'rss')
442 * Test the api_call() function with an Atom result.
444 * @runInSeparateProcess
445 * @preserveGlobalState disabled
447 public function testApiCallWithAtom()
451 'method' => 'method',
452 'func' => function () {
456 $_SERVER['REQUEST_METHOD'] = 'method';
457 $_SERVER['QUERY_STRING'] = 'pagename=api_path.atom';
460 '<?xml version="1.0" encoding="UTF-8"?>' . "\n" .
462 api_call('api_path.atom', 'atom')
467 * Test the api_rss_extra() function.
471 public function testApiRssExtra()
474 $user_info = ['url' => 'user_url', 'lang' => 'en'];
475 $result = api_rss_extra([], $user_info);
476 self::assertEquals($user_info, $result['$user']);
477 self::assertEquals($user_info['url'], $result['$rss']['alternate']);
478 self::assertArrayHasKey('self', $result['$rss']);
479 self::assertArrayHasKey('base', $result['$rss']);
480 self::assertArrayHasKey('updated', $result['$rss']);
481 self::assertArrayHasKey('atom_updated', $result['$rss']);
482 self::assertArrayHasKey('language', $result['$rss']);
483 self::assertArrayHasKey('logo', $result['$rss']);
488 * Test the api_rss_extra() function without any user info.
492 public function testApiRssExtraWithoutUserInfo()
495 $result = api_rss_extra([], null);
496 self::assertIsArray($result['$user']);
497 self::assertArrayHasKey('alternate', $result['$rss']);
498 self::assertArrayHasKey('self', $result['$rss']);
499 self::assertArrayHasKey('base', $result['$rss']);
500 self::assertArrayHasKey('updated', $result['$rss']);
501 self::assertArrayHasKey('atom_updated', $result['$rss']);
502 self::assertArrayHasKey('language', $result['$rss']);
503 self::assertArrayHasKey('logo', $result['$rss']);
508 * Test the api_get_user() function.
512 public function testApiGetUser()
514 // $user = api_get_user();
515 // self::assertSelfUser($user);
516 // self::assertEquals('708fa0', $user['profile_sidebar_fill_color']);
517 // self::assertEquals('6fdbe8', $user['profile_link_color']);
518 // self::assertEquals('ededed', $user['profile_background_color']);
522 * Test the api_get_user() function with a Frio schema.
526 public function testApiGetUserWithFrioSchema()
528 // $pConfig = $this->dice->create(IManagePersonalConfigValues::class);
529 // $pConfig->set($this->selfUser['id'], 'frio', 'schema', 'red');
530 // $user = api_get_user();
531 // self::assertSelfUser($user);
532 // self::assertEquals('708fa0', $user['profile_sidebar_fill_color']);
533 // self::assertEquals('6fdbe8', $user['profile_link_color']);
534 // self::assertEquals('ededed', $user['profile_background_color']);
538 * Test the api_get_user() function with an empty Frio schema.
542 public function testApiGetUserWithEmptyFrioSchema()
544 // $pConfig = $this->dice->create(IManagePersonalConfigValues::class);
545 // $pConfig->set($this->selfUser['id'], 'frio', 'schema', '---');
546 // $user = api_get_user();
547 // self::assertSelfUser($user);
548 // self::assertEquals('708fa0', $user['profile_sidebar_fill_color']);
549 // self::assertEquals('6fdbe8', $user['profile_link_color']);
550 // self::assertEquals('ededed', $user['profile_background_color']);
554 * Test the api_get_user() function with a custom Frio schema.
558 public function testApiGetUserWithCustomFrioSchema()
560 // $pConfig = $this->dice->create(IManagePersonalConfigValues::class);
561 // $pConfig->set($this->selfUser['id'], 'frio', 'schema', '---');
562 // $pConfig->set($this->selfUser['id'], 'frio', 'nav_bg', '#123456');
563 // $pConfig->set($this->selfUser['id'], 'frio', 'link_color', '#123456');
564 // $pConfig->set($this->selfUser['id'], 'frio', 'background_color', '#123456');
565 // $user = api_get_user();
566 // self::assertSelfUser($user);
567 // self::assertEquals('123456', $user['profile_sidebar_fill_color']);
568 // self::assertEquals('123456', $user['profile_link_color']);
569 // self::assertEquals('123456', $user['profile_background_color']);
573 * Test the api_get_user() function with an user that is not allowed to use the API.
575 * @runInSeparateProcess
576 * @preserveGlobalState disabled
578 public function testApiGetUserWithoutApiUser()
580 // api_get_user() with empty parameters is not used anymore
582 $_SERVER['PHP_AUTH_USER'] = 'Test user';
583 $_SERVER['PHP_AUTH_PW'] = 'password';
584 BasicAuth::setCurrentUserID();
585 self::assertFalse(api_get_user());
590 * Test the api_get_user() function with an user ID in a GET parameter.
594 public function testApiGetUserWithGetId()
596 // self::assertOtherUser(api_get_user());
600 * Test the api_get_user() function with a wrong user ID in a GET parameter.
604 public function testApiGetUserWithWrongGetId()
606 // $this->expectException(\Friendica\Network\HTTPException\BadRequestException::class);
607 // self::assertOtherUser(api_get_user());
611 * Test the api_get_user() function with an user name in a GET parameter.
615 public function testApiGetUserWithGetName()
617 // self::assertSelfUser(api_get_user());
621 * Test the api_get_user() function with a profile URL in a GET parameter.
625 public function testApiGetUserWithGetUrl()
627 // self::assertSelfUser(api_get_user());
631 * Test the api_get_user() function with an user ID in the API path.
635 public function testApiGetUserWithNumericCalledApi()
637 // global $called_api;
638 // $called_api = ['api_path'];
639 // DI::args()->setArgv(['', $this->otherUser['id'] . '.json']);
640 // self::assertOtherUser(api_get_user());
644 * Test the api_get_user() function with the $called_api global variable.
648 public function testApiGetUserWithCalledApi()
650 // global $called_api;
651 // $called_api = ['api', 'api_path'];
652 // self::assertSelfUser(api_get_user());
656 * Test the Arrays::walkRecursive() function.
660 public function testApiWalkRecursive()
665 Arrays::walkRecursive(
668 // Should we test this with a callback that actually does something?
676 * Test the Arrays::walkRecursive() function with an array.
680 public function testApiWalkRecursiveWithArray()
682 $array = [['item1'], ['item2']];
685 Arrays::walkRecursive(
688 // Should we test this with a callback that actually does something?
696 * Test the BaseApi::reformatXML() function.
700 public function testApiReformatXml()
704 self::assertTrue(ApiResponse::reformatXML($item, $key));
705 self::assertEquals('true', $item);
709 * Test the BaseApi::reformatXML() function with a statusnet_api key.
713 public function testApiReformatXmlWithStatusnetKey()
716 $key = 'statusnet_api';
717 self::assertTrue(ApiResponse::reformatXML($item, $key));
718 self::assertEquals('statusnet:api', $key);
722 * Test the BaseApi::reformatXML() function with a friendica_api key.
726 public function testApiReformatXmlWithFriendicaKey()
729 $key = 'friendica_api';
730 self::assertTrue(ApiResponse::reformatXML($item, $key));
731 self::assertEquals('friendica:api', $key);
735 * Test the BaseApi::createXML() function.
739 public function testApiCreateXml()
742 '<?xml version="1.0"?>' . "\n" .
743 '<root_element xmlns="http://api.twitter.com" xmlns:statusnet="http://status.net/schema/api/1/" ' .
744 'xmlns:friendica="http://friendi.ca/schema/api/1/" ' .
745 'xmlns:georss="http://www.georss.org/georss">' . "\n" .
746 ' <data>some_data</data>' . "\n" .
747 '</root_element>' . "\n",
748 DI::apiResponse()->createXML(['data' => ['some_data']], 'root_element')
753 * Test the BaseApi::createXML() function without any XML namespace.
757 public function testApiCreateXmlWithoutNamespaces()
760 '<?xml version="1.0"?>' . "\n" .
762 ' <data>some_data</data>' . "\n" .
764 DI::apiResponse()->createXML(['data' => ['some_data']], 'ok')
769 * Test the BaseApi::formatData() function.
773 public function testApiFormatData()
775 $data = ['some_data'];
776 self::assertEquals($data, DI::apiResponse()->formatData('root_element', 'json', $data));
780 * Test the BaseApi::formatData() function with an XML result.
784 public function testApiFormatDataWithXml()
787 '<?xml version="1.0"?>' . "\n" .
788 '<root_element xmlns="http://api.twitter.com" xmlns:statusnet="http://status.net/schema/api/1/" ' .
789 'xmlns:friendica="http://friendi.ca/schema/api/1/" ' .
790 'xmlns:georss="http://www.georss.org/georss">' . "\n" .
791 ' <data>some_data</data>' . "\n" .
792 '</root_element>' . "\n",
793 DI::apiResponse()->formatData('root_element', 'xml', ['data' => ['some_data']])
798 * Test the api_statuses_mediap() function.
802 public function testApiStatusesMediap()
805 DI::args()->setArgc(2);
813 'tmp_name' => $this->getTempImage(),
814 'name' => 'spacer.png',
815 'type' => 'image/png'
818 $_GET['status'] = '<b>Status content</b>';
820 $result = api_statuses_mediap('json');
821 self::assertStatus($result['status']);
826 * Test the api_statuses_mediap() function without an authenticated user.
830 public function testApiStatusesMediapWithoutAuthenticatedUser()
832 // $this->expectException(\Friendica\Network\HTTPException\UnauthorizedException::class);
833 // BasicAuth::setCurrentUserID();
834 // $_SESSION['authenticated'] = false;
835 // api_statuses_mediap('json');
839 * Test the api_statuses_update() function.
843 public function testApiStatusesUpdate()
846 $_REQUEST['status'] = 'Status content #friendica';
847 $_REQUEST['in_reply_to_status_id'] = -1;
848 $_REQUEST['lat'] = 48;
849 $_REQUEST['long'] = 7;
856 'tmp_name' => $this->getTempImage(),
857 'name' => 'spacer.png',
858 'type' => 'image/png'
862 $result = api_statuses_update('json');
863 self::assertStatus($result['status']);
868 * Test the api_statuses_update() function with an HTML status.
872 public function testApiStatusesUpdateWithHtml()
875 $_REQUEST['htmlstatus'] = '<b>Status content</b>';
877 $result = api_statuses_update('json');
878 self::assertStatus($result['status']);
883 * Test the api_statuses_update() function without an authenticated user.
887 public function testApiStatusesUpdateWithoutAuthenticatedUser()
890 $this->expectException(\Friendica\Network\HTTPException\UnauthorizedException::class);
891 BasicAuth::setCurrentUserID();
892 $_SESSION['authenticated'] = false;
893 api_statuses_update('json');
898 * Test the api_statuses_update() function with a parent status.
902 public function testApiStatusesUpdateWithParent()
904 $this->markTestIncomplete('This triggers an exit() somewhere and kills PHPUnit.');
908 * Test the api_statuses_update() function with a media_ids parameter.
912 public function testApiStatusesUpdateWithMediaIds()
914 $this->markTestIncomplete();
918 * Test the api_statuses_update() function with the throttle limit reached.
922 public function testApiStatusesUpdateWithDayThrottleReached()
924 $this->markTestIncomplete();
930 * Test the api_statuses_repeat() function.
934 public function testApiStatusesRepeat()
936 // $this->expectException(\Friendica\Network\HTTPException\ForbiddenException::class);
937 // api_statuses_repeat('json');
941 * Test the api_statuses_repeat() function without an authenticated user.
945 public function testApiStatusesRepeatWithoutAuthenticatedUser()
947 // $this->expectException(\Friendica\Network\HTTPException\UnauthorizedException::class);
948 // BasicAuth::setCurrentUserID();
949 // $_SESSION['authenticated'] = false;
950 // api_statuses_repeat('json');
954 * Test the api_statuses_repeat() function with an ID.
958 public function testApiStatusesRepeatWithId()
960 // DI::args()->setArgv(['', '', '', 1]);
961 // $result = api_statuses_repeat('json');
962 // self::assertStatus($result['status']);
964 // Also test with a shared status
965 // DI::args()->setArgv(['', '', '', 5]);
966 // $result = api_statuses_repeat('json');
967 // self::assertStatus($result['status']);
971 * Test the api_favorites_create_destroy() function.
975 public function testApiFavoritesCreateDestroy()
977 // $this->expectException(\Friendica\Network\HTTPException\BadRequestException::class);
978 // DI::args()->setArgv(['api', '1.1', 'favorites', 'create']);
979 // api_favorites_create_destroy('json');
983 * Test the api_favorites_create_destroy() function with an invalid ID.
987 public function testApiFavoritesCreateDestroyWithInvalidId()
989 // $this->expectException(\Friendica\Network\HTTPException\BadRequestException::class);
990 // DI::args()->setArgv(['api', '1.1', 'favorites', 'create', '12.json']);
991 // api_favorites_create_destroy('json');
995 * Test the api_favorites_create_destroy() function with an invalid action.
999 public function testApiFavoritesCreateDestroyWithInvalidAction()
1001 // $this->expectException(\Friendica\Network\HTTPException\BadRequestException::class);
1002 // DI::args()->setArgv(['api', '1.1', 'favorites', 'change.json']);
1003 // $_REQUEST['id'] = 1;
1004 // api_favorites_create_destroy('json');
1008 * Test the api_favorites_create_destroy() function with the create action.
1012 public function testApiFavoritesCreateDestroyWithCreateAction()
1014 // DI::args()->setArgv(['api', '1.1', 'favorites', 'create.json']);
1015 // $_REQUEST['id'] = 3;
1016 // $result = api_favorites_create_destroy('json');
1017 // self::assertStatus($result['status']);
1021 * Test the api_favorites_create_destroy() function with the create action and an RSS result.
1025 public function testApiFavoritesCreateDestroyWithCreateActionAndRss()
1027 // DI::args()->setArgv(['api', '1.1', 'favorites', 'create.rss']);
1028 // $_REQUEST['id'] = 3;
1029 // $result = api_favorites_create_destroy('rss');
1030 // self::assertXml($result, 'status');
1034 * Test the api_favorites_create_destroy() function with the destroy action.
1038 public function testApiFavoritesCreateDestroyWithDestroyAction()
1040 // DI::args()->setArgv(['api', '1.1', 'favorites', 'destroy.json']);
1041 // $_REQUEST['id'] = 3;
1042 // $result = api_favorites_create_destroy('json');
1043 // self::assertStatus($result['status']);
1047 * Test the api_favorites_create_destroy() function without an authenticated user.
1051 public function testApiFavoritesCreateDestroyWithoutAuthenticatedUser()
1054 $this->expectException(\Friendica\Network\HTTPException\UnauthorizedException::class);
1055 DI::args()->setArgv(['api', '1.1', 'favorites', 'create.json']);
1056 BasicAuth::setCurrentUserID();
1057 $_SESSION['authenticated'] = false;
1058 api_favorites_create_destroy('json');
1065 * Test the api_format_messages() function.
1069 public function testApiFormatMessages()
1071 $result = api_format_messages(
1072 ['id' => 1, 'uri-id' => 1, 'title' => 'item_title', 'body' => '[b]item_body[/b]'],
1073 ['id' => 2, 'uri-id' => 2, 'screen_name' => 'recipient_name'],
1074 ['id' => 3, 'uri-id' => 2, 'screen_name' => 'sender_name']
1076 self::assertEquals('item_title' . "\n" . 'item_body', $result['text']);
1077 self::assertEquals(1, $result['id']);
1078 self::assertEquals(2, $result['recipient_id']);
1079 self::assertEquals(3, $result['sender_id']);
1080 self::assertEquals('recipient_name', $result['recipient_screen_name']);
1081 self::assertEquals('sender_name', $result['sender_screen_name']);
1085 * Test the api_format_messages() function with HTML.
1089 public function testApiFormatMessagesWithHtmlText()
1091 $_GET['getText'] = 'html';
1092 $result = api_format_messages(
1093 ['id' => 1, 'uri-id' => 1, 'title' => 'item_title', 'body' => '[b]item_body[/b]'],
1094 ['id' => 2, 'uri-id' => 2, 'screen_name' => 'recipient_name'],
1095 ['id' => 3, 'uri-id' => 3, 'screen_name' => 'sender_name']
1097 self::assertEquals('item_title', $result['title']);
1098 self::assertEquals('<strong>item_body</strong>', $result['text']);
1102 * Test the api_format_messages() function with plain text.
1106 public function testApiFormatMessagesWithPlainText()
1108 $_GET['getText'] = 'plain';
1109 $result = api_format_messages(
1110 ['id' => 1, 'uri-id' => 1, 'title' => 'item_title', 'body' => '[b]item_body[/b]'],
1111 ['id' => 2, 'uri-id' => 2, 'screen_name' => 'recipient_name'],
1112 ['id' => 3, 'uri-id' => 3, 'screen_name' => 'sender_name']
1114 self::assertEquals('item_title', $result['title']);
1115 self::assertEquals('item_body', $result['text']);
1119 * Test the api_format_messages() function with the getUserObjects GET parameter set to false.
1123 public function testApiFormatMessagesWithoutUserObjects()
1125 $_GET['getUserObjects'] = 'false';
1126 $result = api_format_messages(
1127 ['id' => 1, 'uri-id' => 1, 'title' => 'item_title', 'body' => '[b]item_body[/b]'],
1128 ['id' => 2, 'uri-id' => 2, 'screen_name' => 'recipient_name'],
1129 ['id' => 3, 'uri-id' => 3, 'screen_name' => 'sender_name']
1131 self::assertTrue(!isset($result['sender']));
1132 self::assertTrue(!isset($result['recipient']));
1136 * Test the api_convert_item() function.
1140 public function testApiConvertItem()
1143 $result = api_convert_item(
1145 'network' => 'feed',
1146 'title' => 'item_title',
1148 // We need a long string to test that it is correctly cut
1149 'body' => 'perspiciatis impedit voluptatem quis molestiae ea qui ' .
1150 'reiciendis dolorum aut ducimus sunt consequatur inventore dolor ' .
1151 'officiis pariatur doloremque nemo culpa aut quidem qui dolore ' .
1152 'laudantium atque commodi alias voluptatem non possimus aperiam ' .
1153 'ipsum rerum consequuntur aut amet fugit quia aliquid praesentium ' .
1154 'repellendus quibusdam et et inventore mollitia rerum sit autem ' .
1155 'pariatur maiores ipsum accusantium perferendis vel sit possimus ' .
1156 'veritatis nihil distinctio qui eum repellat officia illum quos ' .
1157 'impedit quam iste esse unde qui suscipit aut facilis ut inventore ' .
1158 'omnis exercitationem quo magnam consequatur maxime aut illum ' .
1159 'soluta quaerat natus unde aspernatur et sed beatae nihil ullam ' .
1160 'temporibus corporis ratione blanditiis perspiciatis impedit ' .
1161 'voluptatem quis molestiae ea qui reiciendis dolorum aut ducimus ' .
1162 'sunt consequatur inventore dolor officiis pariatur doloremque ' .
1163 'nemo culpa aut quidem qui dolore laudantium atque commodi alias ' .
1164 'voluptatem non possimus aperiam ipsum rerum consequuntur aut ' .
1165 'amet fugit quia aliquid praesentium repellendus quibusdam et et ' .
1166 'inventore mollitia rerum sit autem pariatur maiores ipsum accusantium ' .
1167 'perferendis vel sit possimus veritatis nihil distinctio qui eum ' .
1168 'repellat officia illum quos impedit quam iste esse unde qui ' .
1169 'suscipit aut facilis ut inventore omnis exercitationem quo magnam ' .
1170 'consequatur maxime aut illum soluta quaerat natus unde aspernatur ' .
1171 'et sed beatae nihil ullam temporibus corporis ratione blanditiis',
1172 'plink' => 'item_plink'
1175 self::assertStringStartsWith('item_title', $result['text']);
1176 self::assertStringStartsWith('<h4>item_title</h4><br>perspiciatis impedit voluptatem', $result['html']);
1181 * Test the api_convert_item() function with an empty item body.
1185 public function testApiConvertItemWithoutBody()
1188 $result = api_convert_item(
1190 'network' => 'feed',
1191 'title' => 'item_title',
1194 'plink' => 'item_plink'
1197 self::assertEquals("item_title", $result['text']);
1198 self::assertEquals('<h4>item_title</h4><br>item_plink', $result['html']);
1203 * Test the api_convert_item() function with the title in the body.
1207 public function testApiConvertItemWithTitleInBody()
1210 $result = api_convert_item(
1212 'title' => 'item_title',
1213 'body' => 'item_title item_body',
1217 self::assertEquals('item_title item_body', $result['text']);
1218 self::assertEquals('<h4>item_title</h4><br>item_title item_body', $result['html']);
1223 * Test the api_get_attachments() function.
1227 public function testApiGetAttachments()
1230 // self::assertEmpty(api_get_attachments($body, 0));
1234 * Test the api_get_attachments() function with an img tag.
1238 public function testApiGetAttachmentsWithImage()
1240 // $body = '[img]http://via.placeholder.com/1x1.png[/img]';
1241 // self::assertIsArray(api_get_attachments($body, 0));
1245 * Test the api_get_attachments() function with an img tag and an AndStatus user agent.
1249 public function testApiGetAttachmentsWithImageAndAndStatus()
1251 // $_SERVER['HTTP_USER_AGENT'] = 'AndStatus';
1252 // $body = '[img]http://via.placeholder.com/1x1.png[/img]';
1253 // self::assertIsArray(api_get_attachments($body, 0));
1257 * Test the api_get_entitities() function.
1261 public function testApiGetEntitities()
1264 // self::assertIsArray(api_get_entitities($text, 'bbcode', 0));
1268 * Test the api_get_entitities() function with the include_entities parameter.
1272 public function testApiGetEntititiesWithIncludeEntities()
1275 $_REQUEST['include_entities'] = 'true';
1277 $result = api_get_entitities($text, 'bbcode', 0);
1278 self::assertIsArray($result['hashtags']);
1279 self::assertIsArray($result['symbols']);
1280 self::assertIsArray($result['urls']);
1281 self::assertIsArray($result['user_mentions']);
1286 * Test the api_format_items_embeded_images() function.
1290 public function testApiFormatItemsEmbededImages()
1294 'text ' . DI::baseUrl() . '/display/item_guid',
1295 api_format_items_embeded_images(['guid' => 'item_guid'], 'text data:image/foo')
1301 * Test the api_format_items_activities() function.
1305 public function testApiFormatItemsActivities()
1307 $item = ['uid' => 0, 'uri-id' => 1];
1308 $result = DI::friendicaActivities()->createFromUriId($item['uri-id'], $item['uid']);
1309 self::assertArrayHasKey('like', $result);
1310 self::assertArrayHasKey('dislike', $result);
1311 self::assertArrayHasKey('attendyes', $result);
1312 self::assertArrayHasKey('attendno', $result);
1313 self::assertArrayHasKey('attendmaybe', $result);
1317 * Test the api_format_items_activities() function with an XML result.
1321 public function testApiFormatItemsActivitiesWithXml()
1323 $item = ['uid' => 0, 'uri-id' => 1];
1324 $result = DI::friendicaActivities()->createFromUriId($item['uri-id'], $item['uid'], 'xml');
1325 self::assertArrayHasKey('friendica:like', $result);
1326 self::assertArrayHasKey('friendica:dislike', $result);
1327 self::assertArrayHasKey('friendica:attendyes', $result);
1328 self::assertArrayHasKey('friendica:attendno', $result);
1329 self::assertArrayHasKey('friendica:attendmaybe', $result);
1333 * Test the api_format_items() function.
1334 * @doesNotPerformAssertions
1336 public function testApiFormatItems()
1339 $items = Post::selectToArray([], ['uid' => 42]);
1340 foreach ($items as $item) {
1341 $status = api_format_item($item);
1342 self::assertStatus($status);
1348 * Test the api_format_items() function with an XML result.
1349 * @doesNotPerformAssertions
1351 public function testApiFormatItemsWithXml()
1354 $items = Post::selectToArray([], ['uid' => 42]);
1355 foreach ($items as $item) {
1356 $status = api_format_item($item, 'xml');
1357 self::assertStatus($status);
1363 * Test the api_lists_list() function.
1367 public function testApiListsList()
1369 $result = api_lists_list('json');
1370 self::assertEquals(['lists_list' => []], $result);
1374 * Test the api_lists_ownerships() function.
1378 public function testApiListsOwnerships()
1380 $result = api_lists_ownerships('json');
1381 foreach ($result['lists']['lists'] as $list) {
1382 self::assertList($list);
1387 * Test the api_lists_ownerships() function without an authenticated user.
1391 public function testApiListsOwnershipsWithoutAuthenticatedUser()
1393 $this->expectException(\Friendica\Network\HTTPException\UnauthorizedException::class);
1394 BasicAuth::setCurrentUserID();
1395 $_SESSION['authenticated'] = false;
1396 api_lists_ownerships('json');
1400 * Test the api_statuses_f() function.
1404 public function testApiStatusesFWithIncoming()
1406 // $result = api_statuses_f('incoming');
1407 // self::assertArrayHasKey('user', $result);
1412 * Test the api_direct_messages_new() function.
1416 public function testApiDirectMessagesNew()
1418 $result = api_direct_messages_new('json');
1419 self::assertNull($result);
1423 * Test the api_direct_messages_new() function without an authenticated user.
1427 public function testApiDirectMessagesNewWithoutAuthenticatedUser()
1429 $this->expectException(\Friendica\Network\HTTPException\UnauthorizedException::class);
1430 BasicAuth::setCurrentUserID();
1431 $_SESSION['authenticated'] = false;
1432 api_direct_messages_new('json');
1436 * Test the api_direct_messages_new() function with an user ID.
1440 public function testApiDirectMessagesNewWithUserId()
1442 $_POST['text'] = 'message_text';
1443 $_REQUEST['user_id'] = $this->otherUser['id'];
1444 $result = api_direct_messages_new('json');
1445 self::assertEquals(['direct_message' => ['error' => -1]], $result);
1449 * Test the api_direct_messages_new() function with a screen name.
1453 public function testApiDirectMessagesNewWithScreenName()
1455 $this->app->setLoggedInUserNickname($this->selfUser['nick']);
1456 $_POST['text'] = 'message_text';
1457 $_REQUEST['user_id'] = $this->friendUser['id'];
1458 $result = api_direct_messages_new('json');
1459 self::assertStringContainsString('message_text', $result['direct_message']['text']);
1460 self::assertEquals('selfcontact', $result['direct_message']['sender_screen_name']);
1461 self::assertEquals(1, $result['direct_message']['friendica_seen']);
1465 * Test the api_direct_messages_new() function with a title.
1469 public function testApiDirectMessagesNewWithTitle()
1471 $this->app->setLoggedInUserNickname($this->selfUser['nick']);
1472 $_POST['text'] = 'message_text';
1473 $_REQUEST['user_id'] = $this->friendUser['id'];
1474 $_REQUEST['title'] = 'message_title';
1475 $result = api_direct_messages_new('json');
1476 self::assertStringContainsString('message_text', $result['direct_message']['text']);
1477 self::assertStringContainsString('message_title', $result['direct_message']['text']);
1478 self::assertEquals('selfcontact', $result['direct_message']['sender_screen_name']);
1479 self::assertEquals(1, $result['direct_message']['friendica_seen']);
1483 * Test the api_direct_messages_new() function with an RSS result.
1487 public function testApiDirectMessagesNewWithRss()
1489 $this->app->setLoggedInUserNickname($this->selfUser['nick']);
1490 $_POST['text'] = 'message_text';
1491 $_REQUEST['user_id'] = $this->friendUser['id'];
1492 $result = api_direct_messages_new('rss');
1493 self::assertXml($result, 'direct-messages');
1497 * Test the api_direct_messages_destroy() function.
1501 public function testApiDirectMessagesDestroy()
1503 $this->expectException(\Friendica\Network\HTTPException\BadRequestException::class);
1504 api_direct_messages_destroy('json');
1508 * Test the api_direct_messages_destroy() function with the friendica_verbose GET param.
1512 public function testApiDirectMessagesDestroyWithVerbose()
1514 $_GET['friendica_verbose'] = 'true';
1515 $result = api_direct_messages_destroy('json');
1519 'result' => 'error',
1520 'message' => 'message id or parenturi not specified'
1528 * Test the api_direct_messages_destroy() function without an authenticated user.
1532 public function testApiDirectMessagesDestroyWithoutAuthenticatedUser()
1534 $this->expectException(\Friendica\Network\HTTPException\UnauthorizedException::class);
1535 BasicAuth::setCurrentUserID();
1536 $_SESSION['authenticated'] = false;
1537 api_direct_messages_destroy('json');
1541 * Test the api_direct_messages_destroy() function with a non-zero ID.
1545 public function testApiDirectMessagesDestroyWithId()
1547 $this->expectException(\Friendica\Network\HTTPException\BadRequestException::class);
1548 $_REQUEST['id'] = 1;
1549 api_direct_messages_destroy('json');
1553 * Test the api_direct_messages_destroy() with a non-zero ID and the friendica_verbose GET param.
1557 public function testApiDirectMessagesDestroyWithIdAndVerbose()
1559 $_REQUEST['id'] = 1;
1560 $_REQUEST['friendica_parenturi'] = 'parent_uri';
1561 $_GET['friendica_verbose'] = 'true';
1562 $result = api_direct_messages_destroy('json');
1566 'result' => 'error',
1567 'message' => 'message id not in database'
1575 * Test the api_direct_messages_destroy() function with a non-zero ID.
1579 public function testApiDirectMessagesDestroyWithCorrectId()
1581 $this->markTestIncomplete('We need to add a dataset for this.');
1585 * Test the api_direct_messages_box() function.
1589 public function testApiDirectMessagesBoxWithSentbox()
1591 $_REQUEST['page'] = -1;
1592 $_REQUEST['max_id'] = 10;
1593 $result = api_direct_messages_box('json', 'sentbox', 'false');
1594 self::assertArrayHasKey('direct_message', $result);
1598 * Test the api_direct_messages_box() function.
1602 public function testApiDirectMessagesBoxWithConversation()
1604 $result = api_direct_messages_box('json', 'conversation', 'false');
1605 self::assertArrayHasKey('direct_message', $result);
1609 * Test the api_direct_messages_box() function.
1613 public function testApiDirectMessagesBoxWithAll()
1615 $result = api_direct_messages_box('json', 'all', 'false');
1616 self::assertArrayHasKey('direct_message', $result);
1620 * Test the api_direct_messages_box() function.
1624 public function testApiDirectMessagesBoxWithInbox()
1626 $result = api_direct_messages_box('json', 'inbox', 'false');
1627 self::assertArrayHasKey('direct_message', $result);
1631 * Test the api_direct_messages_box() function.
1635 public function testApiDirectMessagesBoxWithVerbose()
1637 $result = api_direct_messages_box('json', 'sentbox', 'true');
1641 'result' => 'error',
1642 'message' => 'no mails available'
1650 * Test the api_direct_messages_box() function with a RSS result.
1654 public function testApiDirectMessagesBoxWithRss()
1656 $result = api_direct_messages_box('rss', 'sentbox', 'false');
1657 self::assertXml($result, 'direct-messages');
1661 * Test the api_direct_messages_box() function without an authenticated user.
1665 public function testApiDirectMessagesBoxWithUnallowedUser()
1667 $this->expectException(\Friendica\Network\HTTPException\UnauthorizedException::class);
1668 BasicAuth::setCurrentUserID();
1669 api_direct_messages_box('json', 'sentbox', 'false');
1673 * Test the api_direct_messages_sentbox() function.
1677 public function testApiDirectMessagesSentbox()
1679 $result = api_direct_messages_sentbox('json');
1680 self::assertArrayHasKey('direct_message', $result);
1684 * Test the api_direct_messages_inbox() function.
1688 public function testApiDirectMessagesInbox()
1690 $result = api_direct_messages_inbox('json');
1691 self::assertArrayHasKey('direct_message', $result);
1695 * Test the api_direct_messages_all() function.
1699 public function testApiDirectMessagesAll()
1701 $result = api_direct_messages_all('json');
1702 self::assertArrayHasKey('direct_message', $result);
1706 * Test the api_direct_messages_conversation() function.
1710 public function testApiDirectMessagesConversation()
1712 $result = api_direct_messages_conversation('json');
1713 self::assertArrayHasKey('direct_message', $result);
1717 * Test the api_oauth_request_token() function.
1721 public function testApiOauthRequestToken()
1723 $this->markTestIncomplete('exit() kills phpunit as well');
1727 * Test the api_oauth_access_token() function.
1731 public function testApiOauthAccessToken()
1733 $this->markTestIncomplete('exit() kills phpunit as well');
1737 * Test the api_fr_photos_list() function.
1741 public function testApiFrPhotosList()
1743 $result = api_fr_photos_list('json');
1744 self::assertArrayHasKey('photo', $result);
1748 * Test the api_fr_photos_list() function without an authenticated user.
1752 public function testApiFrPhotosListWithoutAuthenticatedUser()
1754 $this->expectException(\Friendica\Network\HTTPException\UnauthorizedException::class);
1755 BasicAuth::setCurrentUserID();
1756 $_SESSION['authenticated'] = false;
1757 api_fr_photos_list('json');
1761 * Test the api_fr_photo_create_update() function.
1763 public function testApiFrPhotoCreateUpdate()
1765 $this->expectException(\Friendica\Network\HTTPException\BadRequestException::class);
1766 api_fr_photo_create_update('json');
1770 * Test the api_fr_photo_create_update() function without an authenticated user.
1774 public function testApiFrPhotoCreateUpdateWithoutAuthenticatedUser()
1776 $this->expectException(\Friendica\Network\HTTPException\UnauthorizedException::class);
1777 BasicAuth::setCurrentUserID();
1778 $_SESSION['authenticated'] = false;
1779 api_fr_photo_create_update('json');
1783 * Test the api_fr_photo_create_update() function with an album name.
1787 public function testApiFrPhotoCreateUpdateWithAlbum()
1789 $this->expectException(\Friendica\Network\HTTPException\BadRequestException::class);
1790 $_REQUEST['album'] = 'album_name';
1791 api_fr_photo_create_update('json');
1795 * Test the api_fr_photo_create_update() function with the update mode.
1799 public function testApiFrPhotoCreateUpdateWithUpdate()
1801 $this->markTestIncomplete('We need to create a dataset for this');
1805 * Test the api_fr_photo_create_update() function with an uploaded file.
1809 public function testApiFrPhotoCreateUpdateWithFile()
1811 $this->markTestIncomplete();
1815 * Test the api_fr_photo_detail() function.
1819 public function testApiFrPhotoDetail()
1821 $this->expectException(\Friendica\Network\HTTPException\BadRequestException::class);
1822 api_fr_photo_detail('json');
1826 * Test the api_fr_photo_detail() function without an authenticated user.
1830 public function testApiFrPhotoDetailWithoutAuthenticatedUser()
1832 $this->expectException(\Friendica\Network\HTTPException\UnauthorizedException::class);
1833 BasicAuth::setCurrentUserID();
1834 $_SESSION['authenticated'] = false;
1835 api_fr_photo_detail('json');
1839 * Test the api_fr_photo_detail() function with a photo ID.
1843 public function testApiFrPhotoDetailWithPhotoId()
1845 $this->expectException(\Friendica\Network\HTTPException\NotFoundException::class);
1846 $_REQUEST['photo_id'] = 1;
1847 api_fr_photo_detail('json');
1851 * Test the api_fr_photo_detail() function with a correct photo ID.
1855 public function testApiFrPhotoDetailCorrectPhotoId()
1857 $this->markTestIncomplete('We need to create a dataset for this.');
1861 * Test the api_account_update_profile_image() function.
1865 public function testApiAccountUpdateProfileImage()
1867 $this->expectException(\Friendica\Network\HTTPException\BadRequestException::class);
1868 api_account_update_profile_image('json');
1872 * Test the api_account_update_profile_image() function without an authenticated user.
1876 public function testApiAccountUpdateProfileImageWithoutAuthenticatedUser()
1878 $this->expectException(\Friendica\Network\HTTPException\UnauthorizedException::class);
1879 BasicAuth::setCurrentUserID();
1880 $_SESSION['authenticated'] = false;
1881 api_account_update_profile_image('json');
1885 * Test the api_account_update_profile_image() function with an uploaded file.
1889 public function testApiAccountUpdateProfileImageWithUpload()
1891 $this->expectException(\Friendica\Network\HTTPException\BadRequestException::class);
1892 $this->markTestIncomplete();
1896 * Test the check_acl_input() function.
1900 public function testCheckAclInput()
1902 $result = check_acl_input('<aclstring>', BaseApi::getCurrentUserID());
1903 // Where does this result come from?
1904 self::assertEquals(1, $result);
1908 * Test the check_acl_input() function with an empty ACL string.
1912 public function testCheckAclInputWithEmptyAclString()
1914 $result = check_acl_input(' ', BaseApi::getCurrentUserID());
1915 self::assertFalse($result);
1919 * Test the save_media_to_database() function.
1923 public function testSaveMediaToDatabase()
1925 $this->markTestIncomplete();
1929 * Test the post_photo_item() function.
1933 public function testPostPhotoItem()
1935 $this->markTestIncomplete();
1939 * Test the prepare_photo_data() function.
1943 public function testPreparePhotoData()
1945 $this->markTestIncomplete();
1949 * Test the api_share_as_retweet() function with a valid item.
1953 public function testApiShareAsRetweetWithValidItem()
1955 $this->markTestIncomplete();
1959 * Test the api_in_reply_to() function with a valid item.
1963 public function testApiInReplyToWithValidItem()
1965 $this->markTestIncomplete();
1969 * Test the api_clean_plain_items() function.
1973 public function testApiCleanPlainItems()
1975 $_REQUEST['include_entities'] = 'true';
1976 $result = api_clean_plain_items('some_text [url="some_url"]some_text[/url]');
1977 self::assertEquals('some_text [url="some_url"]"some_url"[/url]', $result);
1981 * Test the api_best_nickname() function with contacts.
1985 public function testApiBestNicknameWithContacts()
1987 $this->markTestIncomplete();
1991 * Test the api_friendica_group_show() function.
1995 public function testApiFriendicaGroupShow()
1997 $this->markTestIncomplete();
2001 * Test the api_friendica_group_delete() function.
2005 public function testApiFriendicaGroupDelete()
2007 $this->markTestIncomplete();
2011 * Test the api_lists_destroy() function.
2015 public function testApiListsDestroy()
2017 $this->markTestIncomplete();
2021 * Test the group_create() function.
2025 public function testGroupCreate()
2027 $this->markTestIncomplete();
2031 * Test the api_friendica_group_create() function.
2035 public function testApiFriendicaGroupCreate()
2037 $this->markTestIncomplete();
2041 * Test the api_lists_create() function.
2045 public function testApiListsCreate()
2047 $this->markTestIncomplete();
2051 * Test the api_friendica_group_update() function.
2055 public function testApiFriendicaGroupUpdate()
2057 $this->markTestIncomplete();
2061 * Test the api_lists_update() function.
2065 public function testApiListsUpdate()
2067 $this->markTestIncomplete();
2071 * Test the api_friendica_activity() function.
2075 public function testApiFriendicaActivity()
2077 $this->markTestIncomplete();
2081 * Test the api_friendica_notification_seen() function.
2085 public function testApiFriendicaNotificationSeen()
2087 $this->markTestIncomplete();
2091 * Test the api_friendica_direct_messages_setseen() function.
2095 public function testApiFriendicaDirectMessagesSetseen()
2097 $this->markTestIncomplete();
2101 * Test the api_friendica_direct_messages_search() function.
2105 public function testApiFriendicaDirectMessagesSearch()
2107 $this->markTestIncomplete();