REST easy with kbmMW #21 – Delphi client stubs
在之前的博文中,我提到了存根生成器框架,将具有生成Delphi客户端存根代码的能力,使得开发Delphi智能客户端变的更容易,完全支持编译时的类型检查和IDE类/属性帮助。
本不想把这个新功能包含在即将发布的kbmMW版本中,但我无法自控.... 必须添加它才能生成相当完整的客户端存根代码。
存根生成器将生成普通的Delphi代码,需要kbmMW智能客户端功能进行编译。但代码已完成,并包含所需的所有单元引用。如果您使用的是包含对象的自定义单元,则必须将这些单元与生成的存根代码一起提供给Delphi客户端开发人员。
除OpenAPI支持外,SimpleInvocation服务器还支持通过REST接口直接返回Delphi存根代码。
让我们称之为:http://localhost:888/myserver/delphi
这将返回给你一个类似于这样的对话框:

保存它,将得一个单元文件:
unit uSMARTDEMO; interface uses
kbmMWGlobal,
kbmMWSmartUtils,
kbmMWSmartClient,
kbmMWClient,
kbmMWClientDataset
,uObjects
,System.Generics.Collections
,kbmMWDateTime
,kbmMemTable
; type
// Name.......: SMARTDEMO
// Version....: kbmMW_1.0
// Abstract...: HTTP smart service supp. FastCGI
// Description: Smart HTTP service (with optional FastCGI and URL rewrite support) which use annotations for publishing methods for both regular clients and REST clients.
//-------------------------------------
TSMARTDEMO = class
private
FClient:IkbmMWSmartClient;
public
constructor Create(const ATransport:TkbmMWCustomClientTransport); overload; virtual;
constructor Create(const AConnectionPool:TkbmMWClientConnectionPool); overload; virtual;
constructor Create(const AClient:IkbmMWSmartClient); overload; virtual;
function StorePerson(const APerson:TPerson):Integer;
function EchoPerson(const APerson:TPerson):TPerson;
function EchoURL:string;
function AddNumbers(const AValue1:Integer; const AValue2:Integer):Integer;
function EchoString(const AString:string):string;
function GetPersons:TObjectList<uObjects.TPerson>;
function GetPerson(const id:Integer):TPerson;
function ServerNow1:TkbmMWDateTime;
function ServerNow2:TDateTime;
function GetPerson3:TPerson3;
function StorePerson3(const APerson:IPerson2):Integer;
function StorePerson2(const APerson:TPerson):Integer;
function ReverseString(const AString:string):string;
function EchoBytes(const ABytes:TArray<System.Byte>):TArray<System.Byte>;
function GetMemTable:TkbmMemTable;
function ReverseStringFromConfig:string;
end; implementation constructor TSMARTDEMO.Create(const ATransport:TkbmMWCustomClientTransport);
begin
inherited Create;
FClient:=TkbmMWSmartRemoteClientFactory.GetClient(ATransport,'SMARTDEMO');
end; constructor TSMARTDEMO.Create(const AConnectionPool:TkbmMWClientConnectionPool);
begin
inherited Create;
FClient:=TkbmMWSmartRemoteClientFactory.GetClient(AConnectionPool,'SMARTDEMO');
end; constructor TSMARTDEMO.Create(const AClient:IkbmMWSmartClient);
begin
inherited Create;
FClient:=AClient;
end; function TSMARTDEMO.StorePerson(const APerson:TPerson):Integer;
var
v0:variant;
begin
v0:=Use.AsVariant(APerson,false);
Result:=FClient.Service.StorePerson(v0);
end; function TSMARTDEMO.EchoPerson(const APerson:TPerson):TPerson;
var
v0:variant;
begin
v0:=Use.AsVariant(APerson,false);
Result:=Use.AsMyObject<TPerson>(FClient.Service.EchoPerson(v0));
end; function TSMARTDEMO.EchoURL:string;
begin
Result:=FClient.Service.EchoURL;
end; function TSMARTDEMO.AddNumbers(const AValue1:Integer; const AValue2:Integer):Integer;
begin
Result:=FClient.Service.AddNumbers(AValue1,AValue2);
end; function TSMARTDEMO.EchoString(const AString:string):string;
begin
Result:=FClient.Service.EchoString(AString);
end; function TSMARTDEMO.GetPersons:TObjectList<uObjects.TPerson>;
begin
Result:=Use.AsMyObject<TObjectList<uObjects.TPerson>>(FClient.Service.GetPersons);
end; function TSMARTDEMO.GetPerson(const id:Integer):TPerson;
begin
Result:=Use.AsMyObject<TPerson>(FClient.Service.GetPerson(id));
end; function TSMARTDEMO.ServerNow1:TkbmMWDateTime;
begin
Result:=Use.AsType<TkbmMWDateTime>(FClient.Service.ServerNow1);
end; function TSMARTDEMO.ServerNow2:TDateTime;
begin
Result:=FClient.Service.ServerNow2;
end; function TSMARTDEMO.GetPerson3:TPerson3;
begin
Result:=Use.AsMyObject<TPerson3>(FClient.Service.GetPerson3);
end; function TSMARTDEMO.StorePerson3(const APerson:IPerson2):Integer;
var
v0:variant;
begin
v0:=Use.AsVariant(APerson);
Result:=FClient.Service.StorePerson3(v0);
end; function TSMARTDEMO.StorePerson2(const APerson:TPerson):Integer;
var
v0:variant;
begin
v0:=Use.AsVariant(APerson,false);
Result:=FClient.Service.StorePerson2(v0);
end; function TSMARTDEMO.ReverseString(const AString:string):string;
begin
Result:=FClient.Service.ReverseString(AString);
end; function TSMARTDEMO.EchoBytes(const ABytes:TArray<System.Byte>):TArray<System.Byte>;
var
v0:variant;
begin
v0:=Use.AsVariant(ABytes);
Result:=Use.AsBytes(FClient.Service.EchoBytes(v0));
end; function TSMARTDEMO.GetMemTable:TkbmMemTable;
begin
Result:=Use.AsMyObject<TkbmMemTable>(FClient.Service.GetMemTable);
end; function TSMARTDEMO.ReverseStringFromConfig:string;
begin
Result:=FClient.Service.ReverseStringFromConfig;
end; end.
您可能已经注意到,SmartInvocation服务中已使用kbmMW_Method属性标记的所有方法都在存根代码中生成了。
存根生成器自动检测编译所需的各种其他单元,如kbmMemTable和uObjects,并添加到uses子句中。
那么我们是如何从REST接口返回存根代码的呢?
在包含REST服务的单元中,我们只添加了一个返回字符串的函数,并使其可以从REST访问。
[kbmMW_Rest('method:get, path: "delphi", responseMimeType:"text/plain"')]
function DelphiAPI:string;
功能实现非常简单
// Return Delphi stub.
function TkbmMWCustomService2.DelphiAPI:string;
var
unitName:string;
begin
// Return Delphi client stub unit for all methods in this service marked with
// kbmMW_Method attribute.
// Add 'servers: [ "url1", "url2",.. "urln" ]' to ASettings if you want to
// embed server location information in the comments of the returned unit.
Result:=TkbmMWSmartDelphiStubGenerator.GenerateDelphi('',self,unitName);
SetResponseFileName(unitName);
end;
显然你可以从你想要的任何单位调用GenerateDelphi,你可以参考任何kbmMW Smartservice。GenerateDelphi的第一个参数是可选的单元命名前缀。如果未指定任何内容,则单元名将以“u”开头。在这种情况下,我们将收到一个名为uSMARTDEMO.pas的文件,因为它是在Delphi客户端存根中描述的SMARTDEMO服务。
让我们尝试在客户端中使用生成的代码。我们创建了一个基于VCL的标准客户端(kbmMW也可以与Firemonkey一起使用,以便跨平台使用)。我们添加了一个客户端传输(在本例中为TkbmMWTCPIPIndyClientTransport),一个客户端连接池和一些将激活这些调用的按钮。此外,还有一个编辑框,用于键入我们要连接的服务器的IP地址(如果它未在本地主机上运行)。

然后确保将uSMARTDEMO.pas添加到单元的uses子句中。现在我们可以访问存根代码,这样可以轻松调用kbmMW应用程序服务器托管的智能服务。
StorePerson 1的事件处理程序可能如下所示:
procedure TForm1.Button1Click(Sender: TObject);
var
i:integer;
person:IPerson2;
demo:TSMARTDEMO;
begin
Transport.Host:=eIP.Text; demo:=TSMARTDEMO.Create(Transport);
try
person:=TPerson2.Create;
person.Name:='Frank Borland Jr';
person.Address:='California';
person.Age:=;
i:=demo.StorePerson3(person);
finally
demo.Free;
end;
end;
这里的示例使用我们想要使用的传输实例化TSMARTDEMO实例。如果我们愿意,我们可以提供连接池或智能客户端。存根代码支持所有3种变体。
此按钮显示如何将接口对象发送到服务器。
StorePerson 2的事件处理程序可能如下所示:
procedure TForm1.Button2Click(Sender: TObject);
var
person:TPerson;
i:integer;
demo:TSMARTDEMO;
begin
Transport.Host:=eIP.Text; demo:=TSMARTDEMO.Create(Transport);
try
person:=TPerson.Create;
try
person.Name:='Trudy Borland';
person.Address:='California';
person.Age:=;
i:=demo.StorePerson(person);
finally
person.Free;
end;
finally
demo.Free;
end;
end;
略有不同的是,我们为呼叫提供了一个对象。通话结束后,我们释放了对象。所有完全正常的Delphi代码。
Simple Calls的事件处理程序可能如下所示:
procedure TForm1.Button3Click(Sender: TObject);
var
dt1:TkbmMWDateTime;
dt2:TDateTime;
s:string;
i:integer;
demo:TSMARTDEMO;
ba,ba1:TArray<System.Byte>;
begin
Transport.Host:=eIP.Text; demo:=TSMARTDEMO.Create(Transport);
try
s:=demo.EchoString('abc'); SetLength(ba1,);
ba1[]:=;
ba1[]:=;
ba1[]:=;
ba1[]:=;
ba1[]:=;
ba:=demo.EchoBytes(ba1); i:=demo.AddNumbers(,);
dt1:=demo.ServerNow1;
dt2:=demo.ServerNow2;
finally
demo.Free;
end;
end;
它显示了发送和接收的各种类型。
GetPerson的事件处理程序可能如下所示:
procedure TForm1.Button4Click(Sender: TObject);
var
person:TPerson;
person3:TPerson3;
persons:TObjectList<TPerson>;
demo:TSMARTDEMO;
begin
Transport.Host:=eIP.Text; demo:=TSMARTDEMO.Create(Transport);
try
person:=demo.GetPerson();
person.Free; person3:=demo.GetPerson3;
person3.Free; persons:=demo.GetPersons;
persons.Free;
finally
demo.Free;
end;
end;
再次......只显示如何接收各种类型的对象(不使用kbmMWNullable的常规对象,使用kbmMWNullable的对象和对象列表)。
最后,事件处理程序GetMemTable可能如下所示:
procedure TForm1.Button5Click(Sender: TObject);
var
mt:TkbmMemtable;
demo:TSMARTDEMO;
begin
Transport.Host:=eIP.Text; demo:=TSMARTDEMO.Create(Transport);
try
mt:=demo.GetMemTable;
mt.Free;
finally
demo.Free;
end;
end;
此示例显示如何在服务器上生成包含5条记录的内存表实例,从服务器返回到客户端。
在Delphi中制作高级应用程序服务器是否更容易?Delphi通过REST或本机Delphi代码支持各种客户端?我不这么认为
所以......如果你喜欢你所看到的,请分享这个词。转发博客帖子,让其他人了解该产品!
https://components4developers.blog/2019/01/13/rest-easy-with-kbmmw-21-delphi_client_stubs/
REST easy with kbmMW #21 – Delphi client stubs的更多相关文章
- REST easy with kbmMW #3 – SSL
我在前两篇文章中展示了“REST easy with kbmMW”文章,如何使用kbmMW制作REST服务器,以及如何使用该REST服务器轻松地从数据库返回和存储数据,所有这些都在不到30行的真实数据 ...
- REST easy with kbmMW #14 – DB Controlled login
介绍 关于如何使用授权和登录管理来构建应用服务器还存在一些问题,其中之一就是用户及其角色如何在在数据库中定义.该文将解释使用TkbmMWAuthorizationManager解决此问题的一种方法.有 ...
- REST easy with kbmMW #20 – OpenAPI and Swagger UI
即将推出的kbmMW更新不仅是一些bug修正,同时将包含一个新的主要功能:客户端存根生成器框架. 那什么是客户端存根生成器框架呢? 他是一个基于kbmMW smart services,可以生成由各种 ...
- REST easy with kbmMW #1
kbmMW 5.0支持REST服务器的开发,并且非常简单,下面看看如何实作一个REST服务器. 首先我们制作一个服务器应用程序,增加一个简单的Form,并放置kbmMW组件. 在Delphi中单击Fi ...
- REST easy with kbmMW #24 使用kbmMW实现JSON/XML/YAML转换成对象
你想过没有,把一个给定的xml或json生成一个Delphi类,并通过这个类完成对xml或json的读写操作吗? 不管有没有,现在kbmMW为我们实现了,看下面这行代码: var s:string; ...
- REST easy with kbmMW #17 – Database 6 – Existing databases
kbmMW已经包含了非常精细的功能来确定和解释数据库中表的元数据. 在下一版本中,这个功能将得到进一步加强,可以导入现有数据库中的表,自动创建与表相匹配的ORM实体类. 这意味着你能够使用kbmMW的 ...
- REST easy with kbmMW #15 – Handling HTTP POST
我被问到有关如何通过基于kbmMW智能服务(Smart Service)的REST处理POST的问题. 这篇博客文章解释了典型的POST各种形式的访问,以及如何在kbmMW中处理它们. POST变种W ...
- REST easy with kbmMW #16 – Multiple servers using HTTP.sys transport
前文写过使用HTTP.sys转输层(TkbmMWHTTPSysServerTransport),实现一个kbmMW应用服务器. 如果在一台服务器上,同时运行多个,基于TkbmMWHTTPSysServ ...
- REST easy with kbmMW #4 – Access management
在前面有关如何使用kbmMW创建REST服务器的基础上,现在已经到了考虑该如何控制用户的访问.什么是访问管理?就是“允许谁做什么"的问题. 显然,这个世界中存在数据,应该保护他而不被未授权的 ...
随机推荐
- 宽度优先搜索BFS(Breadth-First-Search)
Breadth-First-Search 1. 与DFS的异同 相同点:搜索所有可能的状态. 不同点:搜索顺序. 2. BFS总是先搜索距离初始状态近的状态,它是按照:开始状态->只需一次转移就 ...
- 牛客小白赛1 F题三视图
链接:https://www.nowcoder.com/acm/contest/85/F来源:牛客网 题目描述 Etéreo 拿出家里的许多的立方体积木,堆成了一个三维空间中的模型.既然你高考选了技术 ...
- POJ No.2386 Lake Counting
题目链接:http://poj.org/problem?id=2386 分析:八联通的则为水洼,我们则需遍历一个单位附近的八个单位并将它们都改成'.',但附近单位可能仍连接着有'W'的区域,这种情况下 ...
- 『计算机视觉』FPN:feature pyramid networks for object detection
对用卷积神经网络进行目标检测方法的一种改进,通过提取多尺度的特征信息进行融合,进而提高目标检测的精度,特别是在小物体检测上的精度.FPN是ResNet或DenseNet等通用特征提取网络的附加组件,可 ...
- 创建springboot的聚合工程(一)
比起传统复杂的单体工程,使用Maven的多模块配置,可以帮助项目划分模块,鼓励重用,防止POM变得过于庞大,方便某个模块的构建,而不用每次都构建整个项目,并且使得针对某个模块的特殊控制更为方便.接下来 ...
- 利用sqlldr从MySQL导出一张表数据到Oracle
根据业务需求,需要从MySQL库中同步一张表tap_application到Oracle中,下面是记录的导入过程. 1. 查看MySQL表结构 desc tap_application; +----- ...
- vue 项目中的坑 在项目中遇到 持续更新ing
1.vue2.0 不支持 v-html 后绑定的内容使用过滤,可是有时候过滤必须使用-----------解决:通过methods中定义方法 然后 v-html='myMethods(string)' ...
- Jquery的jqzoom插件的使用(图片放大镜)
今天学习一下,图片放大镜功能,需要使用插件JQzoom 引入文件 <script type="text/javascript" src="js/jquery.min ...
- linux 安装Git详细过程
1.首先git --version 查看 Git https://mirrors.edge.kernel.org/pub/software/scm/git/ 2.下载 wget https://Gi ...
- AI工具(星形工具)(光晕工具)(移动复制)(柜子绘制)5.12
星形工具;基本操作与矩形一样,拖动星形工具绘制,点击键盘上箭头增加星形的角数.下箭头减少星形的角数. 选择星形工具在屏幕单击,出现星形对话框,可以设置半径1半径2,角点数.图中的星形就可以用星形工具绘 ...