In many scenarios, users expect to see their rendered data update implicitly as server data changes. This is fairly straightforward to accomplish by configuring your QueryCache/EntityCache to include 'polling'.
public robotsQuery = new MsPortalFx.Data.QueryCache<Robot, any>({
entityTypeName: RobotMetadata.name,
sourceUri: () => RobotData._apiRoot,
poll: true,
});
Additionally, the extension can customize the polling interval by using the 'pollingInterval
' option. By default, the polling interval is 60 seconds. It can be customized down to a minimum of 10 seconds. The minimum is enforced to avoid server load that might result from some accidental, overly aggressive change to 'pollingInterval
' by an extension developer. (There have been cases where even this 10 second minumum has caused customer impact due to increased server load.)
For Azure Portal UI to be responsive, it is often important - when data changes - to avoid rerendering entire Blades and Parts. Rather, in most cases, it is better to make granular data changes so that FX controls and Knockout HTML templates can rerender only small portions of Blade/Part UI. In many common cases of refreshing data, the newly-loaded server data precisely matches already-cached data, and in this case no UI rerendering happens.
When QueryCache/EntityCache data is refreshed - either implicitly as above or explicitly as described here - newly-loaded server data is added to already-cached, client-side data through a process called "data merging":
- The newly-loaded server data is compared to already-cached data
- Differences between newly-loaded and already-cached data are detected. For instance, "property on object changed value" and "the Nth item in array was removed".
- The differences are applied to the already-cached data, via changes to Knockout observables.
For many scenarios, "data merging" requires no configuration and is simply an implementation detail of 'polling' and explicitly requested 'refresh
'. In some scenarios, there are gotcha's to look out for...
When detecting changes between items in an already-loaded array and a newly-loaded array, the "data merging" algorithm requires some per-array configuration. Specifically, the "data merging" algorithm - without configuration - doesn't know how to match items between the old and new arrays. Without configuration, the algorithm considers each already-cached array item as 'removed' (since it matches no item in the newly-loaded array) and every newly-loaded array item as 'added' (since it matches no item in the already-cached array). This effectively replaces the entire cached array's contents, even in those cases where the server data hasn't changed. This can often be the cause of performance problems in your Blade/Part (e.g., poor responsiveness), even while users - sadly - see no pixel-level UI problems.
To proactively warn you of these potential performance problems, the "data merge" algorithm will log warnings to the console that resemble:
Base.Diagnostics.js:351 [Microsoft_Azure_FooBar] 18:55:54
MsPortalFx/Data/Data.DataSet Data.DataSet: Data of type [No type specified] is being merged without identity because the type has no metadata. Please supply metadata for this type.
Any array cached in your QueryCache/EntityCache must be configured for "data merging" by using type metadata. Specifically, for each Array<T>
, the extension has to supply type metadata for type T
that describes the "id" properties for that type (see examples of this here). With this "id" metadata, the "data merging" algorithm can match already-cached and newly-loaded array items and can merge these in-place (with no per-item array remove/add). With this, when the server data doesn't change, each cached array item will match a newly-loaded server item, each will merge in-place with no detected chagnes, and the merge will be a no-op for the purposes of UI rerendering.
As "data merging" proceeds, differences are applied to the already-cached data via Knockout observable changes. When these observables are changed, Knockout subscriptions are notified and Knockout reactors
and computeds
are reevaluated. Any associated (often extension-authored) callback here can throw an exception and this will halt/preempt the current "data merge" cycle. When this happens, the "data merging" algorithm issues an error resembling:
Data merge failed for data set 'FooBarDataSet'. The error message was: ...
Importantly, this error is not due to bugs in the data-merging algorithm. Rather, some JavaScript code (frequently extension code) is causing an exception to be thrown. This error should be accompanied with a JavaScript stack trace that extension developers can use to isolate and fix such bugs.
Occasionally, extension developers are surprised that the EntityView 'item
' observable does not change (doesn't notify subscribers) when the EntityView/EntityCache is refreshed (either implicitly or explicitly) and, consequently, code like this doesn't work as expected:
entityView.item.subscribe(lifetime, () => {
const item = entityView.item();
if (item) {
// Do something with 'newItem' after refresh.
doSomething(item.customerName());
}
});
This is because the "data merge" algorithm doesn't replace already-cached objects (unless such an object is an array item). Rather, objects are merged in-place, to optimize for limited/no UI re-rendering when data changes (see here for details).
A better coding pattern to follow is to use ko.reactor
and ko.computed
as below:
ko.reactor(lifetime, () => {
const item = entityView.item();
if (item) {
// Do something with 'newItem' after refresh.
doSomething(item.customerName());
}
});
With this, the supplied callback will be called both when the 'item
' observable changes (when the data first loads) and also when any properties on the entity change (like 'customerName
' above).
As server data changes, there are scenario where the extension should take explicit steps to keep their QueryCache and EntityCache data consistent with the server. In terms of UX, explicit refreshing of a QueryCache/EntityCache is necessary in scenarios like:
- User makes server changes - User initiates some action and, as a consequence, the extension issues an AJAX call that changes server data. As a best-practice, this AJAX call is typically issued from an extension DataContext.
public updateRobot(robot: Robot): Promise<any> {
return FxBaseNet.ajax({
uri: RobotData._apiRoot + robot.name(),
type: "PUT",
contentType: "application/json",
data: ko.toJSON(robot),
}).then(() => {
// This will refresh the set of data that is available in the underlying data cache.
this.robotsQuery.refreshAll();
});
}
In this scenario, since the AJAX call will be issued from a DataContext, refreshing data in QueryCaches/EntityCaches will be done using QueryCache/EntityCache methods directly. See "Refreshing/updating a QueryCache/EntityCache" below.
- User clicks 'Refresh' command - (Less common) The user clicks on some 'Refresh'-like command on a Blade or Part, implemented in that Blade/Part's view model.
class RefreshCommand implements MsPortalFx.ViewModels.Commands.Command<void> {
private _websiteView: MsPortalFx.Data.EntityView<Website>;
public canExecute: KnockoutObservableBase<boolean>;
constructor(websiteView: MsPortalFx.Data.EntityView<Website>) {
this.canExecute = ko.computed(() => {
return !websiteView.loading();
});
this._websiteView = websiteView;
}
public execute(): Promise<any> {
return this._websiteView.refresh();
}
In this scenario, since the data being refreshed is that data rendered in a specific Blade/Part, refreshing data in the associated QueryCache/EntityCache is best done in terms of QueryView/EntityView methods (for that QueryView/EntityView in use by the Blade/Part view model). See "Refreshing a QueryView/EntityView" below.
There are a number of methods available on QueryCache/EntityCache that make it straightforward and efficient to keep client-side cached data consistent with server data. Which of these is appropriate varies by scenario, and these are discussed individually below.
To understand the design behind this collection of methods and how to select the appropriate method to use, it's important to understand a bit about the DataCache/DataView design (in detail here). Specifically, QueryCache/EntityCache is designed as a collection of cache entries. In some cases, where there are multiple active Blades and Parts, a given QueryCache/EntityCache might contain many cache entries. So, below, you'll see 'refreshAll
' - which issues N AJAX calls to refresh all N entries of a QueryCache/EntityCache - as well as alternative, per-cache-entry methods that allow for more granular, often more efficient refreshing of QueryCache/EntityCache data.
As mentioned above, this method will issue an AJAX call (either using the 'supplyData
' or 'sourceUri
' option supplied to the QueryCache/EntityCache) for each entry currently held in the QueryCache/EntityCache. Upon completion, each AJAX result is merged onto its corresponding cache entry.
public updateRobot(robot: Robot): Promise<any> {
return FxBaseNet.ajax({
uri: RobotData._apiRoot + robot.name(),
type: "PUT",
contentType: "application/json",
data: ko.toJSON(robot),
}).then(() => {
// This will refresh the set of data that is available in the underlying data cache.
this.robotsQuery.refreshAll();
});
}
If the (optional) 'predicate
' parameter is supplied to the 'refreshAll
' call, then only those entries for which the predicate returns 'true' will be refreshed. This 'predicate
' feature is useful when the extension undestands the nature of the server data changes and can - based on this knowledge - chose to not refresh QueryCache/EntityCache entries whose server data hasn't changed.
The 'refresh
' method is useful when the server data changes are known to be specific to a single cache entry (a single query in the case of QueryCache, a single entity 'id' in the case of EntityCache).
const promises: Q.Promise<void>[] = [];
this.enginesQuery.refresh({}, null);
MsPortalFx.makeArray(engines).forEach((engine) => {
promises.push(Q(this.engineEntities.refresh(engine, null)));
});
return Q.all(promises);
public updateEngine(engine: EngineModel): Promise<void> {
let promise: Promise<any>;
if (useFrameworkPortal) {
// Using framework portal (NOTE: this is not allowed against ARM).
// NOTE: do NOT use invoke API since it doesn't handle CORS.
promise = FxBaseNet.ajaxExtended<any>({
headers: { accept: applicationJson },
isBackgroundTask: false,
setAuthorizationHeader: true,
setTelemetryHeader: "Update" + entityType,
type: "PATCH",
uri: EngineData._apiRoot + "&api-version=" + entityVersion,
data: ko.toJSON(convertToResource(engine)),
contentType: applicationJson,
useFxArmEndpoint: true,
});
} else {
// Using local controller.
promise = FxBaseNet.ajax({
type: "PATCH",
uri: EngineData._apiRoot + "?id=" + engine.id(),
data: ko.toJSON(convertToArmResource(engine)),
contentType: applicationJson,
});
}
return promise.then(() => {
if (useFrameworkPortal) {
// This will refresh the set of data that is available in the underlying data cache.
EngineData._debouncer.execute([this._getEngineId(engine)]);
} else {
// This will refresh the set of data that is available in the underlying data cache.
// The {} params let the cache know to re-fetch any data that matches these parameters.
// In the case of this contrived scenario, we always fetch all data. In the future we
// will add a way to refresh all (or selective) caches for a given type. The second param
// manages lifetime, which is not needed in this case.
this.enginesQuery.refresh({}, null);
}
});
}
Using 'refresh
', only a single AJAX call will be issued to the server.
In some scenarios, AJAX calls to the server to refresh cached data can be avoided entirely. For instance, the user may have fully described the server data changes by filling out a Form on a Form Blade. In this case, the necessary QueryCache/EntityCache changes are known by the extension directly without having to make an AJAX call to their server. This use of 'applyChanges
' can be a nice optimization to avoid some AJAX traffic to your servers.
Example - Adding an item to a QueryCache entry
public createRobot(robot: Robot): Promise<any> {
return FxBaseNet.ajax({
uri: RobotData._apiRoot,
type: "POST",
contentType: "application/json",
data: ko.toJSON(robot),
}).then(() => {
// This will refresh the set of data that is displayed to the client by applying the change we made to
// each data set in the cache.
// For this particular example, there is only one data set in the cache.
// This function is executed on each data set selected by the query params.
// params: any The query params
// dataSet: MsPortalFx.Data.DataSet The dataset to modify
this.robotsQuery.applyChanges((_ /* params */, dataSet) => {
// Duplicates on the client the same modification to the datacache which has occurred on the server.
// In this case, we created a robot in the ca, so we will reflect this change on the client side.
dataSet.addItems(0, [robot]);
});
});
}
Example - Removing an item to a QueryCache entry
public deleteRobot(robot: Robot): Promise<any> {
return FxBaseNet.ajax({
uri: RobotData._apiRoot + robot.name(),
type: "DELETE",
}).then(() => {
// This will notify the shell that the robot is being removed.
MsPortalFx.UI.AssetManager.notifyAssetDeleted(ExtensionDefinition.AssetTypeNames.robot, robot.name());
// This will refresh the set of data that is displayed to the client by applying the change we made to
// each data set in the cache.
// For this particular example, there is only one data set in the cache.
// This function is executed on each data set selected by the query params.
// params: any The query params
// dataSet: MsPortalFx.Data.DataSet The dataset to modify
this.robotsQuery.applyChanges((_ /* params */, dataSet) => {
// Duplicates on the client the same modification to the datacache which has occurred on the server.
// In this case, we deleted a robot in the cache, so we will reflect this change on the client side.
dataSet.removeItem(robot);
});
});
}
Similar to 'refreshAll
', the 'applyChanges
' method accepts a function that is called for each cache entry currently in the QueryCache/EntityCache, allowing the extension to update only those cache entries known to be effected by the server changes made by the user.
A subtlety of QueryCache/EntityCache is that it can hold onto cache entries for some time after the last Blade/Part has been closed/unpinned. This design supports the common scenario where a user closes a Blade (for instance) and immediately reopens it.
Now, when the server data for a given cache entry has been entirely deleted, then the extension will want to forcibly remove corresponding entries from their QueryCache (less common) and EntityCache (more common). The 'forceRemove
' method does just this.
public deleteComputer(computer: Computer): Promise<any> {
return FxBaseNet.ajax({
uri: ComputerData._apiRoot + computer.name(),
type: "DELETE",
}).then(() => {
// This will notify the shell that the computer is being removed.
MsPortalFx.UI.AssetManager.notifyAssetDeleted(ExtensionDefinition.AssetTypeNames.computer, computer.name());
// This will refresh the set of data that is displayed to the client by applying the change we made to
// each data set in the cache.
// For this particular example, there is only one data set in the cache.
// This function is executed on each data set selected by the query params.
// params: any The query params
// dataSet: MsPortalFx.Data.DataSet The dataset to modify
this.computersQuery.applyChanges((_ /* params */, dataSet) => {
// Duplicates on the client the same modification to the datacache which has occurred on the server.
// In this case, we deleted a computer in the cache, so we will reflect this change on the client side.
dataSet.removeItem(computer);
});
// This will force the removal of the deleted computer from this EntityCache. Subsequently, any Part or
// Blades that use an EntityView to fetch this deleted computer will likely receive an expected 404
// response.
this.computerEntities.forceRemove(computer.name());
});
}
Once called, the corresponding cache entry will be removed. If the user were to - somehow - open a Blade or drag/drop a Part that tried to load the deleted data, the QueryCache/EntityCache would try to create an entirely new cache entry, and - presumably - it would fail to load the corresponding server data. In such a case, by design, the user would see a 'data not found' user experience in that Blade/Part.
When using 'forceRemove
', the extension will also - typically - want to take steps to ensure that any existing Blades/Parts are no longer making use of the removed cache entry (via QueryView/EntityView). When the extension notifies the FX of a deleted ARM resource via 'MsPortalFx.UI.AssetManager.notifyResourceDeleted()
in Knockout blades or notifyResourceDeleted()
from @microsoft/azureportal-reactview/ResourceManagement
in React Views' (see here for details), the FX will automatically show 'deleted' UX in any corresponding Blades/Parts. If the user clicked some 'Delete'-style command on a Blade to trigger the 'forceRemove
', often the extension will elect to programmatically close the Blade with the 'Delete' command (in addition to making associated AJAX and 'forceRemove
' calls from their DataContext).
In some Blades/Parts, there can be a specific 'Refresh' command that is meant to refresh only that data visible in the given Blade/Part. In this scenario, it is the QueryView/EntityView that serves as a reference/pointer to that Blade/Part's data, and it's with that QueryView/EntityView that the extension should refresh the data. (See a sample here).
When 'refresh
' is called, a Promise is returned that reflects the progress/completion of the corresponding AJAX call. In addition, the 'isLoading
' observable property of QueryView/EntityView also changes to 'true' (useful for controlling UI indications that the refresh is in progress, like temporarily disabling the clicked 'Refresh' command).
At the QueryCache/EntityCache level, in response to the QueryView/EntityView 'refresh
' call, an AJAX call will be issued for the corresponding QueryCache/EntityCache cache entry, precisely in the same manner that would happen if the extension called QueryCache/EntityCache's 'refresh
' method with the associated cache key (see here).
There is one subtety to the 'refresh
' method available on QueryView/EntityView that sometimes trips up extension developers. You will notice that 'refresh
' accepts no parameters. This is because 'refresh
' was designed to refresh already-loaded data. An initial call to QueryView/EntityView's 'fetch
' method establishes a cache entry in the corresponding QueryCache/EntityCache that includes the URL information with which to issue an AJAX call when 'refresh
' is later called.
Sometimes, extensions developers feel it is important - when a Blade opens or a Part is first displayed - to implicitly refresh the Blade's/Part's data (in cases where the data may have already been loaded/cached for use in some previously viewed Blade/Part). To make this so, they'll sometimes call QueryView/EntityView's 'fetch
' and 'refresh
' methods in succession. This is a minor anti-pattern that should probably be avoided. This "refresh my data on Blade open" pattern trains the user to open Blades to fix stale data (even close and then immediately reopen the same Blade) and can often be a symptom of a missing 'Refresh' command or auto-refreshing data that's configured with too long a polling interval.