diff --git a/ISSUE.md b/ISSUE.md new file mode 100644 index 00000000000..fe3591515de --- /dev/null +++ b/ISSUE.md @@ -0,0 +1 @@ +[FE] custom app name, favicon and logo #4377 diff --git a/README.md b/README.md index f6a16c862c3..1a56510da41 100644 --- a/README.md +++ b/README.md @@ -139,3 +139,13 @@ All of the environment variables/config properties could be found [here](https:/ # Contributing Please refer to [contributing guide](https://docs.kafka-ui.provectus.io/development/contributing), we'll guide you from there. + +# Brand the kafka ui +New environment vars: +- KAFKA_UI_PREFERENCES_REMOVEGITLINK: remove git link +- KAFKA_UI_PREFERENCES_REMOVEDISCORDLINK: remove discord link +- KAFKA_UI_PREFERENCES_APPNAME: change app name from Kafka UI to the desiderata +- KAFKA_UI_PREFERENCES_FAVICON: change favicon to the desiderata (please add data:image/png;base64,) +- KAFKA_UI_PREFERENCES_LOGO: change favicon to the desiderata (please add data:image/png;base64,) +- KAFKA_UI_PREFERENCES_VERSION: to remove version banner + diff --git a/kafka-ui-api/src/main/java/com/provectus/kafka/ui/controller/PreferencesController.java b/kafka-ui-api/src/main/java/com/provectus/kafka/ui/controller/PreferencesController.java new file mode 100644 index 00000000000..a0ac96cc02c --- /dev/null +++ b/kafka-ui-api/src/main/java/com/provectus/kafka/ui/controller/PreferencesController.java @@ -0,0 +1,47 @@ +package com.provectus.kafka.ui.controller; + +import com.provectus.kafka.ui.api.PreferencesApi; +import com.provectus.kafka.ui.model.ApplicationsPreferencesDTO; +import lombok.RequiredArgsConstructor; +import lombok.extern.slf4j.Slf4j; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.http.ResponseEntity; +import org.springframework.web.bind.annotation.RestController; +import org.springframework.web.server.ServerWebExchange; +import reactor.core.publisher.Mono; + +@RestController +@RequiredArgsConstructor +@Slf4j +public class PreferencesController implements PreferencesApi { + + @Value("${kafka.ui.preferences.removeGitlink}") + private boolean kafkaUiRemoveGitLink; + @Value("${kafka.ui.preferences.removeDiscordLink}") + private boolean kafkaUiRemovDiscordLink; + @Value("${kafka.ui.preferences.appName}") + private String kafkaUiAppName; + @Value("${kafka.ui.preferences.favicon}") + private String kafkaUiFavicon; + @Value("${kafka.ui.preferences.logo}") + private String kafkaUiIcon; + + @Value("${kafka.ui.preferences.version}") + private boolean kafkaUiVersion; + + + @Override + public Mono> getPreferences(ServerWebExchange exchange) { + ApplicationsPreferencesDTO res = new ApplicationsPreferencesDTO(); + res.setAppName(kafkaUiAppName); + res.setRemoveDiscordLink(kafkaUiRemoveGitLink); + res.setRemoveGitLink(kafkaUiRemovDiscordLink); + res.setLogo(kafkaUiIcon); + res.setFavicon(kafkaUiFavicon); + res.setVersion(kafkaUiVersion); + return Mono.just( + ResponseEntity.ok(res + ) + ); + } +} diff --git a/kafka-ui-api/src/main/resources/application.yml b/kafka-ui-api/src/main/resources/application.yml index e8799206132..0fc169ca271 100644 --- a/kafka-ui-api/src/main/resources/application.yml +++ b/kafka-ui-api/src/main/resources/application.yml @@ -19,3 +19,13 @@ logging: reactor.netty.http.server.AccessLog: INFO org.hibernate.validator: WARN +kafka: + ui: + preferences: + removeGitlink: ${KAFKA_UI_PREFERENCES_REMOVEGITLINK:false} + removeDiscordLink: ${KAFKA_UI_PREFERENCES_REMOVEDISCORDLINK:false} + appName: ${KAFKA_UI_PREFERENCES_APPNAME:My kafka ui application} + favicon: ${KAFKA_UI_PREFERENCES_FAVICON:} + logo: ${KAFKA_UI_PREFERENCES_LOGO:} + version: ${KAFKA_UI_PREFERENCES_VERSION:false} + diff --git a/kafka-ui-contract/src/main/resources/swagger/kafka-ui-api.yaml b/kafka-ui-contract/src/main/resources/swagger/kafka-ui-api.yaml index ae51d31568f..fb03f8dc599 100644 --- a/kafka-ui-contract/src/main/resources/swagger/kafka-ui-api.yaml +++ b/kafka-ui-contract/src/main/resources/swagger/kafka-ui-api.yaml @@ -15,6 +15,19 @@ servers: - url: /localhost paths: + /api/preferences: + get: + tags: + - Preferences + summary: getPreferences + operationId: getPreferences + responses: + 200: + description: OK + content: + application/json: + schema: + $ref: '#/components/schemas/ApplicationsPreferences' /api/clusters: get: tags: @@ -2058,6 +2071,26 @@ paths: components: schemas: + + ApplicationsPreferences: + type: object + properties: + removeGitLink: + type: boolean + removeDiscordLink: + type: boolean + appName: + type: string + example: "App Kafka UI My Preferences" + favicon: + type: string + example: "Base64 faicon" + logo: + type: string + example: "Base 64 icon" + version: + type: boolean + example: "Show version info" TopicSerdeSuggestion: type: object properties: @@ -3968,7 +4001,7 @@ components: keystoreLocation: type: string keystorePassword: - type: string + type: string ksqldbServerAuth: type: object properties: diff --git a/kafka-ui-react-app/index.html b/kafka-ui-react-app/index.html index be10fc78a3a..309ecdca21d 100644 --- a/kafka-ui-react-app/index.html +++ b/kafka-ui-react-app/index.html @@ -1,57 +1,73 @@ - - - + + + - - - - - + + + + + - UI for Apache Kafka - + + - @font-face { - font-family: 'Roboto Mono'; - src: url('<%= PUBLIC_PATH %>/fonts/RobotoMono-Regular.ttf') format('truetype'); - font-weight: 400; - font-display: swap; - } - - - - - -
- - + + +
+ + diff --git a/kafka-ui-react-app/src/components/NavBar/NavBar.tsx b/kafka-ui-react-app/src/components/NavBar/NavBar.tsx index 4744eb0bffe..a75520eed66 100644 --- a/kafka-ui-react-app/src/components/NavBar/NavBar.tsx +++ b/kafka-ui-react-app/src/components/NavBar/NavBar.tsx @@ -1,4 +1,4 @@ -import React, { useContext } from 'react'; +import React, { useContext, useState, useEffect } from 'react'; import Select from 'components/common/Select/Select'; import Logo from 'components/common/Logo/Logo'; import Version from 'components/Version/Version'; @@ -11,6 +11,7 @@ import { ThemeModeContext } from 'components/contexts/ThemeModeContext'; import UserInfo from './UserInfo/UserInfo'; import * as S from './NavBar.styled'; +import { preferencesClient as api } from 'lib/api'; interface Props { onBurgerClick: () => void; @@ -50,6 +51,31 @@ const options = [ const NavBar: React.FC = ({ onBurgerClick }) => { const { themeMode, setThemeMode } = useContext(ThemeModeContext); + const [appName, setAppName] = useState(""); + const [showGitHubLink, setShowGitHubLink] = useState(true); + const [showDiscordLink, setShowDiscordLink] = useState(true); + const [showVersion, setShowVersion] = useState(true); + const [logoBase64, setLogoBase64] = useState(""); + + + useEffect(() => { + const fetchPreferences = async () => { + try { + const preferencesData = await api.getPreferences(); + setAppName(preferencesData.appName); + setShowGitHubLink(preferencesData.removeGitLink); + setShowDiscordLink(preferencesData.removeDiscordLink); + setShowVersion(preferencesData.version); + if (preferencesData.logo) { + setLogoBase64(preferencesData.logo); + } + } catch (error) { + console.error('Error fetching preferences:', error); + } + }; + + fetchPreferences(); + }, []); return ( @@ -68,13 +94,22 @@ const NavBar: React.FC = ({ onBurgerClick }) => { - - UI for Apache Kafka + {logoBase64 ? ( + Logo + ) : ( + + )} + {appName ? ( + {appName} + ) : ( + "UI for Apache Kafka" + )} - + {showVersion && ( + )} @@ -84,18 +119,22 @@ const NavBar: React.FC = ({ onBurgerClick }) => { onChange={setThemeMode} isThemeMode /> - - - + {showGitHubLink && ( + + + + )} + {showDiscordLink && ( + )} diff --git a/kafka-ui-react-app/src/components/Version/Version.tsx b/kafka-ui-react-app/src/components/Version/Version.tsx index 775fbfa5d58..3d29bf71687 100644 --- a/kafka-ui-react-app/src/components/Version/Version.tsx +++ b/kafka-ui-react-app/src/components/Version/Version.tsx @@ -1,23 +1,40 @@ -import React from 'react'; +import React, { useState, useEffect } from 'react'; import WarningIcon from 'components/common/Icons/WarningIcon'; import { gitCommitPath } from 'lib/paths'; import { useLatestVersion } from 'lib/hooks/api/latestVersion'; import { formatTimestamp } from 'lib/dateTimeHelpers'; - +import { PreferencesApi } from '../../generated-sources'; import * as S from './Version.styled'; +import { preferencesClient as api } from 'lib/api'; const Version: React.FC = () => { - const { data: latestVersionInfo = {} } = useLatestVersion(); - const { buildTime, commitId, isLatestRelease, version } = - latestVersionInfo.build; - const { versionTag } = latestVersionInfo?.latestRelease || ''; + const [showVersion, setShowVersion] = useState(true); + const [latestVersionInfo, setLatestVersionInfo] = useState({}); + + useEffect(() => { + const fetchLatestVersion = async () => { + try { + const preferencesData = await api.getPreferences(); + const latestVersionInfo = preferencesData?.version || {}; + setLatestVersionInfo(latestVersionInfo); + setShowVersion(preferencesData?.version || true); // Set showVersion based on the preferences data + } catch (error) { + console.error('Error fetching latest version:', error); + setShowVersion(false); // Set showVersion to false in case of error + } + }; + + fetchLatestVersion(); + }, []); + + const { buildTime, commitId, isLatestRelease, version, versionTag } = latestVersionInfo; const currentVersion = isLatestRelease && version?.match(versionTag) ? versionTag : formatTimestamp(buildTime); - return ( + return showVersion ? ( {!isLatestRelease && ( { )} {currentVersion} - ); + ) : null; }; export default Version; diff --git a/kafka-ui-react-app/src/lib/api.ts b/kafka-ui-react-app/src/lib/api.ts index 19423d2ac3e..58ff9d4a3c8 100644 --- a/kafka-ui-react-app/src/lib/api.ts +++ b/kafka-ui-react-app/src/lib/api.ts @@ -11,6 +11,7 @@ import { AuthorizationApi, ApplicationConfigApi, AclsApi, + PreferencesApi } from 'generated-sources'; import { BASE_PARAMS } from 'lib/constants'; @@ -27,3 +28,4 @@ export const consumerGroupsApiClient = new ConsumerGroupsApi(apiClientConf); export const authApiClient = new AuthorizationApi(apiClientConf); export const appConfigApiClient = new ApplicationConfigApi(apiClientConf); export const aclApiClient = new AclsApi(apiClientConf); +export const preferencesClient = new PreferencesApi(apiClientConf);