mirror of
https://gitlab.com/freepascal.org/fpc/pas2js.git
synced 2025-04-05 04:17:54 +02:00
* SQLDBRestBridge components and first demo
This commit is contained in:
parent
5a3377ea94
commit
3d43138c51
12
demo/restbridge/simple/README.txt
Normal file
12
demo/restbridge/simple/README.txt
Normal file
@ -0,0 +1,12 @@
|
||||
|
||||
In order to run this demo, you must run one of the SQLDB REST bridge demos
|
||||
as a server:
|
||||
|
||||
fpc/packages/fcl-web/example/restbridge
|
||||
lazarus/components/fpweb/demo/restbridge/
|
||||
lazarus/components/fpweb/demo/restmodule/
|
||||
|
||||
You need to know how it is configured (The port, base URL)
|
||||
|
||||
The servers are by default set up so the client requires authentication,
|
||||
so unless that was disabled, you need to know what user the demo is using to authenticate requests !
|
81
demo/restbridge/simple/restbridge.html
Normal file
81
demo/restbridge/simple/restbridge.html
Normal file
@ -0,0 +1,81 @@
|
||||
<html>
|
||||
<head>
|
||||
<title>REST Bridge</title>
|
||||
<meta charset="utf-8"/>
|
||||
<link href="https://maxcdn.bootstrapcdn.com/bootstrap/4.0.0/css/bootstrap.min.css" rel="stylesheet">
|
||||
<script src="https://code.jquery.com/jquery-3.2.1.slim.min.js" ></script>
|
||||
<script src="https://cdnjs.cloudflare.com/ajax/libs/popper.js/1.12.9/umd/popper.min.js"> </script>
|
||||
<script src="https://maxcdn.bootstrapcdn.com/bootstrap/4.0.0/js/bootstrap.min.js"></script>
|
||||
<script type="application/javascript" src="restbridgeclient.js"></script>
|
||||
</head>
|
||||
<body>
|
||||
<div class="container-fluid">
|
||||
<div class="row">
|
||||
<div class="col-md-12">
|
||||
<div id="urlform" role="form" class="form-inline">
|
||||
<div class="input-group col-md-6">
|
||||
<label class="input-group-text" for="edtUrl">
|
||||
Base URL
|
||||
</label>
|
||||
<input type="text" class="form-control" id="edtURL" value="http://localhost:8080/"/>
|
||||
</div>
|
||||
<div class="input-group col-md-2">
|
||||
<label class="input-group-text" for="edtUserName">
|
||||
UserName
|
||||
</label>
|
||||
<input type="text" class="form-control" id="edtUserName" value="Michael"/>
|
||||
</div>
|
||||
<div class="input-group col-md-2">
|
||||
<label class="input-group-text" for="exampleInputPassword1">
|
||||
Password
|
||||
</label>
|
||||
<input type="password" class="form-control" id="edtPassword" value="secret" />
|
||||
</div>
|
||||
<button id="btnResources" class="btn col-md-2">
|
||||
Get Resource List
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="row">
|
||||
<div class="col-md-12">
|
||||
<form id="show" role="form" class="form-inline">
|
||||
<div class="input-group col-md-3">
|
||||
<label class="input-group-text " for="selResource">
|
||||
Resource
|
||||
</label>
|
||||
<select class="custom-select col-md-auto" id="selResource" >
|
||||
<option selected>Choose...</option>
|
||||
<option value="1">One</option>
|
||||
<option value="2">Two</option>
|
||||
<option value="3">Three</option>
|
||||
</select>
|
||||
</div>
|
||||
<div class="input-group col-md-7">
|
||||
<label class="input-group-text " for="edtOptions">
|
||||
Additional options:
|
||||
</label>
|
||||
<input type="text" class="form-control" id="edtOptions" />
|
||||
</div>
|
||||
<button id="btnFetch" type="submit" class="btn btn-primary col-md-2">
|
||||
Get data
|
||||
</button>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="row">
|
||||
<div class="col-md-12">
|
||||
<table id="resulttable" class="table table-sm table-striped table-hover">
|
||||
<thead id="datahead"></thead>
|
||||
<tbody id="databody"></tbody>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
<script type="application/javascript">
|
||||
rtl.showUncaughtExceptions=true;
|
||||
rtl.run();
|
||||
</script>
|
||||
</body>
|
||||
</html>
|
||||
|
85
demo/restbridge/simple/restbridgeclient.lpi
Normal file
85
demo/restbridge/simple/restbridgeclient.lpi
Normal file
@ -0,0 +1,85 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<CONFIG>
|
||||
<ProjectOptions>
|
||||
<Version Value="11"/>
|
||||
<General>
|
||||
<Flags>
|
||||
<MainUnitHasCreateFormStatements Value="False"/>
|
||||
<MainUnitHasTitleStatement Value="False"/>
|
||||
<MainUnitHasScaledStatement Value="False"/>
|
||||
<Runnable Value="False"/>
|
||||
</Flags>
|
||||
<SessionStorage Value="InProjectDir"/>
|
||||
<MainUnit Value="0"/>
|
||||
<Title Value="restbridgeclient"/>
|
||||
<UseAppBundle Value="False"/>
|
||||
<ResourceType Value="res"/>
|
||||
</General>
|
||||
<CustomData Count="1">
|
||||
<Item0 Name="PasJSWebBrowserProject" Value="1"/>
|
||||
</CustomData>
|
||||
<BuildModes Count="1">
|
||||
<Item1 Name="Default" Default="True"/>
|
||||
</BuildModes>
|
||||
<PublishOptions>
|
||||
<Version Value="2"/>
|
||||
<UseFileFilters Value="True"/>
|
||||
</PublishOptions>
|
||||
<RunParams>
|
||||
<FormatVersion Value="2"/>
|
||||
<Modes Count="0"/>
|
||||
</RunParams>
|
||||
<Units Count="2">
|
||||
<Unit0>
|
||||
<Filename Value="restbridgeclient.lpr"/>
|
||||
<IsPartOfProject Value="True"/>
|
||||
</Unit0>
|
||||
<Unit1>
|
||||
<Filename Value="restbridge.html"/>
|
||||
<IsPartOfProject Value="True"/>
|
||||
</Unit1>
|
||||
</Units>
|
||||
</ProjectOptions>
|
||||
<CompilerOptions>
|
||||
<Version Value="11"/>
|
||||
<Target>
|
||||
<Filename Value="restbridgeclient.js" ApplyConventions="False"/>
|
||||
</Target>
|
||||
<SearchPaths>
|
||||
<IncludeFiles Value="$(ProjOutDir)"/>
|
||||
<UnitOutputDirectory Value="js"/>
|
||||
</SearchPaths>
|
||||
<Parsing>
|
||||
<SyntaxOptions>
|
||||
<AllowLabel Value="False"/>
|
||||
<UseAnsiStrings Value="False"/>
|
||||
</SyntaxOptions>
|
||||
</Parsing>
|
||||
<CodeGeneration>
|
||||
<TargetOS Value="browser"/>
|
||||
</CodeGeneration>
|
||||
<Linking>
|
||||
<Debugging>
|
||||
<GenerateDebugInfo Value="False"/>
|
||||
<UseLineInfoUnit Value="False"/>
|
||||
</Debugging>
|
||||
</Linking>
|
||||
<Other>
|
||||
<CustomOptions Value="-Jeutf-8 -Jirtl.js -Jc"/>
|
||||
<CompilerPath Value="$(pas2js)"/>
|
||||
</Other>
|
||||
</CompilerOptions>
|
||||
<Debugging>
|
||||
<Exceptions Count="3">
|
||||
<Item1>
|
||||
<Name Value="EAbort"/>
|
||||
</Item1>
|
||||
<Item2>
|
||||
<Name Value="ECodetoolError"/>
|
||||
</Item2>
|
||||
<Item3>
|
||||
<Name Value="EFOpenError"/>
|
||||
</Item3>
|
||||
</Exceptions>
|
||||
</Debugging>
|
||||
</CONFIG>
|
180
demo/restbridge/simple/restbridgeclient.lpr
Normal file
180
demo/restbridge/simple/restbridgeclient.lpr
Normal file
@ -0,0 +1,180 @@
|
||||
program restbridgeclient;
|
||||
|
||||
{$mode objfpc}
|
||||
|
||||
uses
|
||||
JS, Classes, SysUtils, Web, db, jsondataset, sqldbrestdataset;
|
||||
|
||||
Type
|
||||
|
||||
{ TMainForm }
|
||||
|
||||
TMainForm = class(TComponent)
|
||||
Private
|
||||
FData: TSQLDBRestDataset;
|
||||
FConn: TSQLDBRestConnection;
|
||||
FBtnResources : TJSHTMLButtonElement;
|
||||
FBtnData : TJSHTMLButtonElement;
|
||||
FEdtURL : TJSHTMLInputElement;
|
||||
FEdtUserName : TJSHTMLInputElement;
|
||||
FEdtPassword : TJSHTMLInputElement;
|
||||
FSelResource : TJSHTMLSelectElement;
|
||||
FDataHead : TJSHTMLElement;
|
||||
FDataBody : TJSHTMLElement;
|
||||
function ConfigureConnection: Boolean;
|
||||
function ConfigureDataset: Boolean;
|
||||
procedure CreateDataHead(Dataset: TDataset);
|
||||
function CreateDataRow(aRowNo: Integer; Dataset: TDataset): String;
|
||||
procedure CreateDataTable(Dataset: TDataset);
|
||||
procedure DoGetResources(Sender: TObject);
|
||||
procedure DoOpen(DataSet: TDataSet);
|
||||
function GetData(aEvent: TJSMouseEvent): boolean;
|
||||
function GetElement(const aID: String): TJSHTMLElement;
|
||||
function GetResources(aEvent: TJSMouseEvent): boolean;
|
||||
Public
|
||||
Constructor Create(aOwner : TComponent); override;
|
||||
Procedure BindElements;
|
||||
end;
|
||||
|
||||
{ TMainForm }
|
||||
|
||||
Function TMainForm.ConfigureDataset : Boolean;
|
||||
|
||||
begin
|
||||
FData.ResourceName:=FSelResource.value;
|
||||
Result:=(FData.ResourceName<>'');
|
||||
if not Result then
|
||||
Window.Alert('Dataset not correctly configured');
|
||||
end;
|
||||
|
||||
Function TMainForm.ConfigureConnection : Boolean;
|
||||
|
||||
begin
|
||||
FConn.BaseURL:=FedtURL.value;
|
||||
FConn.Password:=FEdtPassword.value;
|
||||
FConn.UserName:=FEdtUserName.value;
|
||||
Result:=FConn.BaseURL<>'';
|
||||
if not Result then
|
||||
Window.Alert('Connection not correctly configured');
|
||||
end;
|
||||
|
||||
function TMainForm.GetResources(aEvent: TJSMouseEvent): boolean;
|
||||
begin
|
||||
if ConfigureConnection then
|
||||
FConn.GetResources
|
||||
end;
|
||||
|
||||
procedure TMainForm.DoGetResources(Sender: TObject);
|
||||
|
||||
Var
|
||||
S : String;
|
||||
I : Integer;
|
||||
|
||||
begin
|
||||
S:='<option selected>Choose...</option>';
|
||||
For I:=0 to FConn.ResourceList.Count-1 do
|
||||
S:=S+sLineBreak+'<option>'+FConn.ResourceList[i]+'</option>';
|
||||
FSelResource.innerHTML:=S;
|
||||
end;
|
||||
|
||||
procedure TMainForm.CreateDataHead(Dataset : TDataset);
|
||||
|
||||
Var
|
||||
sHTML : String;
|
||||
I : integer;
|
||||
|
||||
begin
|
||||
sHTML:='<tr>';
|
||||
sHTML:=sHTML+'<th>#</th>';
|
||||
For I:=0 to Dataset.FieldDefs.Count-1 do
|
||||
sHTML:=sHTML+'<th>'+Dataset.FieldDefs[i].Name+'</th>'+sLineBreak;
|
||||
sHTML:=sHTML+'</tr>';
|
||||
FDataHead.innerHTML:=sHTML;
|
||||
end;
|
||||
|
||||
Function TMainForm.CreateDataRow(aRowNo : Integer; Dataset : TDataset) : String;
|
||||
|
||||
Var
|
||||
I : integer;
|
||||
sHTML : String;
|
||||
|
||||
begin
|
||||
sHtml:=Format('<tr id="row-%d"><td>%d</td>',[DataSet.RecNo,aRowNo]);
|
||||
For I:=0 to Dataset.Fields.Count-1 do
|
||||
sHTML:=sHTML+'<td>'+Dataset.Fields[i].AsString+'</td>'+sLineBreak;
|
||||
Result:=sHTML;
|
||||
end;
|
||||
|
||||
procedure TMainForm.CreateDataTable(Dataset : TDataset);
|
||||
|
||||
Var
|
||||
sHTML : String;
|
||||
I : integer;
|
||||
|
||||
begin
|
||||
sHTML:='';
|
||||
I:=0;
|
||||
while not Dataset.EOF do
|
||||
begin
|
||||
inc(i);
|
||||
sHtml:=SHTML+CreateDataRow(i,Dataset);
|
||||
Dataset.Next;
|
||||
end;
|
||||
FDataBody.innerHTML:=sHTML;
|
||||
end;
|
||||
|
||||
|
||||
procedure TMainForm.DoOpen(DataSet: TDataSet);
|
||||
begin
|
||||
CreateDataHead(Dataset);
|
||||
CreateDataTable(Dataset);
|
||||
end;
|
||||
|
||||
function TMainForm.GetData(aEvent: TJSMouseEvent): boolean;
|
||||
begin
|
||||
If not ConfigureConnection then
|
||||
exit;
|
||||
if not ConfigureDataset then
|
||||
exit;
|
||||
FData.Load([],Nil);
|
||||
end;
|
||||
|
||||
constructor TMainForm.Create(aOwner: TComponent);
|
||||
begin
|
||||
inherited Create(aOwner);
|
||||
// Ideally, this is done in an IDE
|
||||
FConn:=TSQLDBRestConnection.Create(Self);
|
||||
FCOnn.OnGetResources:=@DoGetResources;
|
||||
FData:=TSQLDBRestDataset.Create(Self);
|
||||
FData.AfterOpen:=@DoOpen;
|
||||
FData.Connection:=FConn;
|
||||
// This must always be done in code
|
||||
BindElements;
|
||||
end;
|
||||
|
||||
Function TMainForm.GetElement(Const aID : String): TJSHTMLElement;
|
||||
|
||||
begin
|
||||
Result:=TJSHTMLElement(document.getElementById(aID));
|
||||
if (Result=Nil) then
|
||||
Console.Log('Could not find element '+aID);
|
||||
end;
|
||||
|
||||
procedure TMainForm.BindElements;
|
||||
begin
|
||||
FBtnResources:=TJSHTMLButtonElement(getElement('btnResources'));
|
||||
FBtnResources.OnClick:=@GetResources;
|
||||
FBtnData:=TJSHTMLButtonElement(getElement('btnFetch'));
|
||||
FBtnData.OnClick:=@GetData;
|
||||
FSelResource:=TJSHTMLSelectElement(GetElement('selResource'));
|
||||
FEdtURL:=TJSHTMLInputElement(getElement('edtURL'));
|
||||
FEdtUserName:=TJSHTMLInputElement(getElement('edtUserName'));
|
||||
FEdtPassword:=TJSHTMLInputElement(getElement('edtPassword'));
|
||||
FDataHead:=getElement('datahead');
|
||||
FDataBody:=getElement('databody');
|
||||
end;
|
||||
|
||||
|
||||
begin
|
||||
TMainForm.Create(Nil);
|
||||
end.
|
@ -1,3 +1,18 @@
|
||||
{
|
||||
This file is part of the Free Pascal run time library.
|
||||
Copyright (c) 2019 by Michael Van Canneyt, member of the
|
||||
Free Pascal development team
|
||||
|
||||
Simple EXTJS JSON dataset component.
|
||||
|
||||
See the file COPYING.FPC, included in this distribution,
|
||||
for details about the copyright.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||
|
||||
**********************************************************************}
|
||||
unit ExtJSDataset;
|
||||
|
||||
{$mode objfpc}
|
||||
|
@ -1,3 +1,18 @@
|
||||
{
|
||||
This file is part of the Free Pascal run time library.
|
||||
Copyright (c) 2019 by Michael Van Canneyt, member of the
|
||||
Free Pascal development team
|
||||
|
||||
Simple JSON dataset component.
|
||||
|
||||
See the file COPYING.FPC, included in this distribution,
|
||||
for details about the copyright.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||
|
||||
**********************************************************************}
|
||||
{$mode objfpc}
|
||||
|
||||
unit JSONDataset;
|
||||
|
@ -1,3 +1,19 @@
|
||||
{
|
||||
This file is part of the Free Pascal run time library.
|
||||
Copyright (c) 2019 by Michael Van Canneyt, member of the
|
||||
Free Pascal development team
|
||||
|
||||
Simple REST connection component for use with Datasets.
|
||||
|
||||
See the file COPYING.FPC, included in this distribution,
|
||||
for details about the copyright.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||
|
||||
**********************************************************************}
|
||||
|
||||
unit RestConnection;
|
||||
|
||||
{$mode objfpc}
|
||||
|
416
packages/fcl-db/sqldbrestdataset.pp
Normal file
416
packages/fcl-db/sqldbrestdataset.pp
Normal file
@ -0,0 +1,416 @@
|
||||
{
|
||||
This file is part of the Free Pascal run time library.
|
||||
Copyright (c) 2019 by Michael Van Canneyt, member of the
|
||||
Free Pascal development team
|
||||
|
||||
Simple SQLDBRESTBridge JSON dataset component and connection.
|
||||
|
||||
See the file COPYING.FPC, included in this distribution,
|
||||
for details about the copyright.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||
|
||||
**********************************************************************}
|
||||
unit sqldbrestdataset;
|
||||
|
||||
{$mode objfpc}
|
||||
|
||||
interface
|
||||
|
||||
uses
|
||||
Classes, SysUtils, JS, web, db, JSONDataset, restconnection;
|
||||
|
||||
Type
|
||||
|
||||
{ TSQLDBRestConnection }
|
||||
|
||||
TSQLDBRestConnection = Class(TRestConnection)
|
||||
private
|
||||
FDataProperty: String;
|
||||
FmetaDataProperty: String;
|
||||
FMetaDataResourceName: String;
|
||||
FonGetResources: TNotifyEvent;
|
||||
FPassword: String;
|
||||
FResourceList: TStrings;
|
||||
FUserName: String;
|
||||
procedure DoResources(Sender: TObject);
|
||||
function DoStoreDataProp: Boolean;
|
||||
function DoStoreMetadata: Boolean;
|
||||
function DoStoreMetadataProp: Boolean;
|
||||
Protected
|
||||
Procedure SetupRequest(aXHR : TJSXMLHttpRequest); override;
|
||||
Function GetUpdateBaseURL(aRequest: TRecordUpdateDescriptor): String; override;
|
||||
Function GetReadBaseURL(aRequest: TDataRequest): String; Override;
|
||||
Public
|
||||
Constructor create(aOwner : TComponent); override;
|
||||
Destructor Destroy; override;
|
||||
Procedure GetResources(OnResult : TNotifyEvent = Nil);
|
||||
Property ResourceList : TStrings Read FResourceList;
|
||||
Published
|
||||
Property OnGetResources : TNotifyEvent Read FonGetResources Write FOnGetResources;
|
||||
Property metaDataProperty : String read FmetaDataProperty Write FmetaDataProperty Stored DoStoreMetadataProp;
|
||||
Property DataProperty : String read FDataProperty Write FDataProperty Stored DoStoreDataProp;
|
||||
Property MetaDataResourceName : String Read FMetaDataResourceName Write FMetaDataResourceName Stored DoStoreMetadata;
|
||||
Property UserName : String Read FUserName Write FUserName;
|
||||
Property Password : String Read FPassword Write FPassword;
|
||||
end;
|
||||
|
||||
{ TSQLDBRestDataset }
|
||||
|
||||
TSQLDBRestDataset = Class(TJSONDataset)
|
||||
private
|
||||
FConnection: TSQLDBRestConnection;
|
||||
FResourceName: String;
|
||||
procedure SetConnection(AValue: TSQLDBRestConnection);
|
||||
procedure SetResourceName(AValue: String);
|
||||
Protected
|
||||
function DataPacketReceived(ARequest: TDataRequest): Boolean; override;
|
||||
function GetStringFieldLength(F: TJSObject; AName: String; AIndex: Integer): integer;virtual;
|
||||
function StringToFieldType(S: String): TFieldType; virtual;
|
||||
Function DoGetDataProxy: TDataProxy; override;
|
||||
Procedure MetaDataToFieldDefs; override;
|
||||
Public
|
||||
Property Connection: TSQLDBRestConnection Read FConnection Write SetConnection;
|
||||
Property ResourceName : String Read FResourceName Write SetResourceName;
|
||||
end;
|
||||
|
||||
implementation
|
||||
|
||||
Type
|
||||
|
||||
{ TServiceRequest }
|
||||
|
||||
TServiceRequest = Class(TObject)
|
||||
Private
|
||||
FOnMyDone,
|
||||
FOnDone : TNotifyEvent;
|
||||
FXHR: TJSXMLHttpRequest;
|
||||
function GetResult: String;
|
||||
function GetResultJSON: TJSObject;
|
||||
function GetStatusCode: Integer;
|
||||
function onLoad(Event{%H-}: TEventListenerEvent): boolean;
|
||||
Public
|
||||
Constructor Create(Const aMethod,aURL,aUserName,aPassword : String; aOnDone1 : TNotifyEvent; aOnDone2 : TNotifyEvent = Nil);
|
||||
Procedure Execute;
|
||||
Property RequestResult : String read GetResult;
|
||||
Property ResultJSON : TJSObject read GetResultJSON;
|
||||
Property OnDone : TNotifyEvent Read FOnDone;
|
||||
Property StatusCode : Integer Read GetStatusCode;
|
||||
end;
|
||||
|
||||
{ TServiceRequest }
|
||||
|
||||
constructor TServiceRequest.Create(const aMethod,aURL, aUserName, aPassword: String; aOnDone1 : TNotifyEvent; aOnDone2 : TNotifyEvent = Nil);
|
||||
begin
|
||||
FOnMyDone:=aOnDone1;
|
||||
FOnDone:=aOnDone2;
|
||||
FXHR:=TJSXMLHttpRequest.New;
|
||||
FXHR.AddEventListener('load',@onLoad);
|
||||
FXHR.open(aMethod,aURL,true);
|
||||
(* else
|
||||
begin
|
||||
// FXHR.withCredentials := true;
|
||||
FXHR.open(aMethod,aURL,true,aUserName,aPassword);
|
||||
end;*)
|
||||
FXHR.setRequestHeader('Content-Type', 'application/json');
|
||||
FXHR.setRequestHeader('Authorization', 'Basic '+window.btoa(aUserName+':'+aPassword));
|
||||
end;
|
||||
|
||||
procedure TServiceRequest.Execute;
|
||||
begin
|
||||
FXHR.send;
|
||||
end;
|
||||
|
||||
function TServiceRequest.GetResult: String;
|
||||
begin
|
||||
Result:=FXHR.responseText;
|
||||
end;
|
||||
|
||||
function TServiceRequest.GetResultJSON: TJSObject;
|
||||
begin
|
||||
if SameText(FXHR.getResponseHeader('Content-Type'),'application/json') then
|
||||
Result:=TJSJSON.parseObject(GetResult)
|
||||
else
|
||||
Result:=nil;
|
||||
end;
|
||||
|
||||
function TServiceRequest.GetStatusCode: Integer;
|
||||
begin
|
||||
Result:=FXHR.Status;
|
||||
end;
|
||||
|
||||
function TServiceRequest.onLoad(Event: TEventListenerEvent): boolean;
|
||||
begin
|
||||
if Assigned(FOnMyDone) then
|
||||
FOnMyDone(Self);
|
||||
end;
|
||||
|
||||
|
||||
|
||||
|
||||
{ TSQLDBRestConnection }
|
||||
|
||||
function TSQLDBRestConnection.DoStoreMetadata: Boolean;
|
||||
begin
|
||||
Result:=(FMetadataResourceName<>'metadata');
|
||||
end;
|
||||
|
||||
function TSQLDBRestConnection.DoStoreMetadataProp: Boolean;
|
||||
begin
|
||||
Result:=(FMetaDataProperty<>'metaData');
|
||||
end;
|
||||
|
||||
procedure TSQLDBRestConnection.SetupRequest(aXHR: TJSXMLHttpRequest);
|
||||
begin
|
||||
inherited SetupRequest(aXHR);
|
||||
aXHR.setRequestHeader('Content-Type', 'application/json');
|
||||
aXHR.setRequestHeader('Accept', 'application/json');
|
||||
if (UserName<>'') then
|
||||
aXHR.setRequestHeader('Authorization', 'Basic '+window.btoa(UserName+':'+Password));
|
||||
end;
|
||||
|
||||
function TSQLDBRestConnection.GetUpdateBaseURL(aRequest: TRecordUpdateDescriptor): String;
|
||||
begin
|
||||
Result:=inherited GetUpdateBaseURL(aRequest);
|
||||
Result:=IncludeTrailingPathDelimiter(Result)+TSQLDBRestDataset(aRequest.Dataset).ResourceName;
|
||||
end;
|
||||
|
||||
function TSQLDBRestConnection.GetReadBaseURL(aRequest: TDataRequest): String;
|
||||
begin
|
||||
Result:=inherited GetReadBaseURL(aRequest);
|
||||
Result:=IncludeTrailingPathDelimiter(Result)+TSQLDBRestDataset(aRequest.Dataset).ResourceName;
|
||||
end;
|
||||
|
||||
procedure TSQLDBRestConnection.DoResources(Sender: TObject);
|
||||
|
||||
Var
|
||||
R : TServiceRequest absolute Sender;
|
||||
J,Res : TJSObject;
|
||||
A : TJSArray;
|
||||
i : Integer;
|
||||
|
||||
begin
|
||||
FResourceList.Clear;
|
||||
if (R.StatusCode=200) then
|
||||
begin
|
||||
J:=R.ResultJSON;
|
||||
if J=Nil then
|
||||
exit;
|
||||
A:=TJSArray(J.Properties['data']);
|
||||
For I:=0 to A.Length-1 do
|
||||
begin
|
||||
Res:=TJSObject(A[i]);
|
||||
FResourceList.Add(String(Res.Properties['name']));
|
||||
end;
|
||||
end;
|
||||
If Assigned(R.OnDone) then
|
||||
R.OnDone(Self);
|
||||
If Assigned(OnGetResources) then
|
||||
OnGetResources(Self);
|
||||
end;
|
||||
|
||||
function TSQLDBRestConnection.DoStoreDataProp: Boolean;
|
||||
begin
|
||||
Result:=(FDataProperty<>'data');
|
||||
end;
|
||||
|
||||
constructor TSQLDBRestConnection.create(aOwner: TComponent);
|
||||
begin
|
||||
inherited create(aOwner);
|
||||
FResourceList:=TStringList.Create;
|
||||
FMetaDataResourceName:='metadata';
|
||||
FmetaDataProperty:='metaData';
|
||||
FDataProperty:='data';
|
||||
TStringList(FResourceList).Sorted:=true;
|
||||
end;
|
||||
|
||||
destructor TSQLDBRestConnection.Destroy;
|
||||
begin
|
||||
FreeAndNil(FResourceList);
|
||||
inherited Destroy;
|
||||
end;
|
||||
|
||||
procedure TSQLDBRestConnection.GetResources(OnResult: TNotifyEvent);
|
||||
|
||||
Var
|
||||
aURL : String;
|
||||
R : TServiceRequest;
|
||||
|
||||
begin
|
||||
aURL:=IncludeTrailingPathDelimiter(BaseURL)+MetaDataResourceName+'?fmt=json';
|
||||
R:=TServiceRequest.Create('GET',aURL,Self.UserName,Self.Password,@DoResources,OnResult);
|
||||
R.Execute;
|
||||
end;
|
||||
|
||||
{ TSQLDBRestDataset }
|
||||
|
||||
procedure TSQLDBRestDataset.SetConnection(AValue: TSQLDBRestConnection);
|
||||
begin
|
||||
if FConnection=AValue then Exit;
|
||||
if Assigned(FConnection) then
|
||||
FConnection.RemoveFreeNotification(Self);
|
||||
FConnection:=AValue;
|
||||
if Assigned(FConnection) then
|
||||
FConnection.FreeNotification(Self);
|
||||
end;
|
||||
|
||||
procedure TSQLDBRestDataset.SetResourceName(AValue: String);
|
||||
begin
|
||||
if FResourceName=AValue then Exit;
|
||||
CheckInactive;
|
||||
FResourceName:=AValue;
|
||||
end;
|
||||
|
||||
function TSQLDBRestDataset.DoGetDataProxy: TDataProxy;
|
||||
begin
|
||||
Result:=Connection.DataProxy;
|
||||
end;
|
||||
|
||||
function TSQLDBRestDataset.StringToFieldType(S: String): TFieldType;
|
||||
|
||||
begin
|
||||
if (s='int') then
|
||||
Result:=ftInteger
|
||||
else if (s='bigint') then
|
||||
Result:=ftLargeInt
|
||||
else if (s='float') then
|
||||
Result:=ftFloat
|
||||
else if (s='bool') then
|
||||
Result:=ftBoolean
|
||||
else if (s='date') then
|
||||
Result:=ftDate
|
||||
else if (s='datetime') then
|
||||
Result:=ftDateTime
|
||||
else if (s='time') then
|
||||
Result:=ftTime
|
||||
else if (s='blob') then
|
||||
Result:=ftBlob
|
||||
else if (s='string') then
|
||||
Result:=ftString
|
||||
else
|
||||
if MapUnknownToStringType then
|
||||
Result:=ftString
|
||||
else
|
||||
Raise EJSONDataset.CreateFmt('Unknown JSON data type : %s',[s]);
|
||||
end;
|
||||
|
||||
function TSQLDBRestDataset.GetStringFieldLength(F: TJSObject; AName: String;
|
||||
AIndex: Integer): integer;
|
||||
|
||||
Var
|
||||
I,L : Integer;
|
||||
D : JSValue;
|
||||
|
||||
begin
|
||||
Result:=0;
|
||||
D:=F.Properties['maxLen'];
|
||||
if Not jsIsNan(toNumber(D)) then
|
||||
begin
|
||||
Result:=Trunc(toNumber(D));
|
||||
if (Result<=0) then
|
||||
Raise EJSONDataset.CreateFmt('Invalid maximum length specifier for field %s',[AName])
|
||||
end
|
||||
else
|
||||
begin
|
||||
For I:=0 to Rows.Length-1 do
|
||||
begin
|
||||
D:=FieldMapper.GetJSONDataForField(Aname,AIndex,Rows[i]);
|
||||
if isString(D) then
|
||||
begin
|
||||
l:=Length(String(D));
|
||||
if L>Result then
|
||||
Result:=L;
|
||||
end;
|
||||
end;
|
||||
end;
|
||||
if (Result=0) then
|
||||
Result:=20;
|
||||
end;
|
||||
|
||||
procedure TSQLDBRestDataset.MetaDataToFieldDefs;
|
||||
Var
|
||||
A : TJSArray;
|
||||
F : TJSObject;
|
||||
I,FS : Integer;
|
||||
N: String;
|
||||
ft: TFieldType;
|
||||
D : JSValue;
|
||||
|
||||
begin
|
||||
FieldDefs.Clear;
|
||||
D:=Metadata.Properties['fields'];
|
||||
if Not IsArray(D) then
|
||||
Raise EJSONDataset.Create('Invalid metadata object');
|
||||
A:=TJSArray(D);
|
||||
For I:=0 to A.Length-1 do
|
||||
begin
|
||||
If Not isObject(A[i]) then
|
||||
Raise EJSONDataset.CreateFmt('Field definition %d in metadata is not an object',[i]);
|
||||
F:=TJSObject(A[i]);
|
||||
D:=F.Properties['name'];
|
||||
If Not isString(D) then
|
||||
Raise EJSONDataset.CreateFmt('Field definition %d in has no or invalid name property',[i]);
|
||||
N:=String(D);
|
||||
D:=F.Properties['type'];
|
||||
If IsNull(D) or isUndefined(D) then
|
||||
ft:=ftstring
|
||||
else If Not isString(D) then
|
||||
begin
|
||||
Raise EJSONDataset.CreateFmt('Field definition %d in has invalid type property',[i])
|
||||
end
|
||||
else
|
||||
begin
|
||||
ft:=StringToFieldType(String(D));
|
||||
end;
|
||||
if (ft=ftString) then
|
||||
fs:=GetStringFieldLength(F,N,I)
|
||||
else
|
||||
fs:=0;
|
||||
FieldDefs.Add(N,ft,fs);
|
||||
end;
|
||||
end;
|
||||
|
||||
function TSQLDBRestDataset.DataPacketReceived(ARequest: TDataRequest): Boolean;
|
||||
|
||||
Var
|
||||
O : TJSObject;
|
||||
A : TJSArray;
|
||||
smetadata,sroot : String;
|
||||
begin
|
||||
Result:=False;
|
||||
If isNull(aRequest.Data) then
|
||||
exit;
|
||||
If isString(aRequest.Data) then
|
||||
O:=TJSOBject(TJSJSON.Parse(String(aRequest.Data)))
|
||||
else if isObject(aRequest.Data) then
|
||||
O:=TJSOBject(aRequest.Data)
|
||||
else
|
||||
DatabaseError('Cannot handle data packet');
|
||||
sRoot:=Connection.DataProperty;
|
||||
sMetaData:=Connection.metaDataProperty;
|
||||
if (sroot='') then
|
||||
sroot:='data';
|
||||
if (smetadata='') then
|
||||
smetadata:='metaData';
|
||||
{ if (IDField='') then
|
||||
idField:='id';}
|
||||
if O.hasOwnProperty(sMetaData) and isObject(o[sMetaData]) then
|
||||
begin
|
||||
if not Active then // Load fields from metadata
|
||||
metaData:=TJSObject(o[SMetaData]);
|
||||
{ if metaData.hasOwnProperty('idField') and isString(metaData['idField']) then
|
||||
IDField:=string(metaData['idField']);}
|
||||
end;
|
||||
if O.hasOwnProperty(sRoot) and isArray(o[sRoot]) then
|
||||
begin
|
||||
A:=TJSArray(o[sRoot]);
|
||||
Result:=A.Length>0;
|
||||
AddToRows(A);
|
||||
end;
|
||||
end;
|
||||
|
||||
|
||||
end.
|
||||
|
Loading…
Reference in New Issue
Block a user