1
1
Fork 0
mirror of https://code.mensbeam.com/MensBeam/Arsse.git synced 2025-01-12 10:52:40 +00:00
Arsse/tests/cases/REST/TestREST.php

330 lines
19 KiB
PHP
Raw Normal View History

<?php
/** @license MIT
* Copyright 2017 J. King, Dustin Wilson et al.
* See LICENSE and AUTHORS files for details */
declare(strict_types=1);
namespace JKingWeb\Arsse\TestCase\REST;
use JKingWeb\Arsse\Arsse;
use JKingWeb\Arsse\User;
use JKingWeb\Arsse\REST;
use JKingWeb\Arsse\REST\Exception501;
use JKingWeb\Arsse\REST\NextcloudNews\V1_2 as NCN;
use JKingWeb\Arsse\REST\TinyTinyRSS\API as TTRSS;
use JKingWeb\Arsse\Misc\HTTP;
use Psr\Http\Message\RequestInterface;
use Psr\Http\Message\ResponseInterface;
2022-08-06 17:40:02 +00:00
use GuzzleHttp\Psr7\Response;
use GuzzleHttp\Psr7\Request;
use GuzzleHttp\Psr7\ServerRequest;
/** @covers \JKingWeb\Arsse\REST */
class TestREST extends \JKingWeb\Arsse\Test\AbstractTest {
/** @dataProvider provideApiMatchData */
2020-01-20 18:52:48 +00:00
public function testMatchAUrlToAnApi($apiList, string $input, array $exp): void {
$r = new REST($apiList);
try {
$out = $r->apiMatch($input);
} catch (Exception501 $e) {
$out = [];
}
$this->assertEquals($exp, $out);
}
2019-10-16 18:42:43 +00:00
public function provideApiMatchData(): iterable {
$real = null;
$fake = [
'unstripped' => ['match' => "/full/url", 'strip' => "", 'class' => "UnstrippedProtocol"],
];
return [
[$real, "/index.php/apps/news/api/v1-2/feeds", ["ncn_v1-2", "/feeds", \JKingWeb\Arsse\REST\NextcloudNews\V1_2::class]],
[$real, "/index.php/apps/news/api/v1-2", ["ncn", "/v1-2", \JKingWeb\Arsse\REST\NextcloudNews\Versions::class]],
[$real, "/index.php/apps/news/api/", ["ncn", "/", \JKingWeb\Arsse\REST\NextcloudNews\Versions::class]],
[$real, "/index%2Ephp/apps/news/api/", ["ncn", "/", \JKingWeb\Arsse\REST\NextcloudNews\Versions::class]],
[$real, "/index.php/apps/news/", []],
[$real, "/index!php/apps/news/api/", []],
[$real, "/tt-rss/api/index.php", ["ttrss_api", "/index.php", \JKingWeb\Arsse\REST\TinyTinyRSS\API::class]],
[$real, "/tt-rss/api", ["ttrss_api", "", \JKingWeb\Arsse\REST\TinyTinyRSS\API::class]],
[$real, "/tt-rss/API", []],
[$real, "/tt-rss/api-bogus", []],
[$real, "/tt-rss/api bogus", []],
[$real, "/tt-rss/feed-icons/", ["ttrss_icon", "", \JKingWeb\Arsse\REST\TinyTinyRSS\Icon::class]],
[$real, "/tt-rss/feed-icons/", ["ttrss_icon", "", \JKingWeb\Arsse\REST\TinyTinyRSS\Icon::class]],
[$real, "/tt-rss/feed-icons", []],
[$fake, "/full/url/", ["unstripped", "/full/url/", "UnstrippedProtocol"]],
[$fake, "/full/url-not", []],
];
}
2018-10-26 18:58:04 +00:00
/** @dataProvider provideAuthenticableRequests */
2020-01-20 18:52:48 +00:00
public function testAuthenticateRequests(array $serverParams, array $expAttr): void {
$r = new REST();
// create a mock user manager
2021-02-27 20:24:02 +00:00
$this->userMock = $this->mock(User::class);
$this->userMock->auth->returns(false);
$this->userMock->auth->with("john.doe@example.com", "secret")->returns(true);
$this->userMock->auth->with("john.doe@example.com", "")->returns(true);
$this->userMock->auth->with("someone.else@example.com", "")->returns(true);
Arsse::$user = $this->userMock->get();
// create an input server request
2022-08-06 17:40:02 +00:00
$req = new ServerRequest("GET", "/", [], null, "1.1", $serverParams);
// create the expected output
$exp = $req;
foreach ($expAttr as $key => $value) {
$exp = $exp->withAttribute($key, $value);
}
$act = $r->authenticateRequest($req);
$this->assertMessage($exp, $act);
}
2019-10-16 18:42:43 +00:00
public function provideAuthenticableRequests(): iterable {
return [
[['PHP_AUTH_USER' => "john.doe@example.com", 'PHP_AUTH_PW' => "secret"], ['authenticated' => true, 'authenticatedUser' => "john.doe@example.com"]],
[['PHP_AUTH_USER' => "john.doe@example.com", 'PHP_AUTH_PW' => "secret", 'REMOTE_USER' => "jane.doe@example.com"], ['authenticated' => true, 'authenticatedUser' => "john.doe@example.com"]],
[['PHP_AUTH_USER' => "jane.doe@example.com", 'PHP_AUTH_PW' => "secret"], ['authenticationFailed' => true]],
[['PHP_AUTH_USER' => "john.doe@example.com", 'PHP_AUTH_PW' => "superman"], ['authenticationFailed' => true]],
[['REMOTE_USER' => "john.doe@example.com"], ['authenticated' => true, 'authenticatedUser' => "john.doe@example.com"]],
[['REMOTE_USER' => "someone.else@example.com"], ['authenticated' => true, 'authenticatedUser' => "someone.else@example.com"]],
[['REMOTE_USER' => "jane.doe@example.com"], ['authenticationFailed' => true]],
[[], []],
];
}
2020-01-20 18:52:48 +00:00
public function testSendAuthenticationChallenges(): void {
2018-11-23 00:55:54 +00:00
self::setConf();
$r = new REST();
$in = HTTP::respEmpty(401);
$exp = $in->withHeader("WWW-Authenticate", 'Basic realm="OOK", charset="UTF-8"');
$act = $r->challenge($in, "OOK");
$this->assertMessage($exp, $act);
$exp = $in->withHeader("WWW-Authenticate", 'Basic realm="'.Arsse::$conf->httpRealm.'", charset="UTF-8"');
$act = $r->challenge($in);
$this->assertMessage($exp, $act);
}
2018-10-26 18:58:04 +00:00
2018-01-09 17:31:40 +00:00
/** @dataProvider provideUnnormalizedOrigins */
2020-01-20 18:52:48 +00:00
public function testNormalizeOrigins(string $origin, string $exp, array $ports = null): void {
2018-01-09 17:31:40 +00:00
$r = new REST();
$act = $r->corsNormalizeOrigin($origin, $ports);
$this->assertSame($exp, $act);
}
2019-10-16 18:42:43 +00:00
public function provideUnnormalizedOrigins(): iterable {
2018-01-09 17:31:40 +00:00
return [
["null", "null"],
["http://example.com", "http://example.com"],
["http://example.com:80", "http://example.com"],
["http://example.com:8%30", "http://example.com"],
["http://example.com:8080", "http://example.com:8080"],
["http://[2001:0db8:0:0:0:0:2:1]", "http://[2001:db8::2:1]"],
["http://example", "http://example"],
["http://ex%41mple", "http://example"],
["http://ex%41mple.co.uk", "http://example.co.uk"],
["http://ex%41mple.co%2euk", "http://example.co%2Euk"],
["http://example/", ""],
["http://example?", ""],
["http://example#", ""],
["http://user@example", ""],
["http://user:pass@example", ""],
["http://[example", ""],
["http://[2bef]", ""],
["http://example%2F", "http://example%2F"],
["HTTP://example", "http://example"],
["HTTP://EXAMPLE", "http://example"],
["%48%54%54%50://example", "http://example"],
["http:%2F%2Fexample", ""],
["https://example", "https://example"],
["https://example:443", "https://example"],
["https://example:80", "https://example:80"],
["ssh://example", "ssh://example"],
["ssh://example:22", "ssh://example:22"],
["ssh://example:22", "ssh://example", ['ssh' => 22]],
["SSH://example:22", "ssh://example", ['ssh' => 22]],
["ssh://example:22", "ssh://example", ['ssh' => "22"]],
["ssh://example:22", "ssh://example:22", ['SSH' => "22"]],
];
}
/** @dataProvider provideCorsNegotiations */
2020-01-20 18:52:48 +00:00
public function testNegotiateCors($origin, bool $exp, string $allowed = null, string $denied = null): void {
2018-11-23 00:55:54 +00:00
self::setConf();
2021-02-27 20:24:02 +00:00
$rMock = $this->partialMock(REST::class);
$rMock->corsNormalizeOrigin->does(function($origin) {
2018-01-09 17:31:40 +00:00
return $origin;
});
2018-10-26 20:27:18 +00:00
$headers = isset($origin) ? ['Origin' => $origin] : [];
2022-08-06 17:40:02 +00:00
$req = new Request("GET", "", $headers);
2021-02-27 20:24:02 +00:00
$act = $rMock->get()->corsNegotiate($req, $allowed, $denied);
2018-01-09 17:31:40 +00:00
$this->assertSame($exp, $act);
}
2019-10-16 18:42:43 +00:00
public function provideCorsNegotiations(): iterable {
2018-01-09 17:31:40 +00:00
return [
["http://example", true ],
["http://example", true, "http://example", "*" ],
["http://example", false, "http://example", "http://example"],
["http://example", false, "https://example", "*" ],
["http://example", false, "*", "*" ],
["http://example", true, "*", "" ],
["http://example", false, "", "" ],
["null", false ],
["null", true, "null", "*" ],
["null", false, "null", "null" ],
["null", false, "*", "*" ],
["null", false, "*", "" ],
["null", false, "", "" ],
["", false ],
["", false, "", "*" ],
["", false, "", "" ],
["", false, "*", "*" ],
["", false, "*", "" ],
[["null", "http://example"], false, "*", "" ],
2018-10-26 20:27:18 +00:00
[null, false, "*", "" ],
2018-01-09 17:31:40 +00:00
];
}
/** @dataProvider provideCorsHeaders */
2020-01-20 18:52:48 +00:00
public function testAddCorsHeaders(string $reqMethod, array $reqHeaders, array $resHeaders, array $expHeaders): void {
2018-01-09 17:31:40 +00:00
$r = new REST();
2022-08-06 17:40:02 +00:00
$req = new Request($reqMethod, "php://memory", $reqHeaders);
$res = HTTP::respEmpty(204, $resHeaders);
$exp = HTTP::respEmpty(204, $expHeaders);
2018-01-09 17:31:40 +00:00
$act = $r->corsApply($res, $req);
$this->assertMessage($exp, $act);
2018-01-09 17:31:40 +00:00
}
2019-10-16 18:42:43 +00:00
public function provideCorsHeaders(): iterable {
2018-01-09 17:31:40 +00:00
return [
["GET", ['Origin' => "null"], [], [
2020-03-01 20:16:50 +00:00
'Access-Control-Allow-Origin' => "null",
2018-01-09 17:31:40 +00:00
'Access-Control-Allow-Credentials' => "true",
2020-03-01 20:16:50 +00:00
'Vary' => "Origin",
2018-01-09 17:31:40 +00:00
]],
["GET", ['Origin' => "http://example"], [], [
2020-03-01 20:16:50 +00:00
'Access-Control-Allow-Origin' => "http://example",
2018-01-09 17:31:40 +00:00
'Access-Control-Allow-Credentials' => "true",
2020-03-01 20:16:50 +00:00
'Vary' => "Origin",
2018-01-09 17:31:40 +00:00
]],
["GET", ['Origin' => "http://example"], ['Content-Type' => "text/plain; charset=utf-8"], [
2020-03-01 20:16:50 +00:00
'Access-Control-Allow-Origin' => "http://example",
2018-01-09 17:31:40 +00:00
'Access-Control-Allow-Credentials' => "true",
2020-03-01 20:16:50 +00:00
'Vary' => "Origin",
'Content-Type' => "text/plain; charset=utf-8",
2018-01-09 17:31:40 +00:00
]],
["GET", ['Origin' => "http://example"], ['Vary' => "Content-Type"], [
2020-03-01 20:16:50 +00:00
'Access-Control-Allow-Origin' => "http://example",
2018-01-09 17:31:40 +00:00
'Access-Control-Allow-Credentials' => "true",
2020-03-01 20:16:50 +00:00
'Vary' => ["Content-Type", "Origin"],
2018-01-09 17:31:40 +00:00
]],
["OPTIONS", ['Origin' => "http://example"], [], [
2020-03-01 20:16:50 +00:00
'Access-Control-Allow-Origin' => "http://example",
2018-01-09 17:31:40 +00:00
'Access-Control-Allow-Credentials' => "true",
2020-03-01 20:16:50 +00:00
'Access-Control-Max-Age' => (string) (60 * 60 * 24),
'Vary' => "Origin",
2018-01-09 17:31:40 +00:00
]],
["OPTIONS", ['Origin' => "http://example"], ['Allow' => "GET, PUT, HEAD, OPTIONS"], [
2020-03-01 20:16:50 +00:00
'Allow' => "GET, PUT, HEAD, OPTIONS",
'Access-Control-Allow-Origin' => "http://example",
2018-01-09 17:31:40 +00:00
'Access-Control-Allow-Credentials' => "true",
2020-03-01 20:16:50 +00:00
'Access-Control-Allow-Methods' => "GET, PUT, HEAD, OPTIONS",
'Access-Control-Max-Age' => (string) (60 * 60 * 24),
'Vary' => "Origin",
2018-01-09 17:31:40 +00:00
]],
["OPTIONS", ['Origin' => "http://example", 'Access-Control-Request-Headers' => "Content-Type, If-None-Match"], [], [
2020-03-01 20:16:50 +00:00
'Access-Control-Allow-Origin' => "http://example",
2018-01-09 17:31:40 +00:00
'Access-Control-Allow-Credentials' => "true",
2020-03-01 20:16:50 +00:00
'Access-Control-Allow-Headers' => "Content-Type, If-None-Match",
'Access-Control-Max-Age' => (string) (60 * 60 * 24),
'Vary' => "Origin",
2018-01-09 17:31:40 +00:00
]],
["OPTIONS", ['Origin' => "http://example", 'Access-Control-Request-Headers' => ["Content-Type", "If-None-Match"]], [], [
2020-03-01 20:16:50 +00:00
'Access-Control-Allow-Origin' => "http://example",
2018-01-09 17:31:40 +00:00
'Access-Control-Allow-Credentials' => "true",
'Access-Control-Allow-Headers' => "Content-Type, If-None-Match",
2020-03-01 20:16:50 +00:00
'Access-Control-Max-Age' => (string) (60 * 60 * 24),
'Vary' => "Origin",
2018-01-09 17:31:40 +00:00
]],
];
}
/** @dataProvider provideUnnormalizedResponses */
2020-01-20 18:52:48 +00:00
public function testNormalizeHttpResponses(ResponseInterface $res, ResponseInterface $exp, RequestInterface $req = null): void {
2021-02-27 20:24:02 +00:00
$rMock = $this->partialMock(REST::class);
$rMock->corsNegotiate->returns(true);
$rMock->challenge->does(function($res) {
return $res->withHeader("WWW-Authenticate", "Fake Value");
});
2021-02-27 20:24:02 +00:00
$rMock->corsApply->does(function($res) {
2018-01-09 17:31:40 +00:00
return $res;
});
2021-02-27 20:24:02 +00:00
$act = $rMock->get()->normalizeResponse($res, $req);
$this->assertMessage($exp, $act);
}
2019-10-16 18:42:43 +00:00
public function provideUnnormalizedResponses(): iterable {
$stream = fopen("php://memory", "w+b");
2018-01-12 14:48:33 +00:00
fwrite($stream, "ook");
return [
[HTTP::respEmpty(204), HTTP::respEmpty(204)],
[HTTP::respEmpty(401), HTTP::respEmpty(401, ['WWW-Authenticate' => "Fake Value"])],
[HTTP::respEmpty(204, ['Allow' => "PUT"]), HTTP::respEmpty(204, ['Allow' => "PUT, OPTIONS"])],
[HTTP::respEmpty(204, ['Allow' => "PUT, OPTIONS"]), HTTP::respEmpty(204, ['Allow' => "PUT, OPTIONS"])],
[HTTP::respEmpty(204, ['Allow' => "PUT,OPTIONS"]), HTTP::respEmpty(204, ['Allow' => "PUT, OPTIONS"])],
[HTTP::respEmpty(204, ['Allow' => ["PUT", "OPTIONS"]]), HTTP::respEmpty(204, ['Allow' => "PUT, OPTIONS"])],
[HTTP::respEmpty(204, ['Allow' => ["PUT, DELETE", "OPTIONS"]]), HTTP::respEmpty(204, ['Allow' => "PUT, DELETE, OPTIONS"])],
[HTTP::respEmpty(204, ['Allow' => "HEAD,GET"]), HTTP::respEmpty(204, ['Allow' => "HEAD, GET, OPTIONS"])],
[HTTP::respEmpty(204, ['Allow' => "GET"]), HTTP::respEmpty(204, ['Allow' => "GET, HEAD, OPTIONS"])],
2022-08-06 17:40:02 +00:00
[HTTP::respText("ook", 200), HTTP::respText("ook", 200, ['Content-Length' => "3"])],
[HTTP::respText("", 200), HTTP::respText("", 200, ['Content-Length' => "0"])],
[HTTP::respText("ook", 404), HTTP::respText("ook", 404, ['Content-Length' => "3"])],
[HTTP::respText("", 404), HTTP::respText("", 404)],
[new Response(200, [], $stream), new Response(200, ['Content-Length' => "3"], $stream), new Request("GET", "")],
[new Response(200, [], $stream), HTTP::respEmpty(200, ['Content-Length' => "3"]), new Request("HEAD", "")],
];
}
/** @dataProvider provideMockRequests */
2020-03-01 20:16:50 +00:00
public function testDispatchRequests(ServerRequest $req, string $method, bool $called, string $class = "", string $target = ""): void {
2021-02-27 20:24:02 +00:00
$rMock = $this->partialMock(REST::class);
$rMock->normalizeResponse->does(function($res) {
2018-01-09 17:31:40 +00:00
return $res;
});
2021-02-27 20:24:02 +00:00
$rMock->authenticateRequest->does(function($req) {
return $req;
});
if ($called) {
2021-02-27 20:24:02 +00:00
$hMock = $this->mock($class);
$hMock->dispatch->returns(HTTP::respEmpty(204));
2021-02-27 20:24:02 +00:00
$this->objMock->get->with($class)->returns($hMock);
Arsse::$obj = $this->objMock->get();
}
2021-02-27 20:24:02 +00:00
$out = $rMock->get()->dispatch($req);
$this->assertInstanceOf(ResponseInterface::class, $out);
if ($called) {
2021-02-27 20:24:02 +00:00
$rMock->authenticateRequest->called();
$hMock->dispatch->once()->called();
2021-07-06 01:47:44 +00:00
$in = $hMock->dispatch->firstCall()->argument();
$this->assertSame($method, $in->getMethod());
$this->assertSame($target, $in->getRequestTarget());
} else {
$this->assertSame(501, $out->getStatusCode());
}
2021-02-27 20:24:02 +00:00
$rMock->apiMatch->called();
$rMock->normalizeResponse->called();
}
2019-10-16 18:42:43 +00:00
public function provideMockRequests(): iterable {
return [
2022-08-06 17:40:02 +00:00
[new ServerRequest("GET", "/index.php/apps/news/api/v1-2/feeds"), "GET", true, NCN::class, "/feeds"],
[new ServerRequest("GET", "/index.php/apps/news/api/v1-2/feeds"), "GET", true, NCN::class, "/feeds"],
[new ServerRequest("get", "/index.php/apps/news/api/v1-2/feeds"), "GET", true, NCN::class, "/feeds"],
[new ServerRequest("head", "/index.php/apps/news/api/v1-2/feeds"), "GET", true, NCN::class, "/feeds"],
[new ServerRequest("POST", "/tt-rss/api/"), "POST", true, TTRSS::class, "/"],
[new ServerRequest("HEAD", "/no/such/api/"), "GET", false],
[new ServerRequest("GET", "/no/such/api/"), "GET", false],
];
}
2018-01-12 14:48:33 +00:00
}