From 92546849f8283ae5ca36d2982c63530eff0b9b8b Mon Sep 17 00:00:00 2001 From: DenisSinelnikov <142215442+DenisSinelnikov@users.noreply.github.com> Date: Tue, 13 Aug 2024 15:16:56 +0400 Subject: [PATCH] =?UTF-8?q?CB-5123.=20Added=20backend=20part,=20added=20ne?= =?UTF-8?q?w=20parameter=20licenseStatus=20to=20ser=E2=80=A6=20(#2831)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * CB-5123. Added backend part, added new parameter licenseStatus to server config gql * CB-5123 feat: display license status information * CB-5123 fix: after review * CB-5123. Added version --------- Co-authored-by: Aleksei Potsetsuev Co-authored-by: Evgenia Bezborodova <139753579+EvgeniaBzzz@users.noreply.github.com> --- .../schema/service.core.graphqls | 1 + .../io/cloudbeaver/model/WebServerConfig.java | 5 + .../AdministrationScreenService.ts | 2 +- .../src/PasswordPolicyService.ts | 15 ++- .../core-blocks/src/usePasswordValidation.ts | 3 + .../core-events/src/NotificationService.ts | 53 +++++---- .../core-projects/src/ProjectInfoResource.ts | 6 +- .../src/DefaultNavigatorSettingsResource.ts | 103 +++++++++++++++++ .../core-root/src/PasswordPolicyResource.ts | 31 +++++ .../core-root/src/PermissionsService.ts | 25 ++++- .../core-root/src/ProductInfoResource.ts | 31 +++++ .../packages/core-root/src/QuotasService.ts | 9 +- .../core-root/src/ServerConfigResource.ts | 106 ++---------------- .../src/ServerLicenseStatusResource.ts | 39 +++++++ .../src/ServerResourceQuotasResource.ts | 31 +++++ webapp/packages/core-root/src/index.ts | 5 + webapp/packages/core-root/src/manifest.ts | 5 + .../ServerConfig/DefaultNavigatorSettings.gql | 5 + .../fragments/ServerConfig/PasswordPolicy.gql | 8 ++ .../fragments/ServerConfig/ProductInfo.gql | 12 ++ .../fragments/ServerConfig/ResourceQuotas.gql | 3 + .../fragments/ServerConfig/ServerConfig.gql | 35 ++++++ .../ServerConfig/ServerLicenseStatus.gql | 5 + .../session/getDefaultNavigatorSettings.gql | 5 + .../src/queries/session/getPasswordPolicy.gql | 5 + .../src/queries/session/getProductInfo.gql | 5 + .../session/getServerLicenseStatus.gql | 5 + .../session/getServerResourceQuotas.gql | 5 + .../src/queries/session/serverConfig.gql | 60 +--------- .../core-version/src/VersionResource.ts | 8 +- .../ServerConfigurationPage.tsx | 5 +- .../ServerConfigurationService.ts | 22 +++- ...verConfigurationAuthenticationBootstrap.ts | 4 +- .../ResultSet/ResultSetDataContentAction.ts | 17 ++- .../plugin-product/src/ProductBootstrap.ts | 6 +- .../plugin-product/src/ProductInfoDialog.tsx | 6 +- .../plugin-top-app-bar/src/TopNavBar/Logo.tsx | 8 +- 37 files changed, 487 insertions(+), 212 deletions(-) create mode 100644 webapp/packages/core-root/src/DefaultNavigatorSettingsResource.ts create mode 100644 webapp/packages/core-root/src/PasswordPolicyResource.ts create mode 100644 webapp/packages/core-root/src/ProductInfoResource.ts create mode 100644 webapp/packages/core-root/src/ServerLicenseStatusResource.ts create mode 100644 webapp/packages/core-root/src/ServerResourceQuotasResource.ts create mode 100644 webapp/packages/core-sdk/src/queries/fragments/ServerConfig/DefaultNavigatorSettings.gql create mode 100644 webapp/packages/core-sdk/src/queries/fragments/ServerConfig/PasswordPolicy.gql create mode 100644 webapp/packages/core-sdk/src/queries/fragments/ServerConfig/ProductInfo.gql create mode 100644 webapp/packages/core-sdk/src/queries/fragments/ServerConfig/ResourceQuotas.gql create mode 100644 webapp/packages/core-sdk/src/queries/fragments/ServerConfig/ServerConfig.gql create mode 100644 webapp/packages/core-sdk/src/queries/fragments/ServerConfig/ServerLicenseStatus.gql create mode 100644 webapp/packages/core-sdk/src/queries/session/getDefaultNavigatorSettings.gql create mode 100644 webapp/packages/core-sdk/src/queries/session/getPasswordPolicy.gql create mode 100644 webapp/packages/core-sdk/src/queries/session/getProductInfo.gql create mode 100644 webapp/packages/core-sdk/src/queries/session/getServerLicenseStatus.gql create mode 100644 webapp/packages/core-sdk/src/queries/session/getServerResourceQuotas.gql diff --git a/server/bundles/io.cloudbeaver.server/schema/service.core.graphqls b/server/bundles/io.cloudbeaver.server/schema/service.core.graphqls index 7bdc0106e7..e89cd4bbfa 100644 --- a/server/bundles/io.cloudbeaver.server/schema/service.core.graphqls +++ b/server/bundles/io.cloudbeaver.server/schema/service.core.graphqls @@ -143,6 +143,7 @@ type ServerConfig { licenseRequired: Boolean! licenseValid: Boolean! + licenseStatus: String @since(version: "24.1.5") sessionExpireTime: Int! localHostAddress: String diff --git a/server/bundles/io.cloudbeaver.server/src/io/cloudbeaver/model/WebServerConfig.java b/server/bundles/io.cloudbeaver.server/src/io/cloudbeaver/model/WebServerConfig.java index d4f09a9ce6..826331eea4 100644 --- a/server/bundles/io.cloudbeaver.server/src/io/cloudbeaver/model/WebServerConfig.java +++ b/server/bundles/io.cloudbeaver.server/src/io/cloudbeaver/model/WebServerConfig.java @@ -121,6 +121,11 @@ public boolean isLicenseValid() { return application.isLicenseValid(); } + @Property + public String getLicenseStatus() { + return application.getLicenseStatus(); + } + @Property public boolean isConfigurationMode() { return application.isConfigurationMode(); diff --git a/webapp/packages/core-administration/src/AdministrationScreen/AdministrationScreenService.ts b/webapp/packages/core-administration/src/AdministrationScreen/AdministrationScreenService.ts index b0e247f2cc..9c8ff331f1 100644 --- a/webapp/packages/core-administration/src/AdministrationScreen/AdministrationScreenService.ts +++ b/webapp/packages/core-administration/src/AdministrationScreen/AdministrationScreenService.ts @@ -50,7 +50,7 @@ export class AdministrationScreenService { } get publicDisabled(): boolean { - return this.serverConfigResource.publicDisabled; + return this.permissionsService.publicDisabled; } readonly ensurePermissions: IExecutor; diff --git a/webapp/packages/core-authentication/src/PasswordPolicyService.ts b/webapp/packages/core-authentication/src/PasswordPolicyService.ts index 480e11bcec..12efcaf6a3 100644 --- a/webapp/packages/core-authentication/src/PasswordPolicyService.ts +++ b/webapp/packages/core-authentication/src/PasswordPolicyService.ts @@ -9,7 +9,7 @@ import { computed, makeObservable } from 'mobx'; import { injectable } from '@cloudbeaver/core-di'; import { LocalizationService } from '@cloudbeaver/core-localization'; -import { ServerConfigResource } from '@cloudbeaver/core-root'; +import { PasswordPolicyResource } from '@cloudbeaver/core-root'; import type { PasswordPolicyConfig } from '@cloudbeaver/core-sdk'; const DEFAULT_PASSWORD_POLICY: PasswordPolicyConfig = { @@ -25,15 +25,15 @@ type ValidationResult = { isValid: true; errorMessage: null } | { isValid: false export class PasswordPolicyService { get config(): PasswordPolicyConfig { return { - minLength: this.serverConfigResource.data?.passwordPolicyConfiguration?.minLength || DEFAULT_PASSWORD_POLICY.minLength, - minNumberCount: this.serverConfigResource.data?.passwordPolicyConfiguration?.minNumberCount || DEFAULT_PASSWORD_POLICY.minNumberCount, - minSymbolCount: this.serverConfigResource.data?.passwordPolicyConfiguration?.minSymbolCount || DEFAULT_PASSWORD_POLICY.minSymbolCount, - requireMixedCase: this.serverConfigResource.data?.passwordPolicyConfiguration?.requireMixedCase || DEFAULT_PASSWORD_POLICY.requireMixedCase, + minLength: this.passwordPolicyResource.data?.minLength || DEFAULT_PASSWORD_POLICY.minLength, + minNumberCount: this.passwordPolicyResource.data?.minNumberCount || DEFAULT_PASSWORD_POLICY.minNumberCount, + minSymbolCount: this.passwordPolicyResource.data?.minSymbolCount || DEFAULT_PASSWORD_POLICY.minSymbolCount, + requireMixedCase: this.passwordPolicyResource.data?.requireMixedCase || DEFAULT_PASSWORD_POLICY.requireMixedCase, }; } constructor( - private readonly serverConfigResource: ServerConfigResource, + private readonly passwordPolicyResource: PasswordPolicyResource, private readonly localizationService: LocalizationService, ) { makeObservable(this, { @@ -41,6 +41,9 @@ export class PasswordPolicyService { }); } + /** + * PasswordPolicyResource should be loaded before calling this method + */ validatePassword(password: string): ValidationResult { const trimmedPassword = password.trim(); diff --git a/webapp/packages/core-blocks/src/usePasswordValidation.ts b/webapp/packages/core-blocks/src/usePasswordValidation.ts index 1dc464a524..33d7b8903f 100644 --- a/webapp/packages/core-blocks/src/usePasswordValidation.ts +++ b/webapp/packages/core-blocks/src/usePasswordValidation.ts @@ -7,10 +7,13 @@ */ import { PasswordPolicyService } from '@cloudbeaver/core-authentication'; import { useService } from '@cloudbeaver/core-di'; +import { PasswordPolicyResource } from '@cloudbeaver/core-root'; import { useCustomInputValidation } from './FormControls/useCustomInputValidation'; +import { useResource } from './ResourcesHooks/useResource'; export function usePasswordValidation() { + useResource(usePasswordValidation, PasswordPolicyResource, undefined); const passwordPolicyService = useService(PasswordPolicyService); const ref = useCustomInputValidation(value => { diff --git a/webapp/packages/core-events/src/NotificationService.ts b/webapp/packages/core-events/src/NotificationService.ts index 410f51900f..95e49be844 100644 --- a/webapp/packages/core-events/src/NotificationService.ts +++ b/webapp/packages/core-events/src/NotificationService.ts @@ -80,26 +80,41 @@ export class NotificationService { const id = this.notificationNextId++; - const notification: INotification = { - id, - uuid: options.uuid, - title: options.title, - message: options.message, - details: options.details, - isSilent: !!options.isSilent, - customComponent: options.customComponent, - extraProps: options.extraProps || ({} as TProps), - autoClose: options.autoClose, - persistent: options.persistent, - state: observable({ deleteDelay: 0 }), - timestamp: options.timestamp || Date.now(), - type, - close: delayDeleting => { - this.close(id, delayDeleting); - options.onClose?.(delayDeleting); + const notification: INotification = observable( + { + id, + uuid: options.uuid, + title: options.title, + message: options.message, + details: options.details, + isSilent: !!options.isSilent, + customComponent: options.customComponent, + extraProps: options.extraProps || ({} as TProps), + autoClose: options.autoClose, + persistent: options.persistent, + state: observable({ deleteDelay: 0 }), + timestamp: options.timestamp || Date.now(), + type, + close: delayDeleting => { + this.close(id, delayDeleting); + options.onClose?.(delayDeleting); + }, + showDetails: this.showDetails.bind(this, id), + }, + { + title: observable.ref, + message: observable.ref, + details: observable.ref, + persistent: observable.ref, + isSilent: observable.ref, + customComponent: observable.ref, + extraProps: observable.ref, + autoClose: observable.ref, + type: observable.ref, + timestamp: observable.ref, + showDetails: observable.ref, }, - showDetails: this.showDetails.bind(this, id), - }; + ); this.notificationList.addValue(notification); diff --git a/webapp/packages/core-projects/src/ProjectInfoResource.ts b/webapp/packages/core-projects/src/ProjectInfoResource.ts index c38642c6d7..4e2d1ac296 100644 --- a/webapp/packages/core-projects/src/ProjectInfoResource.ts +++ b/webapp/packages/core-projects/src/ProjectInfoResource.ts @@ -8,7 +8,7 @@ import { AppAuthService, UserInfoResource } from '@cloudbeaver/core-authentication'; import { injectable } from '@cloudbeaver/core-di'; import { CachedMapAllKey, CachedMapResource, resourceKeyList } from '@cloudbeaver/core-resource'; -import { ServerConfigResource } from '@cloudbeaver/core-root'; +import { PermissionsService } from '@cloudbeaver/core-root'; import { GraphQLService, RmResourceType, ProjectInfo as SchemaProjectInfo } from '@cloudbeaver/core-sdk'; import { createResourceOfType } from './createResourceOfType'; @@ -21,7 +21,7 @@ export class ProjectInfoResource extends CachedMapResource constructor( private readonly graphQLService: GraphQLService, private readonly userInfoResource: UserInfoResource, - serverConfigResource: ServerConfigResource, + permissionsService: PermissionsService, appAuthService: AppAuthService, ) { super(() => new Map(), []); @@ -32,7 +32,7 @@ export class ProjectInfoResource extends CachedMapResource () => CachedMapAllKey, ); appAuthService.requireAuthentication(this); - serverConfigResource.requirePublic(this); + permissionsService.requirePublic(this); this.userInfoResource.onUserChange.addPostHandler(() => { this.clear(); }); diff --git a/webapp/packages/core-root/src/DefaultNavigatorSettingsResource.ts b/webapp/packages/core-root/src/DefaultNavigatorSettingsResource.ts new file mode 100644 index 0000000000..5d4312d815 --- /dev/null +++ b/webapp/packages/core-root/src/DefaultNavigatorSettingsResource.ts @@ -0,0 +1,103 @@ +/* + * CloudBeaver - Cloud Database Manager + * Copyright (C) 2020-2024 DBeaver Corp and others + * + * Licensed under the Apache License, Version 2.0. + * you may not use this file except in compliance with the License. + */ +import { action, makeObservable, observable } from 'mobx'; + +import { injectable } from '@cloudbeaver/core-di'; +import { CachedDataResource } from '@cloudbeaver/core-resource'; +import { DefaultNavigatorSettingsFragment, GraphQLService, NavigatorSettingsInput } from '@cloudbeaver/core-sdk'; + +import { isNavigatorViewSettingsEqual } from './ConnectionNavigatorViewSettings'; +import { ServerConfigResource } from './ServerConfigResource'; + +export type DefaultNavigatorSettings = DefaultNavigatorSettingsFragment['defaultNavigatorSettings']; + +@injectable() +export class DefaultNavigatorSettingsResource extends CachedDataResource { + update: NavigatorSettingsInput; + + constructor( + private readonly graphQLService: GraphQLService, + serverConfigResource: ServerConfigResource, + ) { + super(() => null, undefined, []); + this.sync(serverConfigResource); + + this.update = getDefaultNavigatorSettings(); + + makeObservable(this, { + update: observable, + unlinkUpdate: action, + syncUpdateData: action, + }); + } + + isChanged(): boolean { + if (!this.data) { + return false; + } + + return !isNavigatorViewSettingsEqual(this.data, this.update); + } + + setDataUpdate(update: NavigatorSettingsInput): void { + this.update = update; + } + + resetUpdate(): void { + if (this.data) { + this.syncUpdateData(this.data); + } + } + + unlinkUpdate(): void { + if (this.data) { + Object.assign(this.update, this.data); + } else { + this.update = getDefaultNavigatorSettings(); + } + } + + async save(): Promise { + await this.performUpdate( + undefined, + undefined, + async () => { + await this.graphQLService.sdk.setDefaultNavigatorSettings({ settings: this.update }); + + this.setData(await this.loader()); + + this.onDataOutdated.execute(); + }, + () => !this.isChanged(), + ); + } + + protected async loader(): Promise { + const { defaultNavigatorSettings } = await this.graphQLService.sdk.getDefaultNavigatorSettings(); + + this.syncUpdateData(defaultNavigatorSettings.defaultNavigatorSettings); + + return defaultNavigatorSettings.defaultNavigatorSettings; + } + + private syncUpdateData(defaultNavigatorSettings: DefaultNavigatorSettings) { + Object.assign(this.update, defaultNavigatorSettings); + } +} + +function getDefaultNavigatorSettings(): NavigatorSettingsInput { + return { + hideFolders: false, + hideSchemas: false, + hideVirtualModel: false, + mergeEntities: false, + showOnlyEntities: false, + showSystemObjects: false, + showUtilityObjects: false, + }; +} diff --git a/webapp/packages/core-root/src/PasswordPolicyResource.ts b/webapp/packages/core-root/src/PasswordPolicyResource.ts new file mode 100644 index 0000000000..d4d0c9fb8c --- /dev/null +++ b/webapp/packages/core-root/src/PasswordPolicyResource.ts @@ -0,0 +1,31 @@ +/* + * CloudBeaver - Cloud Database Manager + * Copyright (C) 2020-2024 DBeaver Corp and others + * + * Licensed under the Apache License, Version 2.0. + * you may not use this file except in compliance with the License. + */ +import { injectable } from '@cloudbeaver/core-di'; +import { CachedDataResource } from '@cloudbeaver/core-resource'; +import { GraphQLService, PasswordPolicyFragment } from '@cloudbeaver/core-sdk'; + +import { ServerConfigResource } from './ServerConfigResource'; + +export type PasswordPolicy = PasswordPolicyFragment['passwordPolicyConfiguration']; + +@injectable() +export class PasswordPolicyResource extends CachedDataResource { + constructor( + private readonly graphQLService: GraphQLService, + serverConfigResource: ServerConfigResource, + ) { + super(() => null, undefined, []); + this.sync(serverConfigResource); + } + + protected async loader(): Promise { + const { passwordPolicy } = await this.graphQLService.sdk.getPasswordPolicy(); + + return passwordPolicy.passwordPolicyConfiguration; + } +} diff --git a/webapp/packages/core-root/src/PermissionsService.ts b/webapp/packages/core-root/src/PermissionsService.ts index 06dd86b598..a264ae1dae 100644 --- a/webapp/packages/core-root/src/PermissionsService.ts +++ b/webapp/packages/core-root/src/PermissionsService.ts @@ -6,14 +6,37 @@ * you may not use this file except in compliance with the License. */ import { injectable } from '@cloudbeaver/core-di'; +import { ExecutorInterrupter } from '@cloudbeaver/core-executor'; +import { CachedResource } from '@cloudbeaver/core-resource'; +import { ServerConfigResource } from './ServerConfigResource'; +import { ServerLicenseStatusResource } from './ServerLicenseStatusResource'; import { SessionPermissionsResource } from './SessionPermissionsResource'; export enum EPermission {} @injectable() export class PermissionsService { - constructor(private readonly permissions: SessionPermissionsResource) {} + get publicDisabled(): boolean { + return ( + this.serverConfigResource.configurationMode || + (this.serverLicenseStatusResource?.licenseRequired && !this.serverLicenseStatusResource.licenseValid) + ); + } + + constructor( + private readonly permissions: SessionPermissionsResource, + private readonly serverConfigResource: ServerConfigResource, + private readonly serverLicenseStatusResource: ServerLicenseStatusResource, + ) {} + + requirePublic(resource: CachedResource, map?: (param: void) => T): this { + resource.preloadResource(this.serverLicenseStatusResource, () => {}).before(ExecutorInterrupter.interrupter(() => this.publicDisabled)); + + this.serverLicenseStatusResource.outdateResource(resource, map as any); + + return this; + } has(id: string): boolean { return this.permissions.has(id); diff --git a/webapp/packages/core-root/src/ProductInfoResource.ts b/webapp/packages/core-root/src/ProductInfoResource.ts new file mode 100644 index 0000000000..fc25a1ea64 --- /dev/null +++ b/webapp/packages/core-root/src/ProductInfoResource.ts @@ -0,0 +1,31 @@ +/* + * CloudBeaver - Cloud Database Manager + * Copyright (C) 2020-2024 DBeaver Corp and others + * + * Licensed under the Apache License, Version 2.0. + * you may not use this file except in compliance with the License. + */ +import { injectable } from '@cloudbeaver/core-di'; +import { CachedDataResource } from '@cloudbeaver/core-resource'; +import { GraphQLService, ProductInfoFragment } from '@cloudbeaver/core-sdk'; + +import { ServerConfigResource } from './ServerConfigResource'; + +export type ProductInfo = ProductInfoFragment['productInfo']; + +@injectable() +export class ProductInfoResource extends CachedDataResource { + constructor( + private readonly graphQLService: GraphQLService, + serverConfigResource: ServerConfigResource, + ) { + super(() => null, undefined, []); + this.sync(serverConfigResource); + } + + protected async loader(): Promise { + const { productInfo } = await this.graphQLService.sdk.getProductInfo(); + + return productInfo.productInfo; + } +} diff --git a/webapp/packages/core-root/src/QuotasService.ts b/webapp/packages/core-root/src/QuotasService.ts index 81ba0d3ba3..d52cb5f63a 100644 --- a/webapp/packages/core-root/src/QuotasService.ts +++ b/webapp/packages/core-root/src/QuotasService.ts @@ -7,7 +7,7 @@ */ import { injectable } from '@cloudbeaver/core-di'; -import { ServerConfigResource } from './ServerConfigResource'; +import { ServerResourceQuotasResource } from './ServerResourceQuotasResource'; interface IQuotas { dataExportFileSizeLimit: number; @@ -46,10 +46,13 @@ export class QuotasService { }; } - constructor(private readonly serverConfigResource: ServerConfigResource) {} + constructor(private readonly serverResourceQuotasResource: ServerResourceQuotasResource) {} + /** + * Quotas should be manually loaded from ServerResourceQuotasResource before using this method + */ getQuota(key: QuotaKey) { - const serverQuota = this.serverConfigResource.data?.resourceQuotas[key]; + const serverQuota = this.serverResourceQuotasResource.data?.resourceQuotas[key]; if (isNumber(serverQuota)) { return serverQuota; diff --git a/webapp/packages/core-root/src/ServerConfigResource.ts b/webapp/packages/core-root/src/ServerConfigResource.ts index b8ab5c11b3..9227081242 100644 --- a/webapp/packages/core-root/src/ServerConfigResource.ts +++ b/webapp/packages/core-root/src/ServerConfigResource.ts @@ -8,24 +8,21 @@ import { action, makeObservable, observable } from 'mobx'; import { injectable } from '@cloudbeaver/core-di'; -import { ExecutorInterrupter } from '@cloudbeaver/core-executor'; -import { CachedDataResource, type CachedResource } from '@cloudbeaver/core-resource'; -import { GraphQLService, NavigatorSettingsInput, ServerConfig as SDKServerConfig, ServerConfigInput } from '@cloudbeaver/core-sdk'; +import { CachedDataResource } from '@cloudbeaver/core-resource'; +import { GraphQLService, ServerConfigFragment, ServerConfigInput } from '@cloudbeaver/core-sdk'; import { isArraysEqual } from '@cloudbeaver/core-utils'; -import { isNavigatorViewSettingsEqual } from './ConnectionNavigatorViewSettings'; import { DataSynchronizationQueue } from './DataSynchronization/DataSynchronizationQueue'; import { DataSynchronizationService } from './DataSynchronization/DataSynchronizationService'; import { ServerConfigEventHandler } from './ServerConfigEventHandler'; export const FEATURE_GIT_ID = 'git'; -export type ServerConfig = Omit; +export type ServerConfig = ServerConfigFragment; @injectable() export class ServerConfigResource extends CachedDataResource { update: ServerConfigInput; - navigatorSettingsUpdate: NavigatorSettingsInput; private readonly syncQueue: DataSynchronizationQueue; @@ -42,19 +39,9 @@ export class ServerConfigResource extends CachedDataResource(this, { update: observable, - navigatorSettingsUpdate: observable, unlinkUpdate: action, syncUpdateData: action, }); @@ -69,14 +56,6 @@ export class ServerConfigResource extends CachedDataResource(resource: CachedResource, map?: (param: void) => T): this { - resource.preloadResource(this, () => {}).before(ExecutorInterrupter.interrupter(() => this.publicDisabled)); - - this.outdateResource(resource, map as any); - - return this; - } - get redirectOnFederatedAuth(): boolean { return this.data?.redirectOnFederatedAuth ?? false; } @@ -93,26 +72,10 @@ export class ServerConfigResource extends CachedDataResource { + async save(): Promise { await this.performUpdate( undefined, undefined, async () => { - if (this.isNavigatorSettingsChanged()) { - await this.graphQLService.sdk.setDefaultNavigatorSettings({ settings: this.navigatorSettingsUpdate }); - - if (this.data) { - this.data.defaultNavigatorSettings = { ...this.navigatorSettingsUpdate }; - } else { - this.setData(await this.loader()); - } - } - - if (this.isChanged() && !skipConfigUpdate) { - await this.graphQLService.sdk.configureServer({ - configuration: this.update, - }); - this.setData(await this.loader()); - } - + await this.graphQLService.sdk.configureServer({ + configuration: this.update, + }); + this.setData(await this.loader()); this.onDataOutdated.execute(); }, - () => !this.isNavigatorSettingsChanged() && (!this.isChanged() || skipConfigUpdate), + () => !this.isChanged(), ); } @@ -289,7 +205,7 @@ export class ServerConfigResource extends CachedDataResource { + constructor( + private readonly graphQLService: GraphQLService, + serverConfigResource: ServerConfigResource, + ) { + super(() => null, undefined, []); + this.sync(serverConfigResource); + } + + get licenseRequired(): boolean { + return this.data?.licenseRequired ?? false; + } + + get licenseValid(): boolean { + return this.data?.licenseValid ?? false; + } + + protected async loader(): Promise { + const { licenseStatus } = await this.graphQLService.sdk.getServerLicenseStatus(); + + return licenseStatus; + } +} diff --git a/webapp/packages/core-root/src/ServerResourceQuotasResource.ts b/webapp/packages/core-root/src/ServerResourceQuotasResource.ts new file mode 100644 index 0000000000..1af57e41f0 --- /dev/null +++ b/webapp/packages/core-root/src/ServerResourceQuotasResource.ts @@ -0,0 +1,31 @@ +/* + * CloudBeaver - Cloud Database Manager + * Copyright (C) 2020-2024 DBeaver Corp and others + * + * Licensed under the Apache License, Version 2.0. + * you may not use this file except in compliance with the License. + */ +import { injectable } from '@cloudbeaver/core-di'; +import { CachedDataResource } from '@cloudbeaver/core-resource'; +import { GraphQLService, ResourceQuotasFragment } from '@cloudbeaver/core-sdk'; + +import { ServerConfigResource } from './ServerConfigResource'; + +export type ServerResourceQuotas = ResourceQuotasFragment['resourceQuotas']; + +@injectable() +export class ServerResourceQuotasResource extends CachedDataResource { + constructor( + private readonly graphQLService: GraphQLService, + serverConfigResource: ServerConfigResource, + ) { + super(() => null, undefined, []); + this.sync(serverConfigResource); + } + + protected async loader(): Promise { + const { resourceQuotas } = await this.graphQLService.sdk.getServerResourceQuotas(); + + return resourceQuotas.resourceQuotas; + } +} diff --git a/webapp/packages/core-root/src/index.ts b/webapp/packages/core-root/src/index.ts index 5531ad75e1..386e0627e4 100644 --- a/webapp/packages/core-root/src/index.ts +++ b/webapp/packages/core-root/src/index.ts @@ -12,11 +12,14 @@ export * from './DataSynchronization/SynchronizationMessage'; export * from './ServerEventEmitter/IServerEventEmitter'; export * from './ServerEventEmitter/TopicEventHandler'; export * from './ConnectionNavigatorViewSettings'; +export * from './DefaultNavigatorSettingsResource'; export * from './FeaturesResource'; export * from './NetworkError'; export * from './NetworkStateService'; +export * from './PasswordPolicyResource'; export * from './SessionPermissionsResource'; export * from './PermissionsService'; +export * from './ProductInfoResource'; export * from './ServerConfigResource'; export * from './SessionEventSource'; export * from './QuotasService'; @@ -31,5 +34,7 @@ export * from './SessionExpireService'; export * from './SessionExpireEventService'; export * from './SessionActivityService'; export * from './ServerNodeService'; +export * from './ServerResourceQuotasResource'; export * from './WindowEventsService'; +export * from './ServerLicenseStatusResource'; export * from './manifest'; diff --git a/webapp/packages/core-root/src/manifest.ts b/webapp/packages/core-root/src/manifest.ts index 96d3d7e560..9a3e3b6dbe 100644 --- a/webapp/packages/core-root/src/manifest.ts +++ b/webapp/packages/core-root/src/manifest.ts @@ -36,5 +36,10 @@ export const coreRootManifest: PluginManifest = { () => import('./Settings/ServerSettingsResource').then(m => m.ServerSettingsResource), () => import('./Settings/ServerSettingsManagerService').then(m => m.ServerSettingsManagerService), () => import('./RootBootstrap').then(m => m.RootBootstrap), + () => import('./ServerLicenseStatusResource').then(m => m.ServerLicenseStatusResource), + () => import('./PasswordPolicyResource').then(m => m.PasswordPolicyResource), + () => import('./ProductInfoResource').then(m => m.ProductInfoResource), + () => import('./DefaultNavigatorSettingsResource').then(m => m.DefaultNavigatorSettingsResource), + () => import('./ServerResourceQuotasResource').then(m => m.ServerResourceQuotasResource), ], }; diff --git a/webapp/packages/core-sdk/src/queries/fragments/ServerConfig/DefaultNavigatorSettings.gql b/webapp/packages/core-sdk/src/queries/fragments/ServerConfig/DefaultNavigatorSettings.gql new file mode 100644 index 0000000000..8278665cab --- /dev/null +++ b/webapp/packages/core-sdk/src/queries/fragments/ServerConfig/DefaultNavigatorSettings.gql @@ -0,0 +1,5 @@ +fragment DefaultNavigatorSettings on ServerConfig { + defaultNavigatorSettings { + ...AllNavigatorSettings + } +} diff --git a/webapp/packages/core-sdk/src/queries/fragments/ServerConfig/PasswordPolicy.gql b/webapp/packages/core-sdk/src/queries/fragments/ServerConfig/PasswordPolicy.gql new file mode 100644 index 0000000000..72883c56d3 --- /dev/null +++ b/webapp/packages/core-sdk/src/queries/fragments/ServerConfig/PasswordPolicy.gql @@ -0,0 +1,8 @@ +fragment PasswordPolicy on ServerConfig { + passwordPolicyConfiguration { + minLength + minNumberCount + minSymbolCount + requireMixedCase + } +} diff --git a/webapp/packages/core-sdk/src/queries/fragments/ServerConfig/ProductInfo.gql b/webapp/packages/core-sdk/src/queries/fragments/ServerConfig/ProductInfo.gql new file mode 100644 index 0000000000..2ce93b8765 --- /dev/null +++ b/webapp/packages/core-sdk/src/queries/fragments/ServerConfig/ProductInfo.gql @@ -0,0 +1,12 @@ +fragment ProductInfo on ServerConfig { + productInfo { + id + version + latestVersionInfo + name + description + buildTime + releaseTime + licenseInfo + } +} diff --git a/webapp/packages/core-sdk/src/queries/fragments/ServerConfig/ResourceQuotas.gql b/webapp/packages/core-sdk/src/queries/fragments/ServerConfig/ResourceQuotas.gql new file mode 100644 index 0000000000..7d92dc727f --- /dev/null +++ b/webapp/packages/core-sdk/src/queries/fragments/ServerConfig/ResourceQuotas.gql @@ -0,0 +1,3 @@ +fragment ResourceQuotas on ServerConfig { + resourceQuotas +} diff --git a/webapp/packages/core-sdk/src/queries/fragments/ServerConfig/ServerConfig.gql b/webapp/packages/core-sdk/src/queries/fragments/ServerConfig/ServerConfig.gql new file mode 100644 index 0000000000..1f184e12af --- /dev/null +++ b/webapp/packages/core-sdk/src/queries/fragments/ServerConfig/ServerConfig.gql @@ -0,0 +1,35 @@ +fragment ServerConfig on ServerConfig { + name + version + workspaceId + serverURL + rootURI + containerId + defaultAuthRole + defaultUserTeam + productConfiguration + supportsCustomConnections + sessionExpireTime + + anonymousAccessEnabled + + adminCredentialsSaveEnabled + publicCredentialsSaveEnabled + + resourceManagerEnabled + + configurationMode + developmentMode + redirectOnFederatedAuth + distributed + + enabledFeatures + disabledBetaFeatures + enabledAuthProviders + supportedLanguages { + isoCode + displayName + nativeName + } + disabledDrivers +} diff --git a/webapp/packages/core-sdk/src/queries/fragments/ServerConfig/ServerLicenseStatus.gql b/webapp/packages/core-sdk/src/queries/fragments/ServerConfig/ServerLicenseStatus.gql new file mode 100644 index 0000000000..c6172b834d --- /dev/null +++ b/webapp/packages/core-sdk/src/queries/fragments/ServerConfig/ServerLicenseStatus.gql @@ -0,0 +1,5 @@ +fragment ServerLicenseStatus on ServerConfig { + licenseRequired + licenseValid + licenseStatus +} diff --git a/webapp/packages/core-sdk/src/queries/session/getDefaultNavigatorSettings.gql b/webapp/packages/core-sdk/src/queries/session/getDefaultNavigatorSettings.gql new file mode 100644 index 0000000000..b514f9f4dd --- /dev/null +++ b/webapp/packages/core-sdk/src/queries/session/getDefaultNavigatorSettings.gql @@ -0,0 +1,5 @@ +query getDefaultNavigatorSettings { + defaultNavigatorSettings: serverConfig { + ...DefaultNavigatorSettings + } +} diff --git a/webapp/packages/core-sdk/src/queries/session/getPasswordPolicy.gql b/webapp/packages/core-sdk/src/queries/session/getPasswordPolicy.gql new file mode 100644 index 0000000000..57e1cba85b --- /dev/null +++ b/webapp/packages/core-sdk/src/queries/session/getPasswordPolicy.gql @@ -0,0 +1,5 @@ +query getPasswordPolicy { + passwordPolicy: serverConfig { + ...PasswordPolicy + } +} diff --git a/webapp/packages/core-sdk/src/queries/session/getProductInfo.gql b/webapp/packages/core-sdk/src/queries/session/getProductInfo.gql new file mode 100644 index 0000000000..0ccbeb9643 --- /dev/null +++ b/webapp/packages/core-sdk/src/queries/session/getProductInfo.gql @@ -0,0 +1,5 @@ +query getProductInfo { + productInfo: serverConfig { + ...ProductInfo + } +} diff --git a/webapp/packages/core-sdk/src/queries/session/getServerLicenseStatus.gql b/webapp/packages/core-sdk/src/queries/session/getServerLicenseStatus.gql new file mode 100644 index 0000000000..b8c9546386 --- /dev/null +++ b/webapp/packages/core-sdk/src/queries/session/getServerLicenseStatus.gql @@ -0,0 +1,5 @@ +query getServerLicenseStatus { + licenseStatus: serverConfig { + ...ServerLicenseStatus + } +} diff --git a/webapp/packages/core-sdk/src/queries/session/getServerResourceQuotas.gql b/webapp/packages/core-sdk/src/queries/session/getServerResourceQuotas.gql new file mode 100644 index 0000000000..26b8b66e85 --- /dev/null +++ b/webapp/packages/core-sdk/src/queries/session/getServerResourceQuotas.gql @@ -0,0 +1,5 @@ +query getServerResourceQuotas { + resourceQuotas: serverConfig { + ...ResourceQuotas + } +} diff --git a/webapp/packages/core-sdk/src/queries/session/serverConfig.gql b/webapp/packages/core-sdk/src/queries/session/serverConfig.gql index 6cb48088e0..5b065603bb 100644 --- a/webapp/packages/core-sdk/src/queries/session/serverConfig.gql +++ b/webapp/packages/core-sdk/src/queries/session/serverConfig.gql @@ -1,63 +1,5 @@ query serverConfig { serverConfig { - name - version - workspaceId - serverURL - rootURI - containerId - defaultAuthRole - defaultUserTeam - productConfiguration - supportsCustomConnections - supportsConnectionBrowser - supportsWorkspaces - sessionExpireTime - - anonymousAccessEnabled - - adminCredentialsSaveEnabled - publicCredentialsSaveEnabled - - resourceManagerEnabled - - licenseRequired - licenseValid - - configurationMode - developmentMode - redirectOnFederatedAuth - distributed - - enabledFeatures - disabledBetaFeatures - enabledAuthProviders - supportedLanguages { - isoCode - displayName - nativeName - } - productConfiguration - defaultNavigatorSettings { - ...AllNavigatorSettings - } - resourceQuotas - disabledDrivers - productInfo { - id - version - latestVersionInfo - name - description - buildTime - releaseTime - licenseInfo - } - passwordPolicyConfiguration { - minLength - minNumberCount - minSymbolCount - requireMixedCase - } + ...ServerConfig } } diff --git a/webapp/packages/core-version/src/VersionResource.ts b/webapp/packages/core-version/src/VersionResource.ts index c9f8c5b1f5..e4dae493e8 100644 --- a/webapp/packages/core-version/src/VersionResource.ts +++ b/webapp/packages/core-version/src/VersionResource.ts @@ -9,7 +9,7 @@ import { computed, makeObservable, observable } from 'mobx'; import { injectable } from '@cloudbeaver/core-di'; import { CachedMapResource, resourceKeyList } from '@cloudbeaver/core-resource'; -import { ServerConfigResource } from '@cloudbeaver/core-root'; +import { ProductInfoResource } from '@cloudbeaver/core-root'; export interface IVersion { number: string; @@ -30,11 +30,11 @@ export class VersionResource extends CachedMapResource { return this.values.find(v => v.number === this.latestVersionNumber); } - constructor(private readonly serverConfigResource: ServerConfigResource) { + constructor(private readonly productInfoResource: ProductInfoResource) { super(); this.latestVersionNumber = null; - this.preloadResource(this.serverConfigResource, () => {}); + this.preloadResource(this.productInfoResource, () => {}); makeObservable(this, { latestVersionNumber: observable.ref, @@ -43,7 +43,7 @@ export class VersionResource extends CachedMapResource { } protected async loader(): Promise> { - const versionLink = this.serverConfigResource.data?.productInfo.latestVersionInfo; + const versionLink = this.productInfoResource.data?.latestVersionInfo; if (!versionLink) { return this.data; } diff --git a/webapp/packages/plugin-administration/src/ConfigurationWizard/ServerConfiguration/ServerConfigurationPage.tsx b/webapp/packages/plugin-administration/src/ConfigurationWizard/ServerConfiguration/ServerConfigurationPage.tsx index 0d9554fbae..f1312a74ce 100644 --- a/webapp/packages/plugin-administration/src/ConfigurationWizard/ServerConfiguration/ServerConfigurationPage.tsx +++ b/webapp/packages/plugin-administration/src/ConfigurationWizard/ServerConfiguration/ServerConfigurationPage.tsx @@ -28,7 +28,7 @@ import { } from '@cloudbeaver/core-blocks'; import { useService } from '@cloudbeaver/core-di'; import { CommonDialogService, DialogueStateResult } from '@cloudbeaver/core-dialogs'; -import { ServerConfigResource } from '@cloudbeaver/core-root'; +import { DefaultNavigatorSettingsResource, ServerConfigResource } from '@cloudbeaver/core-root'; import { ServerConfigurationConfigurationForm } from './Form/ServerConfigurationConfigurationForm'; import { ServerConfigurationFeaturesForm } from './Form/ServerConfigurationFeaturesForm'; @@ -45,9 +45,10 @@ export const ServerConfigurationPage: AdministrationItemContentComponent = obser const [focusedRef, state] = useFocus({ focusFirstChild: true }); const service = useService(ServerConfigurationService); const serverConfigResource = useService(ServerConfigResource); + const defaultNavigatorSettingsResource = useService(DefaultNavigatorSettingsResource); const commonDialogService = useService(CommonDialogService); const configurationWizardService = useService(ConfigurationWizardService); - const changed = serverConfigResource.isChanged() || serverConfigResource.isNavigatorSettingsChanged(); + const changed = serverConfigResource.isChanged() || defaultNavigatorSettingsResource.isChanged(); useFormValidator(service.validationTask, state.reference); function handleChange() { diff --git a/webapp/packages/plugin-administration/src/ConfigurationWizard/ServerConfiguration/ServerConfigurationService.ts b/webapp/packages/plugin-administration/src/ConfigurationWizard/ServerConfiguration/ServerConfigurationService.ts index 401562aef8..1f4f89bf32 100644 --- a/webapp/packages/plugin-administration/src/ConfigurationWizard/ServerConfiguration/ServerConfigurationService.ts +++ b/webapp/packages/plugin-administration/src/ConfigurationWizard/ServerConfiguration/ServerConfigurationService.ts @@ -13,7 +13,7 @@ import { DEFAULT_NAVIGATOR_VIEW_SETTINGS } from '@cloudbeaver/core-connections'; import { injectable } from '@cloudbeaver/core-di'; import { ENotificationType, INotification, NotificationService } from '@cloudbeaver/core-events'; import { Executor, ExecutorInterrupter, IExecutor, IExecutorHandler } from '@cloudbeaver/core-executor'; -import { ServerConfigResource, SessionDataResource } from '@cloudbeaver/core-root'; +import { DefaultNavigatorSettingsResource, ProductInfoResource, ServerConfigResource, SessionDataResource } from '@cloudbeaver/core-root'; import { ADMINISTRATION_SERVER_CONFIGURATION_ITEM } from './ADMINISTRATION_SERVER_CONFIGURATION_ITEM'; import type { IServerConfigurationPageState } from './IServerConfigurationPageState'; @@ -53,8 +53,10 @@ export class ServerConfigurationService { constructor( private readonly administrationScreenService: AdministrationScreenService, private readonly serverConfigResource: ServerConfigResource, + private readonly defaultNavigatorSettingsResource: DefaultNavigatorSettingsResource, private readonly notificationService: NotificationService, private readonly sessionDataResource: SessionDataResource, + private readonly productInfoResource: ProductInfoResource, ) { this.done = false; this.loading = true; @@ -119,11 +121,13 @@ export class ServerConfigurationService { reset = true; this.stateLinked = true; await this.serverConfigResource.load(); + await this.defaultNavigatorSettingsResource.load(); this.serverConfigResource.setDataUpdate(this.state.serverConfig); - this.serverConfigResource.setNavigatorSettingsUpdate(this.state.navigatorConfig); + this.defaultNavigatorSettingsResource.setDataUpdate(this.state.navigatorConfig); if (reset) { + this.defaultNavigatorSettingsResource.resetUpdate(); this.serverConfigResource.resetUpdate(); } } @@ -156,12 +160,14 @@ export class ServerConfigurationService { try { const config = await this.serverConfigResource.load(); + const productInfo = await this.productInfoResource.load(); + const defaultNavigatorSettings = await this.defaultNavigatorSettingsResource.load(); if (!config) { return; } - data.state.serverConfig.serverName = config.name || config.productInfo.name; + data.state.serverConfig.serverName = config.name || productInfo?.name; data.state.serverConfig.serverURL = config.serverURL; if (this.administrationScreenService.isConfigurationMode && !config.distributed) { @@ -175,7 +181,7 @@ export class ServerConfigurationService { data.state.serverConfig.customConnectionsEnabled = config.supportsCustomConnections; data.state.serverConfig.disabledDrivers = [...config.disabledDrivers]; - Object.assign(data.state.navigatorConfig, config.defaultNavigatorSettings); + Object.assign(data.state.navigatorConfig, defaultNavigatorSettings); } catch (exception: any) { ExecutorInterrupter.interrupt(contexts); this.notificationService.logException(exception, "Can't load server configuration"); @@ -198,7 +204,10 @@ export class ServerConfigurationService { } try { - await this.serverConfigResource.save(data.configurationWizard); + await this.defaultNavigatorSettingsResource.save(); + if (!data.configurationWizard) { + await this.serverConfigResource.save(); + } if (data.configurationWizard && data.finish) { await this.serverConfigResource.finishConfiguration(); @@ -252,7 +261,7 @@ export class ServerConfigurationService { private showUnsavedNotification(close: boolean) { if ( - (!this.serverConfigResource.isChanged() && !this.serverConfigResource.isNavigatorSettingsChanged()) || + (!this.serverConfigResource.isChanged() && !this.defaultNavigatorSettingsResource.isChanged()) || this.administrationScreenService.activeScreen?.item === ADMINISTRATION_SERVER_CONFIGURATION_ITEM ) { this.unSaveNotification?.close(true); @@ -292,6 +301,7 @@ export class ServerConfigurationService { } this.unSaveNotification?.close(true); + this.defaultNavigatorSettingsResource.unlinkUpdate(); this.serverConfigResource.unlinkUpdate(); this.stateLinked = false; } diff --git a/webapp/packages/plugin-authentication-administration/src/Administration/ServerConfiguration/ServerConfigurationAuthenticationBootstrap.ts b/webapp/packages/plugin-authentication-administration/src/Administration/ServerConfiguration/ServerConfigurationAuthenticationBootstrap.ts index 7df18decf2..2a6b7d4d25 100644 --- a/webapp/packages/plugin-authentication-administration/src/Administration/ServerConfiguration/ServerConfigurationAuthenticationBootstrap.ts +++ b/webapp/packages/plugin-authentication-administration/src/Administration/ServerConfiguration/ServerConfigurationAuthenticationBootstrap.ts @@ -11,7 +11,7 @@ import { Bootstrap, injectable } from '@cloudbeaver/core-di'; import { NotificationService } from '@cloudbeaver/core-events'; import { ExecutorInterrupter, IExecutorHandler } from '@cloudbeaver/core-executor'; import { CachedMapAllKey } from '@cloudbeaver/core-resource'; -import { ServerConfigResource } from '@cloudbeaver/core-root'; +import { PasswordPolicyResource, ServerConfigResource } from '@cloudbeaver/core-root'; import { ILoadConfigData, IServerConfigSaveData, @@ -28,6 +28,7 @@ export class ServerConfigurationAuthenticationBootstrap extends Bootstrap { private readonly serverConfigResource: ServerConfigResource, private readonly notificationService: NotificationService, private readonly passwordPolicyService: PasswordPolicyService, + private readonly passwordPolicyResource: PasswordPolicyResource, ) { super(); } @@ -85,6 +86,7 @@ export class ServerConfigurationAuthenticationBootstrap extends Bootstrap { return validation.invalidate(); } + await this.passwordPolicyResource.load(); const passwordValidation = this.passwordPolicyService.validatePassword(data.state.serverConfig.adminPassword); if (!passwordValidation.isValid) { validation.error(passwordValidation.errorMessage); diff --git a/webapp/packages/plugin-data-viewer/src/DatabaseDataModel/Actions/ResultSet/ResultSetDataContentAction.ts b/webapp/packages/plugin-data-viewer/src/DatabaseDataModel/Actions/ResultSet/ResultSetDataContentAction.ts index 78ed677a46..c54ec2210e 100644 --- a/webapp/packages/plugin-data-viewer/src/DatabaseDataModel/Actions/ResultSet/ResultSetDataContentAction.ts +++ b/webapp/packages/plugin-data-viewer/src/DatabaseDataModel/Actions/ResultSet/ResultSetDataContentAction.ts @@ -7,7 +7,7 @@ */ import { makeObservable, observable } from 'mobx'; -import { QuotasService } from '@cloudbeaver/core-root'; +import { QuotasService, ServerResourceQuotasResource } from '@cloudbeaver/core-root'; import { GraphQLService, ResultDataFormat } from '@cloudbeaver/core-sdk'; import { bytesToSize, download, downloadFromURL, GlobalConstants, isNotNullDefined } from '@cloudbeaver/core-utils'; @@ -34,17 +34,31 @@ interface ICacheEntry { @databaseDataAction() export class ResultSetDataContentAction extends DatabaseDataAction implements IResultSetDataContentAction { static dataFormat = [ResultDataFormat.Resultset]; + private subscriptionDispose?: () => void; constructor( source: IDatabaseDataSource, private readonly data: ResultSetDataAction, private readonly format: ResultSetFormatAction, private readonly graphQLService: GraphQLService, + private readonly serverResourceQuotasResource: ServerResourceQuotasResource, private readonly quotasService: QuotasService, private readonly cache: ResultSetCacheAction, ) { super(source); + function loadQuotas() { + setTimeout(() => serverResourceQuotasResource.load(), 0); + } + + this.serverResourceQuotasResource.onDataOutdated.addHandler(loadQuotas); + + loadQuotas(); + + this.subscriptionDispose = () => { + this.serverResourceQuotasResource.onDataOutdated.removeHandler(loadQuotas); + }; + makeObservable(this, { cache: observable, }); @@ -168,6 +182,7 @@ export class ResultSetDataContentAction extends DatabaseDataAction import('./ProductInfoDialog' @injectable() export class ProductBootstrap extends Bootstrap { constructor( - private readonly serverConfigResource: ServerConfigResource, + private readonly productInfoResource: ProductInfoResource, private readonly commonDialogService: CommonDialogService, private readonly menuService: MenuService, ) { @@ -27,7 +27,7 @@ export class ProductBootstrap extends Bootstrap { register(): void { this.menuService.addCreator({ menus: [TOP_NAV_BAR_SETTINGS_MENU], - isApplicable: () => !!this.serverConfigResource.data?.productInfo, + isApplicable: () => !!this.productInfoResource.data, getItems: (context, items) => [ ...items, new MenuBaseItem( diff --git a/webapp/packages/plugin-product/src/ProductInfoDialog.tsx b/webapp/packages/plugin-product/src/ProductInfoDialog.tsx index 3032463ae2..524ad1d30b 100644 --- a/webapp/packages/plugin-product/src/ProductInfoDialog.tsx +++ b/webapp/packages/plugin-product/src/ProductInfoDialog.tsx @@ -25,7 +25,7 @@ import { } from '@cloudbeaver/core-blocks'; import { useService } from '@cloudbeaver/core-di'; import type { DialogComponentProps } from '@cloudbeaver/core-dialogs'; -import { ServerConfigResource } from '@cloudbeaver/core-root'; +import { ProductInfoResource } from '@cloudbeaver/core-root'; import { ThemeService } from '@cloudbeaver/core-theming'; import { useAppVersion } from '@cloudbeaver/core-version'; import { WebsiteLinks } from '@cloudbeaver/core-website'; @@ -34,12 +34,12 @@ import ProductInfoDialogStyles from './ProductInfoDialog.module.css'; export const ProductInfoDialog = observer>(function ProductInfoDialog(props) { const translate = useTranslate(); - const serverConfigResource = useService(ServerConfigResource); + const serverConfigResource = useService(ProductInfoResource); const themeService = useService(ThemeService); const version = useAppVersion(); - const productInfo = serverConfigResource.data?.productInfo; + const productInfo = serverConfigResource.data; const logoIcon = themeService.themeId === 'light' ? '/icons/product-logo_light.svg' : '/icons/product-logo_dark.svg'; const styles = useS(ProductInfoDialogStyles); diff --git a/webapp/packages/plugin-top-app-bar/src/TopNavBar/Logo.tsx b/webapp/packages/plugin-top-app-bar/src/TopNavBar/Logo.tsx index d0a2c38770..a8f4d0edd9 100644 --- a/webapp/packages/plugin-top-app-bar/src/TopNavBar/Logo.tsx +++ b/webapp/packages/plugin-top-app-bar/src/TopNavBar/Logo.tsx @@ -7,20 +7,20 @@ */ import { observer } from 'mobx-react-lite'; -import { AppLogo } from '@cloudbeaver/core-blocks'; +import { AppLogo, useResource } from '@cloudbeaver/core-blocks'; import { useService } from '@cloudbeaver/core-di'; -import { ServerConfigResource } from '@cloudbeaver/core-root'; +import { ProductInfoResource } from '@cloudbeaver/core-root'; import { ScreenService } from '@cloudbeaver/core-routing'; import { useAppVersion } from '@cloudbeaver/core-version'; export const Logo = observer(function Logo() { - const serverConfigResource = useService(ServerConfigResource); + const productInfoResource = useResource(Logo, ProductInfoResource, undefined); const screenService = useService(ScreenService); const { backendVersion, frontendVersion } = useAppVersion(true); const isSameVersion = backendVersion === frontendVersion; - const productName = serverConfigResource.data?.productInfo.name || 'CloudBeaver'; + const productName = productInfoResource.data?.name || 'CloudBeaver'; const backendVersionTitle = `${productName} ver. ${backendVersion}`; const commonVersionTitle = `${productName} ver. ${frontendVersion}(${backendVersion})`;