CSLA.NET 6.0.0
CSLA .NET is a software development framework that helps you build a reusable, maintainable object-oriented business layer for your app.
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Events Macros Pages
DataPortalTarget.cs
Go to the documentation of this file.
1//-----------------------------------------------------------------------
2// <copyright file="DataPortalTarget.cs" company="Marimer LLC">
3// Copyright (c) Marimer LLC. All rights reserved.
4// Website: https://cslanet.com
5// </copyright>
6// <summary>Encapsulates server-side data portal invocations</summary>
7//-----------------------------------------------------------------------
8using System;
9using System.Collections.Concurrent;
10using System.Threading.Tasks;
11#if NET5_0_OR_GREATER
12using System.Runtime.Loader;
13
14using Csla.Runtime;
15#endif
16using Csla.Reflection;
17
18namespace Csla.Server
19{
20 internal class DataPortalTarget : LateBoundObject
21 {
22#if NET5_0_OR_GREATER
23 private static readonly ConcurrentDictionary<Type, Tuple<string, DataPortalMethodNames>> _methodNameList =
24 new ConcurrentDictionary<Type, Tuple<string, DataPortalMethodNames>>();
25#else
26 private static readonly ConcurrentDictionary<Type, DataPortalMethodNames> _methodNameList = new();
27#endif
28
29 private readonly IDataPortalTarget _target;
30 private readonly DataPortalMethodNames _methodNames;
31
32 public DataPortalTarget(object obj)
33 : base(obj)
34 {
35 _target = obj as IDataPortalTarget;
36
37#if NET5_0_OR_GREATER
38 var objectType = obj.GetType();
39
40 var methodNameListInfo = _methodNameList.GetOrAdd(
41 objectType,
42 (t) => AssemblyLoadContextManager.CreateCacheInstance(
43 objectType,
44 DataPortalMethodNames.Default,
45 OnAssemblyLoadContextUnload
46 )
47 );
48
49 _methodNames = methodNameListInfo.Item2;
50#else
51 _methodNames = _methodNameList.GetOrAdd(obj.GetType(),
52 (t) => DataPortalMethodNames.Default);
53#endif
54 }
55
56 public void OnDataPortalInvoke(DataPortalEventArgs eventArgs)
57 {
58 if (_target != null)
59 _target.DataPortal_OnDataPortalInvoke(eventArgs);
60 else
61 CallMethodIfImplemented(_methodNames.OnDataPortalInvoke, eventArgs);
62 }
63
64 public void Child_OnDataPortalInvoke(DataPortalEventArgs eventArgs)
65 {
66 if (_target != null)
67 _target.Child_OnDataPortalInvoke(eventArgs);
68 else
69 CallMethodIfImplemented("Child_OnDataPortalInvoke", eventArgs);
70 }
71
72
73 public void OnDataPortalInvokeComplete(DataPortalEventArgs eventArgs)
74 {
75 if (_target != null)
76 _target.DataPortal_OnDataPortalInvokeComplete(eventArgs);
77 else
78 CallMethodIfImplemented(_methodNames.OnDataPortalInvokeComplete, eventArgs);
79 }
80
81 internal void Child_OnDataPortalInvokeComplete(DataPortalEventArgs eventArgs)
82 {
83 if (_target != null)
84 _target.Child_OnDataPortalInvokeComplete(eventArgs);
85 else
86 CallMethodIfImplemented("Child_OnDataPortalInvokeComplete", eventArgs);
87 }
88
89 internal void OnDataPortalException(DataPortalEventArgs eventArgs, Exception ex)
90 {
91 if (_target != null)
92 _target.DataPortal_OnDataPortalException(eventArgs, ex);
93 else
94 CallMethodIfImplemented(_methodNames.OnDataPortalException, eventArgs, ex);
95 }
96
97
98 internal void Child_OnDataPortalException(DataPortalEventArgs eventArgs, Exception ex)
99 {
100 if (_target != null)
101 _target.Child_OnDataPortalException(eventArgs, ex);
102 else
103 CallMethodIfImplemented("Child_OnDataPortalException", eventArgs, ex);
104 }
105
106 public void ThrowIfBusy()
107 {
108 if (Instance is Csla.Core.ITrackStatus busy && busy.IsBusy)
109 throw new InvalidOperationException(string.Format("{0}.IsBusy == true", Instance.GetType().Name));
110 }
111
112 public void MarkNew()
113 {
114 if (_target != null)
115 _target.MarkNew();
116 else
117 CallMethodIfImplemented("MarkNew");
118 }
119
120 public void MarkAsChild()
121 {
122 if (_target != null)
123 _target.MarkAsChild();
124 else
125 CallMethodIfImplemented("MarkAsChild");
126 }
127
128 internal void MarkOld()
129 {
130 if (_target != null)
131 _target.MarkOld();
132 else
133 CallMethodIfImplemented("MarkOld");
134 }
135
136 private async Task InvokeOperationAsync<T>(object criteria, bool isSync)
137 where T : DataPortalOperationAttribute
138 {
139 object[] parameters = DataPortal.GetCriteriaArray(criteria);
140 await CallMethodTryAsyncDI<T>(isSync, parameters).ConfigureAwait(false);
141 }
142
143 public async Task CreateAsync(object criteria, bool isSync)
144 {
145 await InvokeOperationAsync<CreateAttribute>(criteria, isSync).ConfigureAwait(false);
146 }
147
148 public async Task CreateChildAsync(params object[] parameters)
149 {
150 await CallMethodTryAsyncDI<CreateChildAttribute>(false, parameters).ConfigureAwait(false);
151 }
152
153 public async Task FetchAsync(object criteria, bool isSync)
154 {
155 await InvokeOperationAsync<FetchAttribute>(criteria, isSync).ConfigureAwait(false);
156 }
157
158 public async Task FetchChildAsync(params object[] parameters)
159 {
160 await CallMethodTryAsyncDI<FetchChildAttribute>(false, parameters).ConfigureAwait(false);
161 }
162
163
164 public async Task UpdateAsync(bool isSync)
165 {
166 if (Instance is Core.BusinessBase busObj)
167 {
168 if (busObj.IsDeleted)
169 {
170 if (!busObj.IsNew)
171 {
172 // tell the object to delete itself
173 await InvokeOperationAsync<DeleteSelfAttribute>(EmptyCriteria.Instance, isSync).ConfigureAwait(false);
174 }
175 MarkNew();
176 }
177 else
178 {
179 if (busObj.IsNew)
180 {
181 // tell the object to insert itself
182 await InvokeOperationAsync<InsertAttribute>(EmptyCriteria.Instance, isSync).ConfigureAwait(false);
183 }
184 else
185 {
186 // tell the object to update itself
187 await InvokeOperationAsync<UpdateAttribute>(EmptyCriteria.Instance, isSync).ConfigureAwait(false);
188 }
189 MarkOld();
190 }
191 }
192 else
193 {
194 // this is an updatable collection or some other
195 // non-BusinessBase type of object
196 // tell the object to update itself
197 await InvokeOperationAsync<UpdateAttribute>(EmptyCriteria.Instance, isSync).ConfigureAwait(false);
198 MarkOld();
199 }
200 }
201
202 public async Task UpdateChildAsync(params object[] parameters)
203 {
204 // tell the business object to update itself
205 if (Instance is Core.BusinessBase busObj)
206 {
207 if (busObj.IsDeleted)
208 {
209 if (!busObj.IsNew)
210 {
211 // tell the object to delete itself
212 await CallMethodTryAsyncDI<DeleteSelfChildAttribute>(false, parameters).ConfigureAwait(false);
213 MarkNew();
214 }
215 }
216 else
217 {
218 if (busObj.IsNew)
219 {
220 // tell the object to insert itself
221 await CallMethodTryAsyncDI<InsertChildAttribute>(false, parameters).ConfigureAwait(false);
222 }
223 else
224 {
225 // tell the object to update itself
226 await CallMethodTryAsyncDI<UpdateChildAttribute>(false, parameters).ConfigureAwait(false);
227 }
228 MarkOld();
229 }
230
231 }
232 else if (Instance is Core.ICommandObject)
233 {
234 // tell the object to update itself
235 await CallMethodTryAsyncDI<ExecuteChildAttribute>(false, parameters).ConfigureAwait(false);
236 }
237 else
238 {
239 // this is an updatable collection or some other
240 // non-BusinessBase type of object
241 // tell the object to update itself
242 await CallMethodTryAsyncDI<UpdateChildAttribute>(false, parameters).ConfigureAwait(false);
243 MarkOld();
244 }
245 }
246
247 public async Task ExecuteAsync(bool isSync)
248 {
249 await InvokeOperationAsync<ExecuteAttribute>(EmptyCriteria.Instance, isSync).ConfigureAwait(false);
250 }
251
252 public async Task DeleteAsync(object criteria, bool isSync)
253 {
254 await InvokeOperationAsync<DeleteAttribute>(criteria, isSync).ConfigureAwait(false);
255 }
256#if NET5_0_OR_GREATER
257
258 private static void OnAssemblyLoadContextUnload(AssemblyLoadContext context)
259 {
260 AssemblyLoadContextManager.RemoveFromCache(_methodNameList, context, true);
261 }
262#endif
263 }
264
265 internal class DataPortalMethodNames
266 {
267 public static readonly DataPortalMethodNames Default = new();
268 public string OnDataPortalInvoke { get; set; } = "DataPortal_OnDataPortalInvoke";
269 public string OnDataPortalInvokeComplete { get; set; } = "DataPortal_OnDataPortalInvokeComplete";
270 public string OnDataPortalException { get; set; } = "DataPortal_OnDataPortalException";
271 }
272}
Enables simple invocation of methods against the contained object using late binding.
object CallMethodIfImplemented(string method)
Uses reflection to dynamically invoke a method if that method is implemented on the target object.
object Instance
Object instance managed by LateBoundObject.
bool IsBusy
Gets a value indicating whether the object, or any of the object's child objects, are busy running an...
Definition: INotifyBusy.cs:29
Defines the common properties required objects that track their own status.
Definition: ITrackStatus.cs:17
@ Default
Default value, rule can run in any context