forked from VSoftTechnologies/Delphi-Mocks
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathDelphi.Mocks.InterfaceProxy.pas
More file actions
164 lines (133 loc) · 5.54 KB
/
Delphi.Mocks.InterfaceProxy.pas
File metadata and controls
164 lines (133 loc) · 5.54 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
{***************************************************************************}
{ }
{ Delphi.Mocks }
{ }
{ Copyright (C) 2011 Vincent Parrett }
{ }
{ http://www.finalbuilder.com }
{ }
{ }
{***************************************************************************}
{ }
{ 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 Delphi.Mocks.InterfaceProxy;
interface
uses
Rtti,
SysUtils,
Generics.Collections,
Delphi.Mocks,
Delphi.Mocks.Interfaces,
Delphi.Mocks.ProxyBase;
type
TProxyBaseInvokeEvent = procedure (Method: TRttiMethod; const Args: TArray<TValue>; out Result: TValue) of object;
TSetupMode = (None, Behavior, Expectation);
TInterfaceProxy<T> = class(TBaseProxy<T>)
private type
TProxyVirtualInterface = class(TVirtualInterface)
private
FProxy : TInterfaceProxy<T>;
protected
public
function QueryInterface(const IID: TGUID; out Obj): HRESULT; override; stdcall;
constructor Create(AProxy : TInterfaceProxy<T>; AInterface: Pointer; InvokeEvent: TVirtualInterfaceInvokeEvent);
end;
private
FVirtualInterfaces : TDictionary<TGUID, IInterface>;
protected
function InternalQueryInterface(const IID: TGUID; out Obj): HRESULT; stdcall;
function QueryInterface(const IID: TGUID; out Obj): HRESULT; override;
function Proxy : T;override;
function CastAs<I: IInterface> : I;
public
constructor Create(const AIsStubOnly : boolean = false);override;
destructor Destroy;override;
end;
implementation
uses
TypInfo;
{ TInterfaceProxy<T> }
function TInterfaceProxy<T>.CastAs<I>: I;
var
virtualProxy : TProxyVirtualInterface;
begin
virtualProxy := TProxyVirtualInterface.Create(Self, TypeInfo(I), Self.DoInvoke);
FVirtualInterfaces.Add(GetTypeData(TypeInfo(I)).Guid, virtualProxy);
virtualProxy.QueryInterface(GetTypeData(TypeInfo(I)).Guid,result);
end;
constructor TInterfaceProxy<T>.Create(const AIsStubOnly : boolean);
var
virtualProxy : TProxyVirtualInterface;
begin
inherited Create(AIsStubOnly);
FVirtualInterfaces := TDictionary<TGUID, IInterface>.Create;
virtualProxy := TProxyVirtualInterface.Create(Self, TypeInfo(T), Self.DoInvoke);
FVirtualInterfaces.Add(GetTypeData(TypeInfo(T)).Guid, virtualProxy);
end;
destructor TInterfaceProxy<T>.Destroy;
begin
FVirtualInterfaces.Clear;
FreeAndNil(FVirtualInterfaces);
inherited;
end;
function TInterfaceProxy<T>.InternalQueryInterface(const IID: TGUID; out Obj): HRESULT;
begin
Result := E_NOINTERFACE;
if (IsEqualGUID(IID, IInterface)) then
if GetInterface(IID, Obj) then
Result := 0;
end;
function TInterfaceProxy<T>.Proxy: T;
var
pInfo : PTypeInfo;
virtualProxy : IInterface;
begin
pInfo := TypeInfo(T);
if FVirtualInterfaces.ContainsKey(GetTypeData(pInfo).Guid) then
virtualProxy := FVirtualInterfaces.Items[GetTypeData(pInfo).Guid]
else
raise EMockNoProxyException.Create('Error proxy casting to interface');
if virtualProxy.QueryInterface(GetTypeData(pInfo).Guid,result) <> 0 then
raise EMockNoProxyException.Create('Error casting to interface ' + pInfo.NameStr + ' , proxy does not appear to implememnt T');
end;
function TInterfaceProxy<T>.QueryInterface(const IID: TGUID; out Obj): HRESULT;
var
virtualProxy : IInterface;
begin
Result := E_NOINTERFACE;
if (FVirtualInterfaces <> nil) then
if (FVirtualInterfaces.Count <> 0) then
if (FVirtualInterfaces.ContainsKey(IID)) then
begin
virtualProxy := FVirtualInterfaces.Items[IID];
Result := virtualProxy.QueryInterface(IID, Obj);
end;
if result <> 0 then
Result := inherited;
end;
{ TInterfaceProxy<T>.TProxyVirtualInterface }
constructor TInterfaceProxy<T>.TProxyVirtualInterface.Create(AProxy: TInterfaceProxy<T>;
AInterface: Pointer; InvokeEvent: TVirtualInterfaceInvokeEvent);
begin
FProxy := AProxy;
inherited Create(Ainterface, InvokeEvent);
end;
function TInterfaceProxy<T>.TProxyVirtualInterface.QueryInterface(const IID: TGUID; out Obj): HRESULT;
begin
Result := inherited;
if Result <> 0 then
Result := FProxy.InternalQueryInterface(IID, Obj);
end;
end.