mirror of
https://github.com/danieleteti/delphimvcframework.git
synced 2024-11-16 08:15:53 +01:00
1062 lines
30 KiB
ObjectPascal
1062 lines
30 KiB
ObjectPascal
// ***************************************************************************
|
||
//
|
||
// Delphi MVC Framework
|
||
//
|
||
// Copyright (c) 2010-2017 Daniele Teti and the DMVCFramework Team
|
||
//
|
||
// https://github.com/danieleteti/delphimvcframework
|
||
//
|
||
// ***************************************************************************
|
||
//
|
||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||
// you may not use this file except in compliance with the License.
|
||
// You may obtain a copy of the License at
|
||
//
|
||
// http://www.apache.org/licenses/LICENSE-2.0
|
||
//
|
||
// Unless required by applicable law or agreed to in writing, software
|
||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
// See the License for the specific language governing permissions and
|
||
// limitations under the License.
|
||
//
|
||
// *************************************************************************** }
|
||
|
||
unit LiveServerTestU;
|
||
|
||
interface
|
||
|
||
uses
|
||
TestFramework,
|
||
MVCFramework.RESTClient;
|
||
|
||
const
|
||
|
||
{$IFDEF LINUX_SERVER}
|
||
|
||
TEST_SERVER_ADDRESS = '192.168.3.88';
|
||
|
||
{$ELSE}
|
||
|
||
TEST_SERVER_ADDRESS = '127.0.0.1';
|
||
|
||
{$ENDIF}
|
||
|
||
type
|
||
TBaseServerTest = class(TTestCase)
|
||
protected
|
||
RESTClient: TRESTClient;
|
||
procedure DoLoginWith(UserName: string);
|
||
procedure DoLogout;
|
||
|
||
protected
|
||
procedure SetUp; override;
|
||
procedure TearDown; override;
|
||
|
||
end;
|
||
|
||
TServerTest = class(TBaseServerTest)
|
||
published
|
||
procedure TestReqWithParams;
|
||
procedure TestPOSTWithParamsAndJSONBody;
|
||
// procedure TestPATCHWithParamsAndJSONBody;
|
||
procedure TestPOSTWithObjectJSONBody;
|
||
procedure TestPUTWithParamsAndJSONBody;
|
||
procedure TestCookies;
|
||
procedure TestSessionWithLogin;
|
||
procedure TestSession;
|
||
procedure TestInvalidateSession;
|
||
procedure TestAsynchRequestPOST;
|
||
procedure TestAsynchRequestPUT;
|
||
procedure TestAsynchRequestGET;
|
||
procedure TestAsynchRequestDELETE;
|
||
procedure TestEncodingRenderJSONValue;
|
||
procedure TestRenderWrappedList;
|
||
procedure TestRenderStreamAndFreeWithOwnerFalse;
|
||
procedure TestRenderStreamAndFreeWithOwnerTrue;
|
||
// procedure TestSerializationType;
|
||
procedure TestProducesConsumes01;
|
||
procedure TestProducesConsumes02;
|
||
procedure TestProducesConsumes03;
|
||
procedure TestProducesConsumesWithWrongAcceptHeader;
|
||
procedure TestExceptionInMVCAfterCreate;
|
||
procedure TestExceptionInMVCBeforeDestroy;
|
||
procedure TestActionFiltersOnBeforeAction;
|
||
procedure TestMiddlewareSpeedMiddleware;
|
||
procedure TestMiddlewareHandler;
|
||
procedure TestPostAListOfObjects;
|
||
// test authentication/authorization with BasicAuth
|
||
procedure TestBasicAuth01;
|
||
procedure TestBasicAuth02;
|
||
procedure TestBasicAuth03;
|
||
procedure TestBasicAuth04;
|
||
procedure TestBasicAuth05;
|
||
|
||
// test authentication/authorization with CustomAuth
|
||
procedure TestCustomAuthRequestWithoutLogin;
|
||
procedure TestCustomAuthRequestsWithValidLogin;
|
||
procedure TestCustomAuthRequestsWithValidLogin_HTML;
|
||
procedure TestCustomAuthWrongRequestBodies;
|
||
procedure TestCustomAuthLoginLogout;
|
||
|
||
// typed actions
|
||
procedure TestTypedString1;
|
||
procedure TestTypedInteger1;
|
||
procedure TestTypedInt641;
|
||
procedure TestTypedSingle1;
|
||
procedure TestTypedDouble1;
|
||
procedure TestTypedExtended1;
|
||
procedure TestTypedAll;
|
||
procedure TestTypedDateTimeTypes;
|
||
procedure TestTypedBooleans;
|
||
end;
|
||
|
||
implementation
|
||
|
||
uses
|
||
System.Math,
|
||
|
||
{$IF CompilerVersion < 27}
|
||
|
||
Data.DBXJSON,
|
||
|
||
{$ELSE}
|
||
|
||
System.JSON,
|
||
|
||
{$ENDIF}
|
||
|
||
MVCFramework.Commons,
|
||
System.SyncObjs,
|
||
System.Generics.Collections,
|
||
System.SysUtils,
|
||
BusinessObjectsU,
|
||
ObjectsMappers,
|
||
Soap.EncdDecd, System.Classes;
|
||
|
||
{ TServerTest }
|
||
|
||
procedure TBaseServerTest.DoLogout;
|
||
var
|
||
res: IRESTResponse;
|
||
begin
|
||
res := RESTClient.doGET('/logout', []);
|
||
CheckTrue(res.ResponseCode = HTTP_STATUS.OK, 'Logout Failed');
|
||
end;
|
||
|
||
procedure TBaseServerTest.SetUp;
|
||
begin
|
||
inherited;
|
||
RESTClient := TRESTClient.Create(TEST_SERVER_ADDRESS, 9999);
|
||
RESTClient.ReadTimeout(60 * 1000 * 30);
|
||
end;
|
||
|
||
procedure TBaseServerTest.TearDown;
|
||
begin
|
||
inherited;
|
||
RESTClient.Free;
|
||
end;
|
||
|
||
procedure TServerTest.TestActionFiltersOnBeforeAction;
|
||
var
|
||
res: IRESTResponse;
|
||
begin
|
||
res := RESTClient.doGET('/actionfilters/beforeaction/alwayscalled', []);
|
||
CheckEquals(HTTP_STATUS.OK, res.ResponseCode);
|
||
|
||
res := RESTClient.doGET('/actionfilters/beforeaction/nevercalled', []);
|
||
CheckEquals(HTTP_STATUS.NotFound, res.ResponseCode);
|
||
end;
|
||
|
||
procedure TServerTest.TestAsynchRequestDELETE;
|
||
var
|
||
evt: TEvent;
|
||
r: TWaitResult;
|
||
OK: boolean;
|
||
begin
|
||
OK := false;
|
||
evt := TEvent.Create;
|
||
try
|
||
RESTClient.Asynch(
|
||
procedure(Response: IRESTResponse)
|
||
begin
|
||
OK := true;
|
||
evt.SetEvent;
|
||
end,
|
||
procedure(E: Exception)
|
||
begin
|
||
OK := false;
|
||
end).doDELETE('/req/with/params', ['1', '2', '3']);
|
||
|
||
// wait for thred finish
|
||
repeat
|
||
r := evt.WaitFor(2000);
|
||
until r = TWaitResult.wrSignaled;
|
||
|
||
CheckEquals(true, OK);
|
||
finally
|
||
evt.Free;
|
||
end;
|
||
end;
|
||
|
||
procedure TServerTest.TestAsynchRequestGET;
|
||
var
|
||
evt: TEvent;
|
||
r: TWaitResult;
|
||
j: TJSONObject;
|
||
begin
|
||
j := nil;
|
||
evt := TEvent.Create;
|
||
try
|
||
RESTClient.Asynch(
|
||
procedure(Response: IRESTResponse)
|
||
begin
|
||
try
|
||
j := Response.BodyAsJsonObject.Clone as TJSONObject;
|
||
except
|
||
// test should not block...never!
|
||
end;
|
||
evt.SetEvent;
|
||
end,
|
||
procedure(E: Exception)
|
||
begin
|
||
end).doGET('/req/with/params', ['1', '2', '3']);
|
||
|
||
// wait for thred finish
|
||
repeat
|
||
r := evt.WaitFor(2000);
|
||
until r = TWaitResult.wrSignaled;
|
||
|
||
CheckTrue(Assigned(j));
|
||
CheckEquals('1', j.Get('par1').JsonValue.Value);
|
||
j.Free;
|
||
finally
|
||
evt.Free;
|
||
end;
|
||
end;
|
||
|
||
procedure TServerTest.TestAsynchRequestPOST;
|
||
var
|
||
evt: TEvent;
|
||
r: TWaitResult;
|
||
j: TJSONObject;
|
||
begin
|
||
j := nil;
|
||
evt := TEvent.Create;
|
||
try
|
||
RESTClient.Asynch(
|
||
procedure(Response: IRESTResponse)
|
||
begin
|
||
try
|
||
j := Response.BodyAsJsonObject.Clone as TJSONObject;
|
||
except
|
||
// test should not block...never!
|
||
end;
|
||
evt.SetEvent;
|
||
end,
|
||
procedure(E: Exception)
|
||
begin
|
||
end).doPOST('/echo', ['1', '2', '3'],
|
||
TJSONObject.Create(TJSONPair.Create('from client', 'hello world')), true);
|
||
|
||
// wait for thred finish
|
||
repeat
|
||
r := evt.WaitFor(2000);
|
||
until r = TWaitResult.wrSignaled;
|
||
|
||
CheckTrue(Assigned(j));
|
||
CheckEquals('from server', j.Get('echo').JsonValue.Value);
|
||
j.Free;
|
||
finally
|
||
evt.Free;
|
||
end;
|
||
end;
|
||
|
||
procedure TServerTest.TestAsynchRequestPUT;
|
||
var
|
||
evt: TEvent;
|
||
r: TWaitResult;
|
||
j: TJSONObject;
|
||
begin
|
||
j := nil;
|
||
evt := TEvent.Create;
|
||
try
|
||
RESTClient.Asynch(
|
||
procedure(Response: IRESTResponse)
|
||
begin
|
||
try
|
||
j := Response.BodyAsJsonObject.Clone as TJSONObject;
|
||
except
|
||
// test should not block...never!
|
||
end;
|
||
evt.SetEvent;
|
||
end,
|
||
procedure(E: Exception)
|
||
begin
|
||
end).doPUT('/echo', ['1', '2', '3'],
|
||
TJSONObject.Create(TJSONPair.Create('from client', 'hello world')), true);
|
||
|
||
// wait for thred finish
|
||
repeat
|
||
r := evt.WaitFor(2000);
|
||
until r = TWaitResult.wrSignaled;
|
||
|
||
CheckTrue(Assigned(j));
|
||
CheckEquals('from server', j.Get('echo').JsonValue.Value);
|
||
j.Free;
|
||
finally
|
||
evt.Free;
|
||
end;
|
||
end;
|
||
|
||
procedure TServerTest.TestBasicAuth01;
|
||
var
|
||
LRes: IRESTResponse;
|
||
begin
|
||
RESTClient.Authentication('user1', 'user1');
|
||
CheckEquals('user1', RESTClient.UserName);
|
||
CheckEquals('user1', RESTClient.Password);
|
||
LRes := RESTClient.doGET('/private/role1', []);
|
||
CheckEquals(HTTP_STATUS.OK, LRes.ResponseCode);
|
||
end;
|
||
|
||
procedure TServerTest.TestBasicAuth02;
|
||
var
|
||
LRes: IRESTResponse;
|
||
begin
|
||
RESTClient.UserName := '';
|
||
RESTClient.Password := '';
|
||
RESTClient.UseBasicAuthentication := false;
|
||
LRes := RESTClient.doGET('/private/role1', []);
|
||
CheckEquals(HTTP_STATUS.Unauthorized, LRes.ResponseCode);
|
||
end;
|
||
|
||
procedure TServerTest.TestBasicAuth03;
|
||
var
|
||
LRes: IRESTResponse;
|
||
begin
|
||
RESTClient.UserName := 'user1';
|
||
RESTClient.Password := 'user1';
|
||
RESTClient.UseBasicAuthentication := true;
|
||
LRes := RESTClient.doGET('/private/role2', []);
|
||
CheckEquals(HTTP_STATUS.Forbidden, LRes.ResponseCode);
|
||
end;
|
||
|
||
procedure TServerTest.TestBasicAuth04;
|
||
var
|
||
LRes: IRESTResponse;
|
||
begin
|
||
RESTClient.UserName := 'user1';
|
||
RESTClient.Password := 'user1';
|
||
RESTClient.UseBasicAuthentication := true;
|
||
LRes := RESTClient.doGET('/private/role1', []);
|
||
CheckEquals(HTTP_STATUS.OK, LRes.ResponseCode);
|
||
LRes := RESTClient.doGET('/people', []);
|
||
CheckEquals(HTTP_STATUS.OK, LRes.ResponseCode);
|
||
end;
|
||
|
||
procedure TServerTest.TestBasicAuth05;
|
||
var
|
||
LRes: IRESTResponse;
|
||
begin
|
||
RESTClient.UserName := 'user1';
|
||
RESTClient.Password := 'user1';
|
||
RESTClient.UseBasicAuthentication := true;
|
||
|
||
// first
|
||
LRes := RESTClient.doGET('/private/role1session?value=danieleteti', []);
|
||
CheckEquals(HTTP_STATUS.OK, LRes.ResponseCode);
|
||
LRes := RESTClient.doGET('/private/role1session', []);
|
||
CheckEquals(HTTP_STATUS.OK, LRes.ResponseCode);
|
||
CheckEquals('danieleteti', LRes.BodyAsString);
|
||
|
||
// second
|
||
LRes := RESTClient.doGET('/private/role1session?value=johndoe', []);
|
||
CheckEquals(HTTP_STATUS.OK, LRes.ResponseCode);
|
||
LRes := RESTClient.doGET('/private/role1session', []);
|
||
CheckEquals(HTTP_STATUS.OK, LRes.ResponseCode);
|
||
CheckEquals('johndoe', LRes.BodyAsString);
|
||
end;
|
||
|
||
procedure TServerTest.TestCookies;
|
||
var
|
||
res: IRESTResponse;
|
||
I: Integer;
|
||
begin
|
||
res := RESTClient.doGET('/lotofcookies', []);
|
||
CheckEquals(HTTP_STATUS.OK, res.ResponseCode);
|
||
CheckEquals(4, res.Cookies.Count, 'Wrong number of cookies');
|
||
for I := 0 to 3 do
|
||
begin
|
||
CheckEquals('usersettings' + IntToStr(I + 1),
|
||
res.Cookies.Cookies[I].CookieName);
|
||
CheckEquals('usersettings' + IntToStr(I + 1) + '-value',
|
||
res.Cookies.Cookies[I].Value);
|
||
CheckEquals('/usersettings' + IntToStr(I + 1), res.Cookies.Cookies[I].Path);
|
||
end;
|
||
|
||
end;
|
||
|
||
procedure TServerTest.TestCustomAuthRequestWithoutLogin;
|
||
var
|
||
LRes: IRESTResponse;
|
||
begin
|
||
LRes := RESTClient.doGET('/privatecustom/role1', []);
|
||
CheckEquals(HTTP_STATUS.Unauthorized, LRes.ResponseCode);
|
||
CheckEquals('/system/users/logged', LRes.HeaderValue('X-LOGIN-URL'));
|
||
CheckEquals('POST', LRes.HeaderValue('X-LOGIN-METHOD'));
|
||
|
||
LRes := RESTClient.doGET('/privatecustom/role2', []);
|
||
CheckEquals(HTTP_STATUS.Unauthorized, LRes.ResponseCode);
|
||
CheckEquals('/system/users/logged', LRes.HeaderValue('X-LOGIN-URL'));
|
||
CheckEquals('POST', LRes.HeaderValue('X-LOGIN-METHOD'));
|
||
end;
|
||
|
||
procedure TServerTest.TestCustomAuthRequestsWithValidLogin;
|
||
var
|
||
LRes: IRESTResponse;
|
||
lJSON: TJSONObject;
|
||
lCookieValue: string;
|
||
begin
|
||
lJSON := TJSONObject.Create;
|
||
try
|
||
lJSON.AddPair('username', 'user1');
|
||
lJSON.AddPair('password', 'user1');
|
||
LRes := RESTClient.doPOST('/system/users/logged', [], lJSON, false);
|
||
CheckEquals('application/json', LRes.ContentType);
|
||
CheckEquals(HTTP_STATUS.OK, LRes.ResponseCode);
|
||
CheckEquals('/system/users/logged', LRes.HeaderValue('X-LOGOUT-URL'));
|
||
CheckEquals('DELETE', LRes.HeaderValue('X-LOGOUT-METHOD'));
|
||
CheckEquals('{"status":"OK"}', LRes.BodyAsString);
|
||
lCookieValue := LRes.Cookies
|
||
[LRes.Cookies.GetCookieIndex(TMVCConstants.SESSION_TOKEN_NAME)].Value;
|
||
CheckNotEquals('', lCookieValue, 'Session cookie not returned after login');
|
||
CheckFalse(lCookieValue.Contains('invalid'),
|
||
'Returned an invalid session token');
|
||
|
||
LRes := RESTClient.doGET('/privatecustom/role2', []);
|
||
CheckEquals(HTTP_STATUS.Forbidden, LRes.ResponseCode,
|
||
'Authorization not respected for not allowed action');
|
||
|
||
LRes := RESTClient.doGET('/privatecustom/role1', []);
|
||
CheckEquals(HTTP_STATUS.OK, LRes.ResponseCode,
|
||
'Authorization not respected for allowed action');
|
||
finally
|
||
lJSON.Free;
|
||
end;
|
||
end;
|
||
|
||
procedure TServerTest.TestCustomAuthRequestsWithValidLogin_HTML;
|
||
var
|
||
LRes: IRESTResponse;
|
||
lJSON: TJSONObject;
|
||
lCookieValue: string;
|
||
begin
|
||
lJSON := TJSONObject.Create;
|
||
try
|
||
lJSON.AddPair('username', 'user1');
|
||
lJSON.AddPair('password', 'user1');
|
||
LRes := RESTClient.Accept('text/html').doPOST('/system/users/logged', [], lJSON, false);
|
||
CheckEquals('application/json', LRes.ContentType);
|
||
CheckEquals(HTTP_STATUS.OK, LRes.ResponseCode);
|
||
CheckEquals('/system/users/logged', LRes.HeaderValue('X-LOGOUT-URL'));
|
||
CheckEquals('DELETE', LRes.HeaderValue('X-LOGOUT-METHOD'));
|
||
CheckEquals('{"status":"OK"}', LRes.BodyAsString);
|
||
lCookieValue := LRes.Cookies
|
||
[LRes.Cookies.GetCookieIndex(TMVCConstants.SESSION_TOKEN_NAME)].Value;
|
||
CheckNotEquals('', lCookieValue, 'Session cookie not returned after login');
|
||
CheckFalse(lCookieValue.Contains('invalid'),
|
||
'Returned an invalid session token');
|
||
|
||
LRes := RESTClient.doGET('/privatecustom/role2', []);
|
||
CheckEquals(HTTP_STATUS.Forbidden, LRes.ResponseCode,
|
||
'Authorization not respected for not allowed action');
|
||
|
||
LRes := RESTClient.doGET('/privatecustom/role1', []);
|
||
CheckEquals(HTTP_STATUS.OK, LRes.ResponseCode,
|
||
'Authorization not respected for allowed action');
|
||
finally
|
||
lJSON.Free;
|
||
end;
|
||
end;
|
||
|
||
procedure TServerTest.TestCustomAuthWrongRequestBodies;
|
||
var
|
||
LRes: IRESTResponse;
|
||
lJSON: TJSONObject;
|
||
begin
|
||
lJSON := TJSONObject.Create;
|
||
try
|
||
// no request body
|
||
LRes := RESTClient.doPOST('/system/users/logged', []);
|
||
CheckEquals(HTTP_STATUS.BadRequest, LRes.ResponseCode,
|
||
'Empty request body doesn''t return HTTP 400 Bad Request');
|
||
|
||
// wrong request body 1
|
||
LRes := RESTClient.doPOST('/system/users/logged', [], lJSON, false);
|
||
CheckEquals(HTTP_STATUS.Unauthorized, LRes.ResponseCode,
|
||
'Invalid json doesn''t return HTTP 401 Unauthorized');
|
||
|
||
// wrong request body 2
|
||
lJSON.AddPair('username', '');
|
||
lJSON.AddPair('password', '');
|
||
LRes := RESTClient.doPOST('/system/users/logged', [], lJSON, false);
|
||
CheckEquals(HTTP_STATUS.Unauthorized, LRes.ResponseCode,
|
||
'Empty username and password doesn''t return HTTP 401 Unauthorized');
|
||
|
||
// wrong username and password 3
|
||
lJSON.RemovePair('username');
|
||
lJSON.RemovePair('password');
|
||
lJSON.AddPair('username', 'notvaliduser');
|
||
lJSON.AddPair('password', 'notvalidpassword');
|
||
LRes := RESTClient.doPOST('/system/users/logged', [], lJSON, false);
|
||
CheckEquals(HTTP_STATUS.Unauthorized, LRes.ResponseCode,
|
||
'Wrong username and password doesn''t return HTTP 401 Unauthorized');
|
||
finally
|
||
lJSON.Free;
|
||
end;
|
||
end;
|
||
|
||
procedure TServerTest.TestCustomAuthLoginLogout;
|
||
var
|
||
LRes: IRESTResponse;
|
||
lJSON: TJSONObject;
|
||
lLogoutUrl: string;
|
||
lValue: string;
|
||
I: Integer;
|
||
lPieces: TArray<string>;
|
||
lPass: boolean;
|
||
begin
|
||
lJSON := TJSONObject.Create;
|
||
try
|
||
lJSON.AddPair('username', 'user1');
|
||
lJSON.AddPair('password', 'user1');
|
||
LRes := RESTClient.doPOST('/system/users/logged', [], lJSON, false);
|
||
|
||
CheckEquals(HTTP_STATUS.OK, LRes.ResponseCode);
|
||
lLogoutUrl := LRes.HeaderValue('X-LOGOUT-URL');
|
||
|
||
LRes := RESTClient.doDELETE(lLogoutUrl, []);
|
||
lPass := false;
|
||
for I := 0 to LRes.Headers.Count do
|
||
begin
|
||
lValue := LRes.Headers[I];
|
||
if lValue.StartsWith('Set-Cookie') then
|
||
begin
|
||
lPieces := lValue.Split([':']);
|
||
lValue := lPieces[1].Trim;
|
||
if lValue.StartsWith(TMVCConstants.SESSION_TOKEN_NAME) and
|
||
lValue.Contains('invalid') then
|
||
begin
|
||
lPass := true;
|
||
Break;
|
||
end;
|
||
end;
|
||
end;
|
||
CheckTrue(lPass, 'No session cookie cleanup in the response');
|
||
finally
|
||
lJSON.Free;
|
||
end;
|
||
|
||
end;
|
||
|
||
procedure TServerTest.TestEncodingRenderJSONValue;
|
||
var
|
||
res: IRESTResponse;
|
||
s: string;
|
||
begin
|
||
res := RESTClient.doGET('/encoding', []);
|
||
|
||
s := res.BodyAsJsonObject.Get('name1').JsonValue.Value;
|
||
CheckEquals('j<>rn', s);
|
||
|
||
s := res.BodyAsJsonObject.Get('name3').JsonValue.Value;
|
||
CheckEquals('<27><><EFBFBD><EFBFBD><EFBFBD><EFBFBD>', s);
|
||
|
||
s := res.BodyAsJsonObject.Get('name2').JsonValue.Value;
|
||
CheckEquals('<27>to je Unicode?', s,
|
||
'If this test fail, check http://qc.embarcadero.com/wc/qcmain.aspx?d=119779');
|
||
{ WARNING!!! }
|
||
{
|
||
If this test fail, check
|
||
http://qc.embarcadero.com/wc/qcmain.aspx?d=119779
|
||
}
|
||
end;
|
||
|
||
procedure TServerTest.TestRenderStreamAndFreeWithOwnerFalse;
|
||
var
|
||
LRes: IRESTResponse;
|
||
begin
|
||
LRes := RESTClient.doGET('/renderstreamandfreewithownerfalse', []);
|
||
CheckEquals(200, LRes.ResponseCode);
|
||
end;
|
||
|
||
procedure TServerTest.TestRenderStreamAndFreeWithOwnerTrue;
|
||
var
|
||
LRes: IRESTResponse;
|
||
begin
|
||
LRes := RESTClient.doGET('/renderstreamandfreewithownertrue', []);
|
||
CheckEquals(200, LRes.ResponseCode);
|
||
end;
|
||
|
||
procedure TServerTest.TestRenderWrappedList;
|
||
var
|
||
LRes: IRESTResponse;
|
||
lJSONArr: TJSONArray;
|
||
I: Integer;
|
||
lJSONObj: TJSONObject;
|
||
begin
|
||
LRes := RESTClient.doGET('/wrappedpeople', []);
|
||
|
||
lJSONArr := LRes.BodyAsJSONArray;
|
||
for I := 0 to lJSONArr.Count - 1 do
|
||
begin
|
||
lJSONObj := lJSONArr.Items[I] as TJSONObject;
|
||
CheckFalse(lJSONObj.GetValue<string>('firstname').IsEmpty);
|
||
end;
|
||
|
||
end;
|
||
|
||
procedure TServerTest.TestExceptionInMVCAfterCreate;
|
||
var
|
||
res: IRESTResponse;
|
||
begin
|
||
res := RESTClient.doGET('/exception/aftercreate/nevercalled', []);
|
||
CheckEquals(HTTP_STATUS.InternalServerError, res.ResponseCode);
|
||
end;
|
||
|
||
procedure TServerTest.TestExceptionInMVCBeforeDestroy;
|
||
var
|
||
res: IRESTResponse;
|
||
begin
|
||
res := RESTClient.doGET('/exception/beforedestroy/nevercalled', []);
|
||
CheckEquals(HTTP_STATUS.InternalServerError, res.ResponseCode);
|
||
end;
|
||
|
||
procedure TServerTest.TestInvalidateSession;
|
||
var
|
||
c1: TRESTClient;
|
||
res: IRESTResponse;
|
||
begin
|
||
c1 := TRESTClient.Create(TEST_SERVER_ADDRESS, 9999);
|
||
try
|
||
c1.Accept(TMVCMediaType.APPLICATION_JSON);
|
||
c1.doPOST('/session', ['daniele teti']); // imposto un valore in sessione
|
||
res := c1.doGET('/session', []); // rileggo il valore dalla sessione
|
||
CheckEquals('daniele teti', res.BodyAsString);
|
||
c1.SessionID := '';
|
||
res := c1.doGET('/session', []); // rileggo il valore dalla sessione
|
||
CheckEquals('', res.BodyAsString);
|
||
finally
|
||
c1.Free;
|
||
end;
|
||
end;
|
||
|
||
procedure TServerTest.TestMiddlewareHandler;
|
||
var
|
||
r: IRESTResponse;
|
||
begin
|
||
r := RESTClient.Accept(TMVCMediaType.APPLICATION_JSON)
|
||
.doGET('/handledbymiddleware', []);
|
||
CheckEquals('This is a middleware response', r.BodyAsString);
|
||
CheckEquals(HTTP_STATUS.OK, r.ResponseCode);
|
||
end;
|
||
|
||
procedure TServerTest.TestMiddlewareSpeedMiddleware;
|
||
var
|
||
r: IRESTResponse;
|
||
P: TPerson;
|
||
begin
|
||
P := TPerson.Create;
|
||
try
|
||
P.FirstName := StringOfChar('*', 1000);
|
||
P.LastName := StringOfChar('*', 1000);
|
||
P.DOB := EncodeDate(1979, 1, 1);
|
||
P.Married := true;
|
||
r := RESTClient.Accept(TMVCMediaType.APPLICATION_JSON)
|
||
.doPOST('/objects', [], Mapper.ObjectToJSONObject(P));
|
||
finally
|
||
P.Free;
|
||
end;
|
||
|
||
CheckNotEquals('', r.HeaderValue('request_gen_time'));
|
||
end;
|
||
|
||
// procedure TServerTest.TestPATCHWithParamsAndJSONBody;
|
||
// var
|
||
// r: IRESTResponse;
|
||
// json: TJSONObject;
|
||
// begin
|
||
// json := TJSONObject.Create;
|
||
// json.AddPair('client', 'clientdata');
|
||
// r := RESTClient.doPATCH('/echo', ['1', '2', '3'], json);
|
||
// CheckEquals('clientdata', r.BodyAsJsonObject.Get('client').JsonValue.Value);
|
||
// CheckEquals('from server', r.BodyAsJsonObject.Get('echo').JsonValue.Value);
|
||
// end;
|
||
|
||
procedure TServerTest.TestPostAListOfObjects;
|
||
var
|
||
LRes: IRESTResponse;
|
||
LCustomers: TObjectList<TCustomer>;
|
||
begin
|
||
LCustomers := TCustomer.GetList;
|
||
try
|
||
LRes := RESTClient.doPOST('/customers/list', [],
|
||
Mapper.ObjectListToJSONArray<TCustomer>(LCustomers));
|
||
CheckEquals(HTTP_STATUS.OK, LRes.ResponseCode);
|
||
finally
|
||
LCustomers.Free;
|
||
end;
|
||
end;
|
||
|
||
procedure TServerTest.TestPOSTWithObjectJSONBody;
|
||
var
|
||
r: IRESTResponse;
|
||
P: TPerson;
|
||
begin
|
||
P := TPerson.Create;
|
||
try
|
||
P.FirstName := 'Daniele';
|
||
P.LastName := '<27><><EFBFBD><EFBFBD><EFBFBD><EFBFBD>';
|
||
P.DOB := EncodeDate(1979, 1, 1);
|
||
P.Married := true;
|
||
try
|
||
r := RESTClient.Accept(TMVCMediaType.APPLICATION_JSON)
|
||
.doPOST('/objects', [], Mapper.ObjectToJSONObject(P));
|
||
except
|
||
Fail('If this test fail, check http://qc.embarcadero.com/wc/qcmain.aspx?d=119779');
|
||
{ WARNING!!! }
|
||
{
|
||
If this test fail, check
|
||
http://qc.embarcadero.com/wc/qcmain.aspx?d=119779
|
||
}
|
||
raise;
|
||
|
||
end;
|
||
finally
|
||
P.Free;
|
||
end;
|
||
P := Mapper.JSONObjectToObject<TPerson>(r.BodyAsJsonObject);
|
||
try
|
||
CheckEquals('Daniele', P.FirstName);
|
||
CheckEquals('<27><><EFBFBD><EFBFBD><EFBFBD><EFBFBD>', P.LastName);
|
||
CheckEquals(true, P.Married);
|
||
CheckEquals(EncodeDate(1979, 1, 1), P.DOB);
|
||
finally
|
||
P.Free;
|
||
end;
|
||
end;
|
||
|
||
procedure TServerTest.TestPOSTWithParamsAndJSONBody;
|
||
var
|
||
r: IRESTResponse;
|
||
JSON: TJSONObject;
|
||
begin
|
||
JSON := TJSONObject.Create;
|
||
JSON.AddPair('client', 'clientdata');
|
||
r := RESTClient.doPOST('/echo', ['1', '2', '3'], JSON);
|
||
CheckEquals('clientdata', r.BodyAsJsonObject.Get('client').JsonValue.Value);
|
||
CheckEquals('from server', r.BodyAsJsonObject.Get('echo').JsonValue.Value);
|
||
end;
|
||
|
||
procedure TServerTest.TestProducesConsumesWithWrongAcceptHeader;
|
||
var
|
||
res: IRESTResponse;
|
||
begin
|
||
res := RESTClient.Accept('text/plain')
|
||
// action is waiting for a accept: application/json
|
||
.ContentType('application/json').doPOST('/testconsumes', [],
|
||
TJSONString.Create('Hello World'));
|
||
CheckEquals(HTTP_STATUS.NotFound, res.ResponseCode);
|
||
end;
|
||
|
||
procedure TServerTest.TestProducesConsumes01;
|
||
var
|
||
res: IRESTResponse;
|
||
begin
|
||
res := RESTClient.Accept('application/json').ContentType('application/json')
|
||
.ContentEncoding('utf-8').doPOST('/testconsumes', [],
|
||
TJSONString.Create('Hello World'));
|
||
CheckEquals(HTTP_STATUS.OK, res.ResponseCode);
|
||
CheckEquals('Hello World', res.BodyAsString);
|
||
CheckEquals('application/json', res.ContentType);
|
||
CheckEquals('utf-8', res.ContentEncoding);
|
||
end;
|
||
|
||
procedure TServerTest.TestProducesConsumes02;
|
||
var
|
||
res: IRESTResponse;
|
||
begin
|
||
res := RESTClient.Accept('text/plain').ContentType('text/plain')
|
||
.doPOST('/testconsumes', [], 'Hello World');
|
||
CheckEquals('Hello World', res.BodyAsString);
|
||
CheckEquals('text/plain', res.ContentType);
|
||
CheckEquals('UTF-8', res.ContentEncoding);
|
||
|
||
res := RESTClient.Accept('text/plain').ContentType('application/json')
|
||
.doPOST('/testconsumes', [], '{"name": "Daniele"}');
|
||
CheckEquals(HTTP_STATUS.NotFound, res.ResponseCode);
|
||
end;
|
||
|
||
procedure TServerTest.TestProducesConsumes03;
|
||
var
|
||
res: IRESTResponse;
|
||
begin
|
||
res := RESTClient
|
||
.Accept(TMVCMediaType.TEXT_PLAIN)
|
||
.ContentType(TMVCMediaType.TEXT_PLAIN)
|
||
.ContentEncoding('iso8859-1')
|
||
.doPOST('/testconsumes/textiso8859_1', [],
|
||
'<27><><EFBFBD><EFBFBD><EFBFBD><EFBFBD>');
|
||
CheckEquals(HTTP_STATUS.OK, res.ResponseCode);
|
||
CheckEquals('<27><><EFBFBD><EFBFBD><EFBFBD><EFBFBD>', res.BodyAsString);
|
||
CheckEquals(TMVCMediaType.TEXT_PLAIN, res.ContentType);
|
||
CheckEquals('iso8859-1', res.ContentEncoding);
|
||
|
||
end;
|
||
|
||
procedure TServerTest.TestPUTWithParamsAndJSONBody;
|
||
var
|
||
r: IRESTResponse;
|
||
JSON: TJSONObject;
|
||
begin
|
||
JSON := TJSONObject.Create;
|
||
JSON.AddPair('client', 'clientdata');
|
||
r := RESTClient.doPUT('/echo', ['1', '2', '3'], JSON);
|
||
CheckEquals('clientdata', r.BodyAsJsonObject.Get('client').JsonValue.Value);
|
||
CheckEquals('from server', r.BodyAsJsonObject.Get('echo').JsonValue.Value);
|
||
end;
|
||
|
||
procedure TServerTest.TestReqWithParams;
|
||
var
|
||
r: IRESTResponse;
|
||
ss: TStringStream;
|
||
begin
|
||
r := RESTClient.doGET('/unknownurl/bla/bla', []);
|
||
|
||
ss := TStringStream.Create;
|
||
try
|
||
ss.CopyFrom(r.Body, 0);
|
||
CheckEquals(ss.DataString, r.BodyAsString,
|
||
'In case of rotocol error, the body doesn''t contain the same of BodyAsString');
|
||
finally
|
||
ss.Free;
|
||
end;
|
||
|
||
CheckEquals(HTTP_STATUS.NotFound, r.ResponseCode, '/unknownurl/bla/bla');
|
||
|
||
r := RESTClient.doGET('/req/with/params/', []);
|
||
CheckEquals(HTTP_STATUS.NotFound, r.ResponseCode, '/req/with/params/');
|
||
|
||
r := RESTClient.doGET('/req/with/params', []);
|
||
CheckEquals(HTTP_STATUS.NotFound, r.ResponseCode, '/req/with/params');
|
||
|
||
r := RESTClient.doGET('/req/with/params', ['1', '2', '3']);
|
||
CheckEquals(HTTP_STATUS.OK, r.ResponseCode);
|
||
CheckEquals('1', r.BodyAsJsonObject.Get('par1').JsonValue.Value);
|
||
CheckEquals('2', r.BodyAsJsonObject.Get('par2').JsonValue.Value);
|
||
CheckEquals('3', r.BodyAsJsonObject.Get('par3').JsonValue.Value);
|
||
CheckEquals('GET', r.BodyAsJsonObject.Get('method').JsonValue.Value);
|
||
|
||
r := RESTClient.doPOST('/req/with/params', ['1', '2', '3']);
|
||
CheckEquals(HTTP_STATUS.NotFound, r.ResponseCode);
|
||
|
||
r := RESTClient.doPUT('/req/with/params', ['1', '2', '3']);
|
||
CheckEquals(HTTP_STATUS.NotFound, r.ResponseCode);
|
||
|
||
r := RESTClient.doDELETE('/req/with/params', ['1', '2', '3']);
|
||
CheckEquals(HTTP_STATUS.OK, r.ResponseCode);
|
||
CheckNull(r.BodyAsJsonObject);
|
||
end;
|
||
|
||
// procedure TServerTest.TestSerializationType;
|
||
// var
|
||
// LResp: IRESTResponse;
|
||
// LPersonProps, LPersonFlds: TPerson;
|
||
// LObj: TObject;
|
||
// begin
|
||
// LResp := RESTClient.doGET('/people', ['1']);
|
||
// LPersonProps := Mapper.JSONObjectToObject<TPerson>(LResp.BodyAsJsonObject);
|
||
// try
|
||
// LResp := RESTClient.doGET('/people', ['1', 'asfields']);
|
||
// LObj := Mapper.JSONObjectFieldsToObject(LResp.BodyAsJsonObject);
|
||
// try
|
||
// CheckEquals('BusinessObjectsU.TPerson', LObj.QualifiedClassName);
|
||
// LPersonFlds := TPerson(LObj);
|
||
// CheckTrue(LPersonFlds.Equals(LPersonProps),
|
||
// 'Object tranferred using field serialization is different from the object serialized in the default way');
|
||
// finally
|
||
// LObj.Free;
|
||
// end;
|
||
// finally
|
||
// LPersonProps.Free;
|
||
// end;
|
||
// end;
|
||
|
||
procedure TServerTest.TestSession;
|
||
var
|
||
c1: TRESTClient;
|
||
res: IRESTResponse;
|
||
begin
|
||
c1 := TRESTClient.Create(TEST_SERVER_ADDRESS, 9999);
|
||
try
|
||
c1.Accept(TMVCMediaType.APPLICATION_JSON);
|
||
c1.doPOST('/session', ['daniele teti']); // imposto un valore in sessione
|
||
res := c1.doGET('/session', []); // rileggo il valore dalla sessione
|
||
CheckEquals('daniele teti', res.BodyAsString);
|
||
c1.Accept(TMVCMediaType.TEXT_PLAIN);
|
||
res := c1.doGET('/session', []);
|
||
// rileggo il valore dalla sessione
|
||
CheckEquals('daniele teti', res.BodyAsString);
|
||
|
||
// aggiungo altri cookies
|
||
res := c1.doGET('/lotofcookies', []); // rileggo il valore dalla sessione
|
||
CheckEquals(HTTP_STATUS.OK, res.ResponseCode);
|
||
c1.Accept(TMVCMediaType.TEXT_PLAIN);
|
||
res := c1.doGET('/session', []); // rileggo il valore dalla sessione
|
||
CheckEquals('daniele teti', res.BodyAsString);
|
||
finally
|
||
c1.Free;
|
||
end;
|
||
end;
|
||
|
||
procedure TServerTest.TestSessionWithLogin;
|
||
begin
|
||
DoLoginWith('daniele');
|
||
DoLogout;
|
||
end;
|
||
|
||
procedure TServerTest.TestTypedAll;
|
||
var
|
||
res: IRESTResponse;
|
||
lJObj: TJSONObject;
|
||
begin
|
||
// ----------------------'/typed/all/($ParString)/($ParInteger)/($ParInt64)/($ParSingle)/($ParDouble)/($ParExtended)')', []);
|
||
res := RESTClient.doGET
|
||
('/typed/all/mystring/1234/12345678/12.3/1234.5678/1234.5678', []);
|
||
CheckTrue(res.ResponseCode = HTTP_STATUS.OK, 'Cannot route');
|
||
lJObj := res.BodyAsJsonObject;
|
||
CheckEquals('mystring', lJObj.GetValue('ParString').Value, 'ParString');
|
||
CheckEquals(1234, TJSONNumber(lJObj.GetValue('ParInteger')).AsInt,
|
||
'ParInteger');
|
||
CheckEquals(12345678, TJSONNumber(lJObj.GetValue('ParInt64')).AsInt64,
|
||
'ParInt64');
|
||
CheckEquals(12.3, RoundTo(TJSONNumber(lJObj.GetValue('ParSingle')).AsDouble,
|
||
-1), 'ParSingle');
|
||
CheckEquals(1234.5678, RoundTo(TJSONNumber(lJObj.GetValue('ParDouble'))
|
||
.AsDouble, -4), 'ParDouble');
|
||
CheckEquals(1234.5678, RoundTo(TJSONNumber(lJObj.GetValue('ParExtended'))
|
||
.AsDouble, -4), 'ParExtended');
|
||
end;
|
||
|
||
procedure TServerTest.TestTypedBooleans;
|
||
var
|
||
res: IRESTResponse;
|
||
begin
|
||
res := RESTClient.doGET('/typed/booleans/true/false/1/0', []);
|
||
CheckTrue(res.ResponseCode = HTTP_STATUS.OK, 'Cannot route');
|
||
CheckEquals('true.false.true.false', res.BodyAsString.ToLower);
|
||
end;
|
||
|
||
procedure TServerTest.TestTypedDouble1;
|
||
var
|
||
res: IRESTResponse;
|
||
begin
|
||
res := RESTClient.doGET('/typed/double1/1234.5678', []);
|
||
CheckTrue(res.ResponseCode = HTTP_STATUS.OK, 'Cannot route');
|
||
CheckEquals('1234.5678 modified from server', res.BodyAsString);
|
||
|
||
end;
|
||
|
||
procedure TServerTest.TestTypedExtended1;
|
||
var
|
||
res: IRESTResponse;
|
||
begin
|
||
res := RESTClient.doGET('/typed/extended1/1234.5678', []);
|
||
CheckTrue(res.ResponseCode = HTTP_STATUS.OK, 'Cannot route');
|
||
CheckEquals('1234.5678 modified from server', res.BodyAsString);
|
||
|
||
end;
|
||
|
||
procedure TServerTest.TestTypedInt641;
|
||
var
|
||
res: IRESTResponse;
|
||
begin
|
||
res := RESTClient.doGET('/typed/int641/12345678', []);
|
||
CheckTrue(res.ResponseCode = HTTP_STATUS.OK, 'Cannot route');
|
||
CheckEquals('12345678 modified from server', res.BodyAsString);
|
||
end;
|
||
|
||
procedure TServerTest.TestTypedInteger1;
|
||
var
|
||
res: IRESTResponse;
|
||
begin
|
||
res := RESTClient.doGET('/typed/integer1/1234', []);
|
||
CheckTrue(res.ResponseCode = HTTP_STATUS.OK, 'Cannot route');
|
||
CheckEquals('1234 modified from server', res.BodyAsString);
|
||
end;
|
||
|
||
procedure TServerTest.TestTypedSingle1;
|
||
var
|
||
res: IRESTResponse;
|
||
begin
|
||
res := RESTClient.doGET('/typed/single1/1234.5', []);
|
||
CheckTrue(res.ResponseCode = HTTP_STATUS.OK, 'Cannot route');
|
||
CheckEquals('1234.5 modified from server', res.BodyAsString);
|
||
|
||
end;
|
||
|
||
procedure TServerTest.TestTypedString1;
|
||
var
|
||
res: IRESTResponse;
|
||
begin
|
||
res := RESTClient.doGET('/typed/string1/daniele', []);
|
||
CheckTrue(res.ResponseCode = HTTP_STATUS.OK, 'Cannot route');
|
||
CheckEquals('daniele modified from server', res.BodyAsString);
|
||
end;
|
||
|
||
procedure TServerTest.TestTypedDateTimeTypes;
|
||
var
|
||
res: IRESTResponse;
|
||
begin
|
||
// TDate, wrong and correct
|
||
res := RESTClient.doGET('/typed/tdate1/20161012', []);
|
||
CheckEquals(HTTP_STATUS.InternalServerError, res.ResponseCode, 'wrong TDate');
|
||
|
||
res := RESTClient.doGET('/typed/tdate1/2016-10-12', []);
|
||
CheckEquals(HTTP_STATUS.OK, res.ResponseCode);
|
||
CheckEquals('2016-10-12 modified from server', res.BodyAsString);
|
||
|
||
// TDateTime, wrong and correct
|
||
res := RESTClient.doGET('/typed/tdatetime1/20161012121212', []);
|
||
CheckEquals(HTTP_STATUS.InternalServerError, res.ResponseCode,
|
||
'wrong TDateTime');
|
||
|
||
res := RESTClient.doGET('/typed/tdatetime1/2016-10-12 12:12:12', []);
|
||
CheckEquals(HTTP_STATUS.OK, res.ResponseCode);
|
||
CheckEquals('2016-10-12 12:12:12 modified from server', res.BodyAsString);
|
||
|
||
// TTime, wrong and correct
|
||
res := RESTClient.doGET('/typed/ttime1/121212', []);
|
||
CheckEquals(HTTP_STATUS.InternalServerError, res.ResponseCode, 'wrong TTime');
|
||
|
||
res := RESTClient.doGET('/typed/ttime1/12:12:12', []);
|
||
CheckEquals(HTTP_STATUS.OK, res.ResponseCode);
|
||
CheckEquals('12:12:12 modified from server', res.BodyAsString);
|
||
|
||
end;
|
||
|
||
procedure TBaseServerTest.DoLoginWith(UserName: string);
|
||
var
|
||
res: IRESTResponse;
|
||
begin
|
||
res := RESTClient.doGET('/login', [UserName]);
|
||
CheckTrue(res.ResponseCode = HTTP_STATUS.OK, 'Login Failed');
|
||
end;
|
||
|
||
initialization
|
||
|
||
RegisterTest(TServerTest.Suite);
|
||
|
||
end.
|