Skip to content
Merged
Show file tree
Hide file tree
Changes from 42 commits
Commits
Show all changes
48 commits
Select commit Hold shift + click to select a range
a12db87
extend controller base
madsrasmussen Aug 11, 2025
5d6415f
extend controller base
madsrasmussen Aug 11, 2025
05a4235
add package for management api
madsrasmussen Aug 11, 2025
6212027
add signalr as external package
madsrasmussen Aug 11, 2025
2b1be18
connect to server event hub
madsrasmussen Aug 11, 2025
2490bf8
do no act on undefined
madsrasmussen Aug 11, 2025
8c592e3
add event subject
madsrasmussen Aug 11, 2025
c139362
correct alias
madsrasmussen Aug 11, 2025
62799bd
export token
madsrasmussen Aug 11, 2025
5a3f8f9
add helper methods
madsrasmussen Aug 11, 2025
2deb7cc
cache server responses
madsrasmussen Aug 11, 2025
ad257cf
fix import
madsrasmussen Aug 11, 2025
03c1fc8
use helpers
madsrasmussen Aug 11, 2025
1f1baa6
add detail request manager
madsrasmussen Aug 12, 2025
a2c3283
implement for document type
madsrasmussen Aug 12, 2025
9aa73a7
implement for data type
madsrasmussen Aug 12, 2025
9d33c00
add method for update
madsrasmussen Aug 12, 2025
ba732e3
add support for create method
madsrasmussen Aug 12, 2025
9e1deb6
align code
madsrasmussen Aug 12, 2025
cefc5e0
Update detail-request.manager.ts
madsrasmussen Aug 12, 2025
b38cdf2
move explicit naming
madsrasmussen Aug 12, 2025
9905302
move into folder
madsrasmussen Aug 12, 2025
eef5acb
collect server code in folder
madsrasmussen Aug 12, 2025
0045a74
add implementation for data type request manager
madsrasmussen Aug 12, 2025
9058d26
implement for document type
madsrasmussen Aug 12, 2025
c9f701f
only cache when we have connection to the server events
madsrasmussen Aug 13, 2025
da203e6
update
madsrasmussen Aug 14, 2025
de86ac7
fix imports
madsrasmussen Aug 14, 2025
107e6d6
introduce item cache
madsrasmussen Aug 14, 2025
f4af0e3
call trough get items controller
madsrasmussen Aug 15, 2025
feba22b
remove log
madsrasmussen Aug 15, 2025
523e63b
add unit tests for item cache
madsrasmussen Aug 15, 2025
ebdb339
Create cache.test.ts
madsrasmussen Aug 15, 2025
1f181c3
use sync method to lookup data type item
madsrasmussen Aug 15, 2025
82be7df
Merge branch 'main' into v16/feature/entity-detail-runtime-cache
madsrasmussen Aug 15, 2025
2cdfef1
use correct alias
madsrasmussen Aug 15, 2025
c663462
Merge branch 'v16/feature/entity-detail-runtime-cache' into v16/featu…
madsrasmussen Aug 15, 2025
bfa9918
remove unused code
madsrasmussen Aug 15, 2025
2df8103
Merge branch 'main' into v16/feature/item-data-runtime-cache
iOvergaard Aug 18, 2025
aa5c1d6
split detail cache invalidation from request manager
madsrasmussen Aug 18, 2025
729d65b
introduce item cache invalidation manager
madsrasmussen Aug 18, 2025
7847a7f
Merge branch 'main' into v16/feature/item-data-runtime-cache
madsrasmussen Aug 18, 2025
5961dd6
remove unused
madsrasmussen Aug 18, 2025
c29c498
invalidate documents when document types changes
madsrasmussen Aug 18, 2025
e8d3f65
align naming
madsrasmussen Aug 18, 2025
cffd86e
add method to get unique
madsrasmussen Aug 18, 2025
1a6f81d
use server model instead of mapping
madsrasmussen Aug 18, 2025
e74a23a
call method
madsrasmussen Aug 18, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 12 additions & 0 deletions src/Umbraco.Web.UI.Client/src/packages/data-type/entry-point.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
import { UmbManagementApiDataTypeDetailDataCacheInvalidationManager } from './repository/detail/server-data-source/data-type-detail.server.cache-invalidation.manager.js';
import type { UmbEntryPointOnInit, UmbEntryPointOnUnload } from '@umbraco-cms/backoffice/extension-api';

let detailDataCacheInvalidationManager: UmbManagementApiDataTypeDetailDataCacheInvalidationManager | undefined;

export const onInit: UmbEntryPointOnInit = (host) => {
detailDataCacheInvalidationManager = new UmbManagementApiDataTypeDetailDataCacheInvalidationManager(host);
};

export const onUnload: UmbEntryPointOnUnload = () => {
detailDataCacheInvalidationManager?.destroy();
};
6 changes: 6 additions & 0 deletions src/Umbraco.Web.UI.Client/src/packages/data-type/manifests.ts
Original file line number Diff line number Diff line change
Expand Up @@ -21,4 +21,10 @@ export const manifests: Array<UmbExtensionManifest | UmbExtensionManifestKind> =
...searchProviderManifests,
...treeManifests,
...workspaceManifests,
{
name: 'Data Type Backoffice Entry Point',
alias: 'Umb.EntryPoint.DataType',
type: 'backofficeEntryPoint',
js: () => import('./entry-point.js'),
},
];
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
import { dataTypeDetailCache } from './data-type-detail.server.cache.js';
import { UmbManagementApiDetailDataCacheInvalidationManager } from '@umbraco-cms/backoffice/management-api';
import type { UmbControllerHost } from '@umbraco-cms/backoffice/controller-api';
import type { DataTypeResponseModel } from '@umbraco-cms/backoffice/external/backend-api';

export class UmbManagementApiDataTypeDetailDataCacheInvalidationManager extends UmbManagementApiDetailDataCacheInvalidationManager<DataTypeResponseModel> {
constructor(host: UmbControllerHost) {
super(host, {
dataCache: dataTypeDetailCache,
serverEventSources: ['Umbraco:CMS:DataType'],
});
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,6 @@ export class UmbManagementApiDataTypeDetailDataRequestManager extends UmbManagem
update: (id: string, body: UpdateDataTypeRequestModel) => DataTypeService.putDataTypeById({ path: { id }, body }),
delete: (id: string) => DataTypeService.deleteDataTypeById({ path: { id } }),
dataCache: dataTypeDetailCache,
serverEventSource: 'Umbraco:CMS:DataType',
});
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
import { UmbManagementApiDocumentTypeDetailDataCacheInvalidationManager } from './repository/detail/server-data-source/document-type-detail.server.cache-invalidation.manager.js';
import type { UmbEntryPointOnInit, UmbEntryPointOnUnload } from '@umbraco-cms/backoffice/extension-api';

let detailDataCacheInvalidationManager: UmbManagementApiDocumentTypeDetailDataCacheInvalidationManager | undefined;

export const onInit: UmbEntryPointOnInit = (host) => {
detailDataCacheInvalidationManager = new UmbManagementApiDocumentTypeDetailDataCacheInvalidationManager(host);
};

export const onUnload: UmbEntryPointOnUnload = () => {
detailDataCacheInvalidationManager?.destroy();
};
Original file line number Diff line number Diff line change
Expand Up @@ -19,4 +19,10 @@ export const manifests: Array<UmbExtensionManifest | UmbExtensionManifestKind> =
...searchManifests,
...treeManifests,
...workspaceManifests,
{
name: 'Document Type Backoffice Entry Point',
alias: 'Umb.EntryPoint.DocumentType',
type: 'backofficeEntryPoint',
js: () => import('./entry-point.js'),
},
];
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
import { documentTypeDetailCache } from './document-type-detail.server.cache.js';
import { UmbManagementApiDetailDataCacheInvalidationManager } from '@umbraco-cms/backoffice/management-api';
import type { UmbControllerHost } from '@umbraco-cms/backoffice/controller-api';
import type { DocumentTypeResponseModel } from '@umbraco-cms/backoffice/external/backend-api';

export class UmbManagementApiDocumentTypeDetailDataCacheInvalidationManager extends UmbManagementApiDetailDataCacheInvalidationManager<DocumentTypeResponseModel> {
constructor(host: UmbControllerHost) {
super(host, {
dataCache: documentTypeDetailCache,
serverEventSources: ['Umbraco:CMS:DocumentType'],
});
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,6 @@ export class UmbManagementApiDocumentTypeDetailDataRequestManager extends UmbMan
DocumentTypeService.putDocumentTypeById({ path: { id }, body }),
delete: (id: string) => DocumentTypeService.deleteDocumentTypeById({ path: { id } }),
dataCache: documentTypeDetailCache,
serverEventSource: 'Umbraco:CMS:DocumentType',
});
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
import { UmbManagementApiDocumentItemDataCacheInvalidationManager } from './item/repository/document-item.server.cache-invalidation.manager.js';
import type { UmbEntryPointOnInit, UmbEntryPointOnUnload } from '@umbraco-cms/backoffice/extension-api';

let itemDataCacheInvalidationManager: UmbManagementApiDocumentItemDataCacheInvalidationManager | undefined;

export const onInit: UmbEntryPointOnInit = (host) => {
itemDataCacheInvalidationManager = new UmbManagementApiDocumentItemDataCacheInvalidationManager(host);
};

export const onUnload: UmbEntryPointOnUnload = () => {
itemDataCacheInvalidationManager?.destroy();
};
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
import { documentItemCache } from './document-item.server.cache.js';
import { UmbManagementApiItemDataCacheInvalidationManager } from '@umbraco-cms/backoffice/management-api';
import type { UmbControllerHost } from '@umbraco-cms/backoffice/controller-api';
import type { DocumentItemResponseModel } from '@umbraco-cms/backoffice/external/backend-api';

export class UmbManagementApiDocumentItemDataCacheInvalidationManager extends UmbManagementApiItemDataCacheInvalidationManager<DocumentItemResponseModel> {
constructor(host: UmbControllerHost) {
super(host, {
dataCache: documentItemCache,
serverEventSources: ['Umbraco:CMS:Document'],
});
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
import type { DocumentItemResponseModel } from '@umbraco-cms/backoffice/external/backend-api';
import { UmbManagementApiItemDataCache } from '@umbraco-cms/backoffice/management-api';

const documentItemCache = new UmbManagementApiItemDataCache<DocumentItemResponseModel>();

export { documentItemCache };
Original file line number Diff line number Diff line change
@@ -1,10 +1,9 @@
import { UMB_DOCUMENT_ENTITY_TYPE } from '../../entity.js';
import type { UmbDocumentItemModel } from './types.js';
import { UmbManagementApiDocumentItemDataRequestManager } from './document-item.server.request-manager.js';
import type { DocumentItemResponseModel } from '@umbraco-cms/backoffice/external/backend-api';
import { DocumentService } from '@umbraco-cms/backoffice/external/backend-api';
import type { UmbControllerHost } from '@umbraco-cms/backoffice/controller-api';
import { UmbItemServerDataSourceBase } from '@umbraco-cms/backoffice/repository';
import { UmbItemDataApiGetRequestController } from '@umbraco-cms/backoffice/entity-item';

/**
* A data source for Document items that fetches data from the server
Expand All @@ -15,6 +14,8 @@ export class UmbDocumentItemServerDataSource extends UmbItemServerDataSourceBase
DocumentItemResponseModel,
UmbDocumentItemModel
> {
#itemRequestManager = new UmbManagementApiDocumentItemDataRequestManager(this);

/**
* Creates an instance of UmbDocumentItemServerDataSource.
* @param {UmbControllerHost} host - The controller host for this controller to be appended to
Expand All @@ -29,13 +30,7 @@ export class UmbDocumentItemServerDataSource extends UmbItemServerDataSourceBase
override async getItems(uniques: Array<string>) {
if (!uniques) throw new Error('Uniques are missing');

const itemRequestManager = new UmbItemDataApiGetRequestController(this, {
// eslint-disable-next-line local-rules/no-direct-api-import
api: (args) => DocumentService.getItemDocument({ query: { id: args.uniques } }),
uniques,
});

const { data, error } = await itemRequestManager.request();
const { data, error } = await this.#itemRequestManager.getItems(uniques);

return { data: this._getMappedItems(data), error };
}
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
/* eslint-disable local-rules/no-direct-api-import */
import { documentItemCache } from './document-item.server.cache.js';
import type { UmbControllerHost } from '@umbraco-cms/backoffice/controller-api';
import { DocumentService, type DocumentItemResponseModel } from '@umbraco-cms/backoffice/external/backend-api';
import { UmbManagementApiItemDataRequestManager } from '@umbraco-cms/backoffice/management-api';

export class UmbManagementApiDocumentItemDataRequestManager extends UmbManagementApiItemDataRequestManager<DocumentItemResponseModel> {
constructor(host: UmbControllerHost) {
super(host, {
getItems: (ids: Array<string>) => DocumentService.getItemDocument({ query: { id: ids } }),
dataCache: documentItemCache,
serverEventSource: 'Umbraco:CMS:Document',
Comment thread
madsrasmussen marked this conversation as resolved.
Outdated
});
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -42,4 +42,10 @@ export const manifests: Array<UmbExtensionManifest | UmbExtensionManifestKind> =
...urlManifests,
...userPermissionManifests,
...workspaceManifests,
{
name: 'Document Backoffice Entry Point',
alias: 'Umb.BackofficeEntryPoint.Document',
type: 'backofficeEntryPoint',
js: () => import('./entry-point.js'),
},
];
Original file line number Diff line number Diff line change
@@ -0,0 +1,46 @@
import { UMB_MANAGEMENT_API_SERVER_EVENT_CONTEXT } from '../server-event/constants.js';
import type { UmbManagementApiDetailDataCache } from './cache.js';
import type { UmbControllerHost } from '@umbraco-cms/backoffice/controller-api';
import { UmbControllerBase } from '@umbraco-cms/backoffice/class-api';

export interface UmbManagementApiDetailDataInvalidationManagerArgs<DetailResponseModelType> {
dataCache: UmbManagementApiDetailDataCache<DetailResponseModelType>;
serverEventSources: Array<string>;
}

export class UmbManagementApiDetailDataCacheInvalidationManager<DetailResponseModelType> extends UmbControllerBase {
#dataCache: UmbManagementApiDetailDataCache<DetailResponseModelType>;
#serverEventSources: Array<string>;
#serverEventContext?: typeof UMB_MANAGEMENT_API_SERVER_EVENT_CONTEXT.TYPE;

constructor(
host: UmbControllerHost,
args: UmbManagementApiDetailDataInvalidationManagerArgs<DetailResponseModelType>,
) {
super(host);
{
this.#dataCache = args.dataCache;
this.#serverEventSources = args.serverEventSources;

this.consumeContext(UMB_MANAGEMENT_API_SERVER_EVENT_CONTEXT, (context) => {
this.#serverEventContext = context;
this.#observeServerEvents();
});
}
}

#observeServerEvents() {
this.observe(
this.#serverEventContext?.byEventSourcesAndTypes(this.#serverEventSources, ['Updated', 'Deleted']),
(event) => {
if (!event) return;
this.#dataCache.delete(event.key);
},
'umbObserveServerEvents',
);
}

override destroy(): void {
this.#dataCache.clear();
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,6 @@ export interface UmbManagementApiDetailDataRequestManagerArgs<
update: (id: string, data: UpdateRequestModelType) => Promise<UmbApiResponse<{ data: unknown }>>;
delete: (id: string) => Promise<UmbApiResponse<{ data: unknown }>>;
dataCache: UmbManagementApiDetailDataCache<DetailResponseModelType>;
serverEventSource: string;
}

export class UmbManagementApiDetailDataRequestManager<
Expand All @@ -29,13 +28,12 @@ export class UmbManagementApiDetailDataRequestManager<
UpdateRequestModelType,
> extends UmbControllerBase {
#dataCache: UmbManagementApiDetailDataCache<DetailResponseModelType>;
#serverEventSource: string;
#serverEventContext?: typeof UMB_MANAGEMENT_API_SERVER_EVENT_CONTEXT.TYPE;

#create;
#read;
#update;
#delete;
#serverEventContext?: typeof UMB_MANAGEMENT_API_SERVER_EVENT_CONTEXT.TYPE;
#isConnectedToServerEvents = false;

constructor(
Expand All @@ -54,7 +52,6 @@ export class UmbManagementApiDetailDataRequestManager<
this.#delete = args.delete;

this.#dataCache = args.dataCache;
this.#serverEventSource = args.serverEventSource;

this.consumeContext(UMB_MANAGEMENT_API_SERVER_EVENT_CONTEXT, (context) => {
this.#serverEventContext = context;
Expand Down Expand Up @@ -130,15 +127,5 @@ export class UmbManagementApiDetailDataRequestManager<
},
'umbObserveServerEventsConnection',
);

// Invalidate cache entries when entities are updated or deleted
this.observe(
this.#serverEventContext?.byEventSourceAndTypes(this.#serverEventSource, ['Updated', 'Deleted']),
(event) => {
if (!event) return;
this.#dataCache.delete(event.key);
},
'umbObserveServerEvents',
);
}
}
Original file line number Diff line number Diff line change
@@ -1,2 +1,3 @@
export * from './detail-data.request-manager.js';
export * from './cache.js';
export * from './cache-invalidation.manager.js';
Original file line number Diff line number Diff line change
@@ -1,2 +1,3 @@
export * from './detail/index.js';
export * from './item/index.js';
export * from './server-event/constants.js';
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
import { UMB_MANAGEMENT_API_SERVER_EVENT_CONTEXT } from '../server-event/constants.js';
import type { UmbManagementApiItemDataCache } from './cache.js';
import type { UmbControllerHost } from '@umbraco-cms/backoffice/controller-api';
import { UmbControllerBase } from '@umbraco-cms/backoffice/class-api';

export interface UmbManagementApiItemDataInvalidationManagerArgs<ItemResponseModelType> {
dataCache: UmbManagementApiItemDataCache<ItemResponseModelType>;
serverEventSources: Array<string>;
}

export class UmbManagementApiItemDataCacheInvalidationManager<ItemResponseModelType> extends UmbControllerBase {
#dataCache: UmbManagementApiItemDataCache<ItemResponseModelType>;
#serverEventSources: Array<string>;
#serverEventContext?: typeof UMB_MANAGEMENT_API_SERVER_EVENT_CONTEXT.TYPE;

constructor(host: UmbControllerHost, args: UmbManagementApiItemDataInvalidationManagerArgs<ItemResponseModelType>) {
super(host);
{
this.#dataCache = args.dataCache;
this.#serverEventSources = args.serverEventSources;

this.consumeContext(UMB_MANAGEMENT_API_SERVER_EVENT_CONTEXT, (context) => {
this.#serverEventContext = context;
this.#observeServerEvents();
});
}
}

#observeServerEvents() {
// Invalidate cache entries when entities are updated or deleted
this.observe(
this.#serverEventContext?.byEventSourcesAndTypes(this.#serverEventSources, ['Updated', 'Deleted']),
(event) => {
if (!event) return;
this.#dataCache.delete(event.key);
},
'umbObserveServerEvents',
);
}

override destroy(): void {
this.#dataCache.clear();
}
}
Loading
Loading