2021-02-04 14:11:33 +01:00
|
|
|
unit ActiveRecordControllerTestU;
|
|
|
|
|
|
|
|
// ***************************************************************************
|
|
|
|
//
|
|
|
|
// Delphi MVC Framework
|
|
|
|
//
|
2022-01-04 15:44:47 +01:00
|
|
|
// Copyright (c) 2010-2022 Daniele Teti and the DMVCFramework Team
|
2021-02-04 14:11:33 +01:00
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
//
|
|
|
|
// *************************************************************************** }
|
|
|
|
|
|
|
|
interface
|
|
|
|
|
|
|
|
uses
|
|
|
|
DUnitX.TestFramework,
|
|
|
|
FireDAC.Comp.Client, FireDAC.ConsoleUI.Wait, FireDAC.VCLUI.Wait,
|
2022-08-01 19:11:42 +02:00
|
|
|
PGUtilsU, LiveServerTestU, MVCFramework.Server, MVCFramework.RESTClient.Intf;
|
|
|
|
|
2021-02-04 14:11:33 +01:00
|
|
|
|
|
|
|
type
|
|
|
|
[TestFixture]
|
|
|
|
TTestActiveRecordController = class(TObject)
|
2022-08-01 19:11:42 +02:00
|
|
|
private
|
|
|
|
fListener: IMVCListener;
|
|
|
|
fClient: IMVCRESTClient;
|
|
|
|
procedure LoadData;
|
2021-02-04 14:11:33 +01:00
|
|
|
public
|
2022-08-01 19:11:42 +02:00
|
|
|
class procedure CreatePrivateFirebirdSQLConnDef(const ConDefName: String; AIsPooled: boolean);
|
2021-02-04 14:11:33 +01:00
|
|
|
[SetupFixture]
|
2022-08-01 19:11:42 +02:00
|
|
|
procedure Setup;
|
|
|
|
[TeardownFixture]
|
|
|
|
procedure TearDown;
|
|
|
|
[Test]
|
|
|
|
procedure TestGetAll;
|
|
|
|
[Test]
|
|
|
|
procedure TestCRUD;
|
|
|
|
[Test]
|
|
|
|
procedure TestDelete;
|
2021-02-04 14:11:33 +01:00
|
|
|
end;
|
|
|
|
|
|
|
|
implementation
|
|
|
|
|
|
|
|
uses
|
|
|
|
System.Classes, System.IOUtils, BOs, MVCFramework.ActiveRecord,
|
|
|
|
System.SysUtils, System.Threading, System.Generics.Collections, Data.DB,
|
2022-08-01 19:11:42 +02:00
|
|
|
FireDAC.Stan.Intf, ShellAPI, Winapi.Windows, FDConnectionConfigU,
|
|
|
|
MVCFramework.Serializer.JsonDataObjects,
|
|
|
|
MVCFramework.Server.Impl, ActiveRecordControllerWebModuleU,
|
|
|
|
MVCFramework.RESTClient, JsonDataObjects, System.StrUtils,
|
|
|
|
MVCFramework.Commons;
|
2021-02-04 14:11:33 +01:00
|
|
|
|
|
|
|
|
|
|
|
{ TTestActiveRecordController }
|
2022-08-01 19:11:42 +02:00
|
|
|
class procedure TTestActiveRecordController.CreatePrivateFirebirdSQLConnDef(const ConDefName: String; AIsPooled: boolean);
|
|
|
|
var
|
|
|
|
LParams: TStringList;
|
|
|
|
lDriver: IFDStanDefinition;
|
|
|
|
GDBFileName: string;
|
|
|
|
GDBTemplateFileName: string;
|
|
|
|
begin
|
|
|
|
if not Assigned(FDManager.DriverDefs.FindDefinition('FBEMBEDDED')) then
|
|
|
|
begin
|
|
|
|
lDriver := FDManager.DriverDefs.Add;
|
|
|
|
lDriver.Name := 'FBEMBEDDED';
|
|
|
|
lDriver.AsString['BaseDriverID'] := 'FB';
|
|
|
|
lDriver.AsString['DriverID'] := 'FBEMBEDDED';
|
|
|
|
lDriver.AsString['VendorLib'] := TPath.Combine(TPath.GetDirectoryName(ParamStr(0)), 'firebird\fbclient.dll');
|
|
|
|
lDriver.Apply;
|
|
|
|
end;
|
|
|
|
|
|
|
|
LParams := TStringList.Create;
|
|
|
|
try
|
|
|
|
GDBFileName := TPath.Combine(TPath.GetDirectoryName(ParamStr(0)), 'firebirdtest2.fdb');
|
|
|
|
GDBTemplateFileName := TPath.Combine(TPath.GetDirectoryName(ParamStr(0)), 'firebirdtest_template.fdb');
|
|
|
|
TFile.Copy(GDBTemplateFileName, GDBFileName, True);
|
|
|
|
LParams.Add('Database=' + GDBFileName);
|
|
|
|
LParams.Add('user_name=sysdba');
|
|
|
|
LParams.Add('password=masterkey');
|
|
|
|
if AIsPooled then
|
|
|
|
begin
|
|
|
|
LParams.Add('Pooled=True');
|
|
|
|
LParams.Add('POOL_MaximumItems=100');
|
|
|
|
end
|
|
|
|
else
|
|
|
|
begin
|
|
|
|
LParams.Add('Pooled=False');
|
|
|
|
end;
|
|
|
|
FDManager.AddConnectionDef(ConDefName, 'FBEMBEDDED', LParams);
|
|
|
|
finally
|
|
|
|
LParams.Free;
|
|
|
|
end;
|
|
|
|
end;
|
2021-02-04 14:11:33 +01:00
|
|
|
|
2022-08-01 19:11:42 +02:00
|
|
|
procedure TTestActiveRecordController.LoadData;
|
|
|
|
var
|
|
|
|
lProc: TProc;
|
|
|
|
const
|
|
|
|
Cities: array [0 .. 4] of string = ('Rome', 'New York', 'London', 'Melbourne', 'Berlin');
|
|
|
|
CompanySuffix: array [0 .. 5] of string = ('Corp.', 'Inc.', 'Ltd.', 'Srl', 'SPA', 'doo');
|
|
|
|
Stuff: array [0 .. 4] of string = ('Burger', 'GAS', 'Motors', 'House', 'Boats');
|
2021-02-04 14:11:33 +01:00
|
|
|
begin
|
2022-08-01 19:11:42 +02:00
|
|
|
TMVCActiveRecord.DeleteRQL(TCustomer, 'in(City,["Rome","New York","London","Melbourne","Berlin"])');
|
|
|
|
lProc := procedure
|
|
|
|
var
|
|
|
|
lCustomer: TCustomer;
|
|
|
|
I: Integer;
|
|
|
|
begin
|
|
|
|
for I := 1 to 30 do
|
|
|
|
begin
|
|
|
|
lCustomer := TCustomer.Create;
|
|
|
|
try
|
|
|
|
lCustomer.Code := Format('%5.5d', [TThread.CurrentThread.ThreadID, I]);
|
|
|
|
lCustomer.City := Cities[I mod Length(Cities)];
|
|
|
|
lCustomer.CompanyName := Format('%s %s %s', [lCustomer.City, Stuff[Random(high(Stuff) + 1)],
|
|
|
|
CompanySuffix[Random(high(CompanySuffix) + 1)]]);
|
|
|
|
lCustomer.Note := Stuff[I mod Length(Stuff)];
|
|
|
|
lCustomer.Rating := 1;
|
|
|
|
lCustomer.CreationTime := EncodeTime(I mod 23, I, 60 - 1, 0);
|
|
|
|
lCustomer.CreationDate := EncodeDate(2020 - I, (I mod 12) + 1, (I mod 27) + 1);
|
|
|
|
lCustomer.Insert;
|
|
|
|
finally
|
|
|
|
lCustomer.Free;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
lProc();
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TTestActiveRecordController.Setup;
|
|
|
|
begin
|
|
|
|
CreatePrivateFirebirdSQLConnDef(AR_CONTROLLER_CON_DEF_NAME, True);
|
|
|
|
|
|
|
|
var lConn := TFDConnection.Create(nil);
|
|
|
|
try
|
|
|
|
lConn.ConnectionDefName := AR_CONTROLLER_CON_DEF_NAME;
|
|
|
|
lConn.Open;
|
|
|
|
for var lSQL in SQLs_FIREBIRD do
|
|
|
|
begin
|
|
|
|
lConn.ExecSQL(lSQL);
|
|
|
|
end;
|
|
|
|
ActiveRecordConnectionsRegistry.AddDefaultConnection(lConn, False);
|
|
|
|
LoadData;
|
|
|
|
ActiveRecordConnectionsRegistry.RemoveDefaultConnection();
|
|
|
|
finally
|
|
|
|
lConn.Free;
|
|
|
|
end;
|
|
|
|
|
|
|
|
fListener := TMVCListener.Create(
|
|
|
|
TMVCListenerProperties
|
|
|
|
.New
|
|
|
|
.SetName('Listener1')
|
|
|
|
.SetPort(5000)
|
|
|
|
.SetMaxConnections(512)
|
|
|
|
.SetWebModuleClass(TActiveRecordControllerWebModule));
|
|
|
|
fListener.Start;
|
|
|
|
|
|
|
|
fClient := TMVCRESTClient.New.BaseURL('http://localhost', 5000);
|
|
|
|
fClient.ReadTimeout(60 * 1000 * 30);
|
2021-02-04 14:11:33 +01:00
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TTestActiveRecordController.Teardown;
|
|
|
|
begin
|
2022-08-01 19:11:42 +02:00
|
|
|
fListener.Stop;
|
|
|
|
FDManager.CloseConnectionDef(AR_CONTROLLER_CON_DEF_NAME);
|
|
|
|
FDManager.DeleteConnectionDef(AR_CONTROLLER_CON_DEF_NAME);
|
|
|
|
end;
|
2021-02-04 14:11:33 +01:00
|
|
|
|
2022-08-01 19:11:42 +02:00
|
|
|
procedure TTestActiveRecordController.TestDelete;
|
|
|
|
var
|
|
|
|
lResp: IMVCRESTResponse;
|
|
|
|
lLocation: String;
|
|
|
|
begin
|
|
|
|
var lCust := TCustomer.Create;
|
|
|
|
try
|
|
|
|
lCust.Code := 'MYCODE';
|
|
|
|
lCust.CompanyName := 'bit Time Professionals';
|
|
|
|
lCust.City := 'Rome';
|
|
|
|
lResp := fClient.Post('/api/entities/customers', lCust, False);
|
|
|
|
lLocation := lResp.HeaderValue('X-REF');
|
|
|
|
finally
|
|
|
|
lCust.Free;
|
|
|
|
end;
|
|
|
|
|
|
|
|
lResp := fClient.Delete(lLocation);
|
|
|
|
Assert.AreEqual(HTTP_STATUS.OK, lResp.StatusCode);
|
|
|
|
lResp := fClient.Get(lLocation);
|
|
|
|
Assert.AreEqual(HTTP_STATUS.NotFound, lResp.StatusCode);
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TTestActiveRecordController.TestGetAll;
|
|
|
|
var
|
|
|
|
lResp: IMVCRESTResponse;
|
|
|
|
begin
|
|
|
|
lResp := fClient.Get('/api/entities/customers');
|
|
|
|
var lJSON := lResp.ToJSONObject;
|
|
|
|
try
|
|
|
|
Assert.IsTrue(lJSON.Contains('data') and (lJSON.Types['data'] = TJsonDataType.jdtArray), 'incorrect "data" property in JSON');
|
|
|
|
var lCustomers := TJSONUtils.JSONArrayToListOf<TCustomer>(lJSON.A['data']);
|
|
|
|
try
|
|
|
|
Assert.AreEqual(20, lCustomers.Count);
|
|
|
|
finally
|
|
|
|
lCustomers.Free;
|
|
|
|
end;
|
|
|
|
finally
|
|
|
|
lJSON.Free;
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TTestActiveRecordController.TestCRUD;
|
|
|
|
var
|
|
|
|
lResp: IMVCRESTResponse;
|
|
|
|
lLocation: String;
|
|
|
|
lJSON: TJsonObject;
|
|
|
|
lPieces: TArray<String>;
|
|
|
|
lIDFromURL: Integer;
|
|
|
|
lCust: TCustomer;
|
|
|
|
begin
|
|
|
|
lCust := TCustomer.Create;
|
|
|
|
try
|
|
|
|
lCust.Code := 'MYCODE';
|
|
|
|
lCust.CompanyName := 'The Company';
|
|
|
|
lCust.City := 'Rome';
|
|
|
|
lCust.CreationTime := Time;
|
|
|
|
lCust.CreationDate := Date;
|
|
|
|
lResp := fClient.Post('/api/entities/customers', lCust, False);
|
|
|
|
lLocation := lResp.HeaderValue('X-REF');
|
|
|
|
finally
|
|
|
|
lCust.Free;
|
|
|
|
end;
|
|
|
|
|
|
|
|
lResp := fClient.Get(lLocation);
|
|
|
|
lJSON := lResp.ToJSONObject;
|
|
|
|
try
|
|
|
|
Assert.IsTrue(lJSON.Contains('data') and (lJSON.Types['data'] = TJsonDataType.jdtObject), 'incorrect "data" property in JSON');
|
|
|
|
lCust := TJSONUtils.JSONObjectToObject<TCustomer>(lJSON.O['data']);
|
|
|
|
try
|
|
|
|
lPieces := lLocation.Split(['/']);
|
|
|
|
lIDFromUrl := lPieces[High(lPieces)].ToInteger;
|
|
|
|
Assert.AreEqual(lIDFromUrl, lCust.ID.Value);
|
|
|
|
Assert.AreEqual('MYCODE', lCust.Code.Value);
|
|
|
|
Assert.AreEqual('The Company', lCust.CompanyName.Value);
|
|
|
|
Assert.AreEqual('Rome', lCust.City);
|
|
|
|
Assert.IsFalse(lCust.Rating.HasValue);
|
|
|
|
Assert.IsTrue(lCust.CreationTime.HasValue);
|
|
|
|
Assert.IsTrue(lCust.CreationDate.HasValue);
|
|
|
|
Assert.IsEmpty(lCust.Note);
|
|
|
|
|
|
|
|
//update
|
|
|
|
lCust.Code := nil; //.SetNull;
|
|
|
|
lCust.CompanyName.Value := 'bit Time Professionals';
|
|
|
|
fClient.Put(lLocation, lCust, False);
|
|
|
|
finally
|
|
|
|
lCust.Free;
|
|
|
|
end;
|
|
|
|
finally
|
|
|
|
lJSON.Free;
|
|
|
|
end;
|
|
|
|
|
|
|
|
lResp := fClient.Get(lLocation);
|
|
|
|
lJSON := lResp.ToJSONObject;
|
|
|
|
try
|
|
|
|
lCust := TJSONUtils.JSONObjectToObject<TCustomer>(lJSON.O['data']);
|
|
|
|
try
|
|
|
|
lPieces := lLocation.Split(['/']);
|
|
|
|
lIDFromUrl := lPieces[High(lPieces)].ToInteger;
|
|
|
|
Assert.AreEqual(lIDFromUrl, lCust.ID.Value);
|
|
|
|
Assert.IsFalse(lCust.Code.HasValue);
|
|
|
|
Assert.AreEqual('bit Time Professionals', lCust.CompanyName.Value);
|
|
|
|
Assert.AreEqual('Rome', lCust.City);
|
|
|
|
Assert.IsFalse(lCust.Rating.HasValue);
|
|
|
|
Assert.IsTrue(lCust.CreationTime.HasValue);
|
|
|
|
Assert.IsTrue(lCust.CreationDate.HasValue);
|
|
|
|
Assert.IsEmpty(lCust.Note);
|
|
|
|
finally
|
|
|
|
lCust.Free;
|
|
|
|
end;
|
|
|
|
finally
|
|
|
|
lJSON.Free;
|
|
|
|
end;
|
|
|
|
lResp := fClient.Delete(lLocation);
|
|
|
|
Assert.AreEqual(HTTP_STATUS.OK, lResp.StatusCode);
|
2021-02-04 14:11:33 +01:00
|
|
|
end;
|
|
|
|
|
|
|
|
initialization
|
|
|
|
|
|
|
|
TDUnitX.RegisterTestFixture(TTestActiveRecordController);
|
|
|
|
|
|
|
|
finalization
|
|
|
|
|
|
|
|
end.
|