Efficient ViewModel fetch
No edit summary
m ((username removed) (log details removed))
 
(8 intermediate revisions by 2 users not shown)
Line 1: Line 1:
ViewModels are good for the efficient fetching of data since they declaratively explain what data will be used. This enables MDriven to scan through the expressions and fetch data with fewer queries.
ViewModels are useful for the efficient fetching of data since they declaratively explain what data will be used. This enables MDriven to scan through the expressions and fetch data with fewer queries.


('''Note''': Most of the things described below are now automatically handled by the [[QueryPlan|QueryPlanner]] and you don't need to create fetch hints.
('''NOTE''': Most of the things described below are ''now automatically handled'' by the [[QueryPlan|QueryPlanner]] and you don't need to create fetch hints.)


If you have a root of Something that has a list of details and the detail, in turn, fetches even more details, you could easily end up with x*y*z queries to the database.
If you have a root of something that has a list of details and the details, in turn, fetch even more details, you could easily end up with x*y*z queries to the database.


Since the ViewModel uses expressions that may be stackable, we can easily fetch all x, all x.y, and all x.y.z with 3 queries.
Since the ViewModel uses expressions that may be stackable, we can easily fetch all x, all x.y, and all x.y.z with 3 queries. This is standard behavior when using the MDriven ViewModels.
* When you use Seekers, the result is collected in a collection variable commonly named vSeekerResult.
The search result is not covered by the standard efficient fetch algorithm because this data is not available at load time. This may, in certain situations, be less than ideal; so, we have added strategies to improve the efficient fetching in ViewModels using Seekers – this article explains how.


This is the standard behavior when using the MDriven ViewModels.
All the ideas implemented and explained below came from a real case where I had a fairly complex UI to visualize and manage demand forecasts for articles:
 
When you use seekers, the result is collected in a collection variable commonly named vSeekerResult.
 
The search result is not covered by the standard efficient fetch algorithm because this data is not available at load time. This may in certain situations be less than ideal so we have added strategies to improve the efficient fetching in ViewModels using seekers – this article explains how.
 
All the ideas implemented and explained below came from a real case where I had a fairly complex UI to visualize and manage demand forecast for articles


[[File:Huge viewmodel.png|frameless|675x675px]]
[[File:Huge viewmodel.png|frameless|675x675px]]


There are thousands of objects shown in this one single image – it is a seeker, but it can also get data from a list tied to current user that is called Favorites.
There are thousands of objects shown in this one single image. It is a Seeker, but it can also get data from a list tied to the current user that is called Favorites.


Even with the normal Efficient fetch turned on this UI typically spewed out up to 900 questions to the persistence server.
Even with the normal Efficient fetch turned on, this UI typically spewed out up to 900 questions to the Persistence Server.


To fully understand where the questions came from I used the debugger:
To fully understand where the questions came from, I used the Debugger:


[[File:OclDebuggerManyPMappercalls.png|frameless|696x696px]]
[[File:OclDebuggerManyPMappercalls.png|frameless|696x696px]]
Line 29: Line 25:
The ones that we look for are the ones that fetch only 1 object – these are candidates for improvement.
The ones that we look for are the ones that fetch only 1 object – these are candidates for improvement.


When the log sees that only 1 object is fetched it also shows the ClassId.
When the log sees that only 1 object is fetched, it also shows the ClassId.


You can look up classid to name in the debugger:
You can look up ClassId to name in the Debugger:


[[File:OclDebuggerFindClass.png|frameless|691x691px]]
[[File:OclDebuggerFindClass.png|frameless|691x691px]]


Once we know what it is that create the queries we can think of ways to fix it.
Once we know what created the queries, we can think of ways to fix it.


The new strategy to further improve the efficient fetch is to add a ViewModel Nesting with fetching hints. Any ViewModel nesting with a name that starts with “FetchHints” will be found and the columns within will be executed on any list that seeker logic delivers.
The new strategy to improve the efficient fetch further is to add a ViewModel Nesting with fetching hints. Any ViewModel nesting with a name that starts with “FetchHints” will be found, and the columns within will be executed on any list that seeker logic delivers.


Like this:
Like this:
Line 43: Line 39:
[[File:FetchHints example.png|frameless|633x633px]]
[[File:FetchHints example.png|frameless|633x633px]]


Working like this I could reduce the load from this UI from 900 queries to 30 something.
Working like this, I could reduce the load from this UI from 900 queries to 30-something.


Now the problem with search was solved but when the data came from the other source – the user favorites – this logic is not applied.
The problem with the search was solved, but when the data came from the other source – the user favorites – this logic is not applied.


I had to devise a way to allow for me to tell the ViewModel that this logic should execute on a list of objects.
I had to devise a way to allow me to tell the ViewModel that this logic should execute on a list of objects.


To solve this a new standard variable was introduced in ViewModels: selfVM
To solve this, a new standard variable was introduced in ViewModels: <code>selfVM</code>
 
* The <code>selfVM</code> variable is a reference to the ViewModel we work on – self is the object context as before – but <code>selfVM</code> is the ViewModel holding the objects.
The selfVM variable is a reference to the ViewModel we work on – self is the object context as before – but selfVM is the ViewModel holding the objects.
I think that this is a good extension that may solve some limitations we have had before.
 
* The <code>selfVM</code> introduces 3 new operations: <code>ExecuteFetchHints</code>, <code>Save</code>, and <code>ExecuteAction</code>.
I think that this is  a good extension that may solve some limitations we have had before.
* The <code>ExecuteFetchHints</code> is the one I was after in this case. <code>Save</code> and <code>ExecuteAction</code> was added since there has been a demand for programmatic access to these functions.
 
My code for showing the users' favorites in the UI above was extended like this:
the selfVM introduce 3 new operations: ExecuteFetchHints, Save, ExecuteAction.
 
The ExecuteFetchHints is obviously the one I was after in this case. Save and ExecuteAction was added since there has been a demand for programmatic access to these functions.
 
My code for showing the users favorites in the UI above was extended like this:


  vSeekerResult->Clear;
  vSeekerResult->Clear;
Line 73: Line 64:
   )
   )
  );
  );
  <code>selfVM.ExecuteFetchHints(vSeekerResult)</code>
  selfVM.ExecuteFetchHints(vSeekerResult)  
And then the same efficient fetching could be applied to the data coming this way as well.
Then, the same efficient fetching could be applied to the data coming this way as well.
 
Also see: [[How to use the ExecutePS function in selfVM]]


Also, see: [[How to use the ExecutePS function in selfVM]]
The MDriven Book - See: [[Introducing MDriven Turnkey]]
[[Category:View Model]]
[[Category:View Model]]
[[Category:Performance]]
[[Category:Optimization]]
[[Category:OCL]]
[[Category:OCL]]
[[Category:Advanced]]
[[Category:Advanced]]

Latest revision as of 06:29, 11 January 2024

ViewModels are useful for the efficient fetching of data since they declaratively explain what data will be used. This enables MDriven to scan through the expressions and fetch data with fewer queries.

(NOTE: Most of the things described below are now automatically handled by the QueryPlanner and you don't need to create fetch hints.)

If you have a root of something that has a list of details and the details, in turn, fetch even more details, you could easily end up with x*y*z queries to the database.

Since the ViewModel uses expressions that may be stackable, we can easily fetch all x, all x.y, and all x.y.z with 3 queries. This is standard behavior when using the MDriven ViewModels.

  • When you use Seekers, the result is collected in a collection variable commonly named vSeekerResult.

The search result is not covered by the standard efficient fetch algorithm because this data is not available at load time. This may, in certain situations, be less than ideal; so, we have added strategies to improve the efficient fetching in ViewModels using Seekers – this article explains how.

All the ideas implemented and explained below came from a real case where I had a fairly complex UI to visualize and manage demand forecasts for articles:

Huge viewmodel.png

There are thousands of objects shown in this one single image. It is a Seeker, but it can also get data from a list tied to the current user that is called Favorites.

Even with the normal Efficient fetch turned on, this UI typically spewed out up to 900 questions to the Persistence Server.

To fully understand where the questions came from, I used the Debugger:

OclDebuggerManyPMappercalls.png

Check the PMapper and you will see all the PMapper calls.

The ones that we look for are the ones that fetch only 1 object – these are candidates for improvement.

When the log sees that only 1 object is fetched, it also shows the ClassId.

You can look up ClassId to name in the Debugger:

OclDebuggerFindClass.png

Once we know what created the queries, we can think of ways to fix it.

The new strategy to improve the efficient fetch further is to add a ViewModel Nesting with fetching hints. Any ViewModel nesting with a name that starts with “FetchHints” will be found, and the columns within will be executed on any list that seeker logic delivers.

Like this:

FetchHints example.png

Working like this, I could reduce the load from this UI from 900 queries to 30-something.

The problem with the search was solved, but when the data came from the other source – the user favorites – this logic is not applied.

I had to devise a way to allow me to tell the ViewModel that this logic should execute on a list of objects.

To solve this, a new standard variable was introduced in ViewModels: selfVM

  • The selfVM variable is a reference to the ViewModel we work on – self is the object context as before – but selfVM is the ViewModel holding the objects.

I think that this is a good extension that may solve some limitations we have had before.

  • The selfVM introduces 3 new operations: ExecuteFetchHints, Save, and ExecuteAction.
  • The ExecuteFetchHints is the one I was after in this case. Save and ExecuteAction was added since there has been a demand for programmatic access to these functions.

My code for showing the users' favorites in the UI above was extended like this:

vSeekerResult->Clear;
Singleton.oclSingleton.User.FavoriteArticle->select(fa|fa.Filter.SqlLike(vFavFilter+’%’) or vFavFilter.IsNullOrEmpty).FavoriteArticles->collect(a|
— Add the SA for the users country
let sa=a.SalesArticles->select(sa|sa.MarketingCompany=self)->first in
  (
     if sa.isnull then
        vSeekerResult
     else
        vSeekerResult.Add(sa)
     endif    
  )
);
selfVM.ExecuteFetchHints(vSeekerResult) 

Then, the same efficient fetching could be applied to the data coming this way as well.

Also see: How to use the ExecutePS function in selfVM

The MDriven Book - See: Introducing MDriven Turnkey

This page was edited more than 1 years ago on 01/11/2024. What links here