From 3d666c6611ea805b7ce726a1524a4a5cc64394ed Mon Sep 17 00:00:00 2001
From: Nikolay Shestakov <tesseract@ydb.tech>
Date: Wed, 5 Mar 2025 14:57:30 +0500
Subject: [PATCH 01/10] WIP

---
 package-lock.json                             |  8 +-
 package.json                                  |  2 +-
 .../Tenant/Diagnostics/DiagnosticsPages.ts    |  3 +
 .../Tenant/Diagnostics/Overview/Overview.tsx  |  2 +
 .../Overview/TransferInfo/Credentials.tsx     | 27 +++++++
 .../Overview/TransferInfo/TransferInfo.tsx    | 78 +++++++++++++++++++
 .../Overview/TransferInfo/i18n/en.json        |  7 ++
 .../Overview/TransferInfo/i18n/index.ts       |  7 ++
 .../Overview/TransferInfo/index.ts            |  1 +
 .../Tenant/ObjectSummary/ObjectSummary.tsx    | 14 ++++
 src/containers/Tenant/Query/NewSQL/NewSQL.tsx | 17 ++++
 .../Tenant/Query/NewSQL/i18n/en.json          |  6 +-
 src/containers/Tenant/i18n/en.json            |  3 +
 src/containers/Tenant/utils/controls.tsx      |  1 +
 .../Tenant/utils/newSQLQueryActions.ts        |  6 ++
 src/containers/Tenant/utils/schema.ts         |  7 ++
 src/containers/Tenant/utils/schemaActions.tsx | 19 +++++
 .../Tenant/utils/schemaQueryTemplates.ts      | 50 ++++++++++++
 src/types/api/schema/schema.ts                |  1 +
 19 files changed, 253 insertions(+), 6 deletions(-)
 create mode 100644 src/containers/Tenant/Diagnostics/Overview/TransferInfo/Credentials.tsx
 create mode 100644 src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx
 create mode 100644 src/containers/Tenant/Diagnostics/Overview/TransferInfo/i18n/en.json
 create mode 100644 src/containers/Tenant/Diagnostics/Overview/TransferInfo/i18n/index.ts
 create mode 100644 src/containers/Tenant/Diagnostics/Overview/TransferInfo/index.ts

diff --git a/package-lock.json b/package-lock.json
index 3dc345fcf..b17d7284b 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -56,7 +56,7 @@
         "use-query-params": "^2.2.1",
         "uuid": "^10.0.0",
         "web-vitals": "^1.1.2",
-        "ydb-ui-components": "^4.5.0",
+        "ydb-ui-components": "^4.6.0",
         "zod": "^3.24.1"
       },
       "devDependencies": {
@@ -29305,9 +29305,9 @@
       }
     },
     "node_modules/ydb-ui-components": {
-      "version": "4.5.0",
-      "resolved": "https://registry.npmjs.org/ydb-ui-components/-/ydb-ui-components-4.5.0.tgz",
-      "integrity": "sha512-XefBB6dWHXbtyZ9jaxkmCJM5tl19V6VYrJ0nB2WcERCbb8ZG0GYJgywi5BBqQgOL8wH1Z5iut8WX7sQPcj+WNA==",
+      "version": "4.6.0",
+      "resolved": "https://registry.npmjs.org/ydb-ui-components/-/ydb-ui-components-4.6.0.tgz",
+      "integrity": "sha512-9gkD6KBda3Jo/zt8SpTgrmUckXP6gophDiL4E41CKjORCmi5umT5C7D/D98awzaB+ZltVL0L5ySdnZKizfD9Yw==",
       "license": "MIT",
       "dependencies": {
         "@bem-react/classname": "^1.6.0",
diff --git a/package.json b/package.json
index 8dc2e7ca0..d8903aaad 100644
--- a/package.json
+++ b/package.json
@@ -58,7 +58,7 @@
     "use-query-params": "^2.2.1",
     "uuid": "^10.0.0",
     "web-vitals": "^1.1.2",
-    "ydb-ui-components": "^4.5.0",
+    "ydb-ui-components": "^4.6.0",
     "zod": "^3.24.1"
   },
   "scripts": {
diff --git a/src/containers/Tenant/Diagnostics/DiagnosticsPages.ts b/src/containers/Tenant/Diagnostics/DiagnosticsPages.ts
index 96bb6b46d..9ea138ce8 100644
--- a/src/containers/Tenant/Diagnostics/DiagnosticsPages.ts
+++ b/src/containers/Tenant/Diagnostics/DiagnosticsPages.ts
@@ -82,6 +82,8 @@ const operations = {
 
 const ASYNC_REPLICATION_PAGES = [overview, tablets, describe];
 
+const TRANSFER_PAGES = [overview, tablets, describe];
+
 const DATABASE_PAGES = [
     overview,
     topQueries,
@@ -133,6 +135,7 @@ const pathTypeToPages: Record<EPathType, Page[] | undefined> = {
     [EPathType.EPathTypeView]: VIEW_PAGES,
 
     [EPathType.EPathTypeReplication]: ASYNC_REPLICATION_PAGES,
+    [EPathType.EPathTypeTransfer]: TRANSFER_PAGES,
     [EPathType.EPathTypeResourcePool]: DIR_PAGES,
 };
 
diff --git a/src/containers/Tenant/Diagnostics/Overview/Overview.tsx b/src/containers/Tenant/Diagnostics/Overview/Overview.tsx
index 762eca8ea..18feb9d39 100644
--- a/src/containers/Tenant/Diagnostics/Overview/Overview.tsx
+++ b/src/containers/Tenant/Diagnostics/Overview/Overview.tsx
@@ -20,6 +20,7 @@ import {AsyncReplicationInfo} from './AsyncReplicationInfo';
 import {ChangefeedInfo} from './ChangefeedInfo';
 import {TableInfo} from './TableInfo';
 import {TopicInfo} from './TopicInfo';
+import {TransferInfo} from './TransferInfo';
 
 interface OverviewProps {
     type?: EPathType;
@@ -94,6 +95,7 @@ function Overview({type, path, database}: OverviewProps) {
             [EPathType.EPathTypeExternalDataSource]: () => <ExternalDataSourceInfo data={data} />,
             [EPathType.EPathTypeView]: () => <ViewInfo data={data} />,
             [EPathType.EPathTypeReplication]: () => <AsyncReplicationInfo data={data} />,
+            [EPathType.EPathTypeTransfer]: () => <TransferInfo data={data} />,
         };
 
         return (type && pathTypeToComponent[type]?.()) || <TableInfo data={data} type={type} />;
diff --git a/src/containers/Tenant/Diagnostics/Overview/TransferInfo/Credentials.tsx b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/Credentials.tsx
new file mode 100644
index 000000000..3a6839a33
--- /dev/null
+++ b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/Credentials.tsx
@@ -0,0 +1,27 @@
+import {Label} from '@gravity-ui/uikit';
+
+import type {TConnectionParams} from '../../../../../types/api/schema/replication';
+
+interface CredentialsProps {
+    connection?: TConnectionParams;
+}
+
+export function Credentials({connection}: CredentialsProps) {
+    if (!connection) {
+        return null;
+    }
+
+    if (connection.StaticCredentials) {
+        return (
+            <Label value={connection.StaticCredentials.User} theme="normal">
+                user
+            </Label>
+        );
+    }
+
+    if ('OAuthToken' in connection) {
+        return 'OAuth';
+    }
+
+    return 'unknown';
+}
diff --git a/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx
new file mode 100644
index 000000000..fc2487281
--- /dev/null
+++ b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx
@@ -0,0 +1,78 @@
+import type {DefinitionListItem} from '@gravity-ui/components';
+import {Flex, Text} from '@gravity-ui/uikit';
+
+import {AsyncReplicationState} from '../../../../../components/AsyncReplicationState';
+import {YDBDefinitionList} from '../../../../../components/YDBDefinitionList/YDBDefinitionList';
+import type {TEvDescribeSchemeResult} from '../../../../../types/api/schema';
+import {getEntityName} from '../../../utils';
+import {AsyncReplicationPaths} from '../AsyncReplicationPaths';
+
+import {Credentials} from './Credentials';
+import i18n from './i18n';
+
+interface TransferProps {
+    data?: TEvDescribeSchemeResult;
+}
+
+/** Displays overview for Transfer EPathType */
+export function TransferInfo({data}: TransferProps) {
+    const entityName = getEntityName(data?.PathDescription);
+
+    if (!data) {
+        return (
+            <div className="error">
+                {i18n('noData')} {entityName}
+            </div>
+        );
+    }
+
+    const transferItems = prepareTransferItems(data);
+
+    return (
+        <Flex direction="column" gap="4">
+            <YDBDefinitionList title={entityName} items={transferItems} />
+            <AsyncReplicationPaths config={data.PathDescription?.ReplicationDescription?.Config} />
+        </Flex>
+    );
+}
+
+function prepareTransferItems(data: TEvDescribeSchemeResult) {
+    const transferDescription = data.PathDescription?.ReplicationDescription || {};
+    const state = transferDescription.State;
+    const srcConnectionParams = transferDescription.Config?.SrcConnectionParams || {};
+    const {Endpoint, Database} = srcConnectionParams;
+
+    const info: DefinitionListItem[] = [];
+
+    if (state) {
+        info.push({
+            name: i18n('state.label'),
+            content: <AsyncReplicationState state={state} />,
+        });
+    }
+
+    if (Endpoint) {
+        info.push({
+            name: i18n('srcConnection.endpoint.label'),
+            copyText: Endpoint,
+            content: <Text variant="code-inline-2">{Endpoint}</Text>,
+        });
+    }
+
+    if (Database) {
+        info.push({
+            name: i18n('srcConnection.database.label'),
+            copyText: Database,
+            content: <Text variant="code-inline-2">{Database}</Text>,
+        });
+    }
+
+    if (srcConnectionParams) {
+        info.push({
+            name: i18n('credentials.label'),
+            content: <Credentials connection={srcConnectionParams} />,
+        });
+    }
+
+    return info;
+}
diff --git a/src/containers/Tenant/Diagnostics/Overview/TransferInfo/i18n/en.json b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/i18n/en.json
new file mode 100644
index 000000000..ecd7f0044
--- /dev/null
+++ b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/i18n/en.json
@@ -0,0 +1,7 @@
+{
+  "credentials.label": "Credentials",
+  "noData": "No data for entity:",
+  "srcConnection.database.label": "Source Database Path",
+  "srcConnection.endpoint.label": "Source Cluster Endpoint",
+  "state.label": "State"
+}
diff --git a/src/containers/Tenant/Diagnostics/Overview/TransferInfo/i18n/index.ts b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/i18n/index.ts
new file mode 100644
index 000000000..83f1ad759
--- /dev/null
+++ b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/i18n/index.ts
@@ -0,0 +1,7 @@
+import {registerKeysets} from '../../../../../../utils/i18n';
+
+import en from './en.json';
+
+const COMPONENT = 'ydb-diagnostics-transfer-info';
+
+export default registerKeysets(COMPONENT, {en});
diff --git a/src/containers/Tenant/Diagnostics/Overview/TransferInfo/index.ts b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/index.ts
new file mode 100644
index 000000000..0206c28be
--- /dev/null
+++ b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/index.ts
@@ -0,0 +1 @@
+export * from './TransferInfo';
diff --git a/src/containers/Tenant/ObjectSummary/ObjectSummary.tsx b/src/containers/Tenant/ObjectSummary/ObjectSummary.tsx
index 301638bae..f1a05e509 100644
--- a/src/containers/Tenant/ObjectSummary/ObjectSummary.tsx
+++ b/src/containers/Tenant/ObjectSummary/ObjectSummary.tsx
@@ -310,6 +310,20 @@ export function ObjectSummary({
                     return [];
                 }
 
+                return [
+                    {
+                        name: i18n('field_state'),
+                        content: <AsyncReplicationState state={state} />,
+                    },
+                ];
+            },
+            [EPathType.EPathTypeTransfer]: () => {
+                const state = PathDescription?.ReplicationDescription?.State;
+
+                if (!state) {
+                    return [];
+                }
+
                 return [
                     {
                         name: i18n('field_state'),
diff --git a/src/containers/Tenant/Query/NewSQL/NewSQL.tsx b/src/containers/Tenant/Query/NewSQL/NewSQL.tsx
index d772ded26..b48028eb7 100644
--- a/src/containers/Tenant/Query/NewSQL/NewSQL.tsx
+++ b/src/containers/Tenant/Query/NewSQL/NewSQL.tsx
@@ -106,6 +106,23 @@ export function NewSQL() {
                 },
             ],
         },
+        {
+            text: i18n('menu.transfer'),
+            items: [
+                {
+                    text: i18n('action.create-transfer'),
+                    action: actions.createTransfer,
+                },
+                {
+                    text: i18n('action.alter-transfer'),
+                    action: actions.alterTransfer,
+                },
+                {
+                    text: i18n('action.drop-transfer'),
+                    action: actions.dropTransfer,
+                },
+            ],
+        },
         {
             text: i18n('menu.capture'),
             items: [
diff --git a/src/containers/Tenant/Query/NewSQL/i18n/en.json b/src/containers/Tenant/Query/NewSQL/i18n/en.json
index dface2060..9e6916686 100644
--- a/src/containers/Tenant/Query/NewSQL/i18n/en.json
+++ b/src/containers/Tenant/Query/NewSQL/i18n/en.json
@@ -16,12 +16,14 @@
   "menu.topics": "Topics",
   "menu.capture": "Change data capture",
   "menu.replication": "Async replication",
+  "menu.transfer": "Transfer",
   "menu.users": "Users",
   "action.create-topic": "Create Topic",
   "action.drop-topic": "Drop Topic",
   "action.alter-topic": "Alter Topic",
   "action.create-cdc-stream": "Create changefeed",
   "action.create-async-replication": "Create async replication",
+  "action.create-transfer": "Create transfer",
   "action.create-user": "Create user",
   "action.create-group": "Create group",
   "action.drop-user": "Drop user",
@@ -29,5 +31,7 @@
   "action.grant-privilege": "Grant privilege",
   "action.revoke-privilege": "Revoke privilege",
   "action.alter-async-replication": "Alter async replication",
-  "action.drop-async-replication": "Drop async replication"
+  "action.drop-async-replication": "Drop async replication",
+  "action.alter-transfer": "Alter transfer",
+  "action.drop-transfer": "Drop transfer"
 }
diff --git a/src/containers/Tenant/i18n/en.json b/src/containers/Tenant/i18n/en.json
index 399969a2e..a7d9313e4 100644
--- a/src/containers/Tenant/i18n/en.json
+++ b/src/containers/Tenant/i18n/en.json
@@ -33,6 +33,7 @@
   "actions.createTopic": "Create topic...",
   "actions.createColumnTable": "Create column table...",
   "actions.createAsyncReplication": "Create async replication...",
+  "actions.createTransfer": "Create transfer...",
   "actions.createView": "Create view...",
   "actions.dropTable": "Drop table...",
   "actions.dropTopic": "Drop topic...",
@@ -46,7 +47,9 @@
   "actions.selectQuery": "Select query...",
   "actions.upsertQuery": "Upsert query...",
   "actions.alterReplication": "Alter async replicaton...",
+  "actions.alterTransfer": "Alter transfer...",
   "actions.dropReplication": "Drop async replicaton...",
+  "actions.dropTransfer": "Drop transfer...",
   "actions.createDirectory": "Create directory",
   "schema.tree.dialog.placeholder": "Relative path",
   "schema.tree.dialog.invalid": "Invalid path",
diff --git a/src/containers/Tenant/utils/controls.tsx b/src/containers/Tenant/utils/controls.tsx
index dc6a3c5d4..5e4f2a656 100644
--- a/src/containers/Tenant/utils/controls.tsx
+++ b/src/containers/Tenant/utils/controls.tsx
@@ -61,6 +61,7 @@ export const getSchemaControls =
 
         const nodeTypeToControls: Record<NavigationTreeNodeType, Controls> = {
             async_replication: undefined,
+            transfer: undefined,
 
             database: undefined,
             directory: undefined,
diff --git a/src/containers/Tenant/utils/newSQLQueryActions.ts b/src/containers/Tenant/utils/newSQLQueryActions.ts
index 9baf9dbb3..ed1526c24 100644
--- a/src/containers/Tenant/utils/newSQLQueryActions.ts
+++ b/src/containers/Tenant/utils/newSQLQueryActions.ts
@@ -3,6 +3,7 @@ import {
     alterAsyncReplicationTemplate,
     alterTableTemplate,
     alterTopicTemplate,
+    alterTransferTemplate,
     createAsyncReplicationTemplate,
     createCdcStreamTemplate,
     createColumnTableTemplate,
@@ -10,6 +11,7 @@ import {
     createGroupTemplate,
     createTableTemplate,
     createTopicTemplate,
+    createTransferTemplate,
     createUserTemplate,
     createViewTemplate,
     deleteRowsTemplate,
@@ -19,6 +21,7 @@ import {
     dropTableIndex,
     dropTableTemplate,
     dropTopicTemplate,
+    dropTransferTemplate,
     dropUserTemplate,
     grantPrivilegeTemplate,
     revokePrivilegeTemplate,
@@ -38,6 +41,9 @@ export const bindActions = (changeUserInput: (input: string) => void) => {
         createAsyncReplication: inputQuery(createAsyncReplicationTemplate),
         alterAsyncReplication: inputQuery(alterAsyncReplicationTemplate),
         dropAsyncReplication: inputQuery(dropAsyncReplicationTemplate),
+        createTransfer: inputQuery(createTransferTemplate),
+        alterTransfer: inputQuery(alterTransferTemplate),
+        dropTransfer: inputQuery(dropTransferTemplate),
         alterTable: inputQuery(alterTableTemplate),
         selectQuery: inputQuery(selectQueryTemplate),
         upsertQuery: inputQuery(upsertQueryTemplate),
diff --git a/src/containers/Tenant/utils/schema.ts b/src/containers/Tenant/utils/schema.ts
index b468b724a..aba1549f0 100644
--- a/src/containers/Tenant/utils/schema.ts
+++ b/src/containers/Tenant/utils/schema.ts
@@ -39,6 +39,7 @@ const pathTypeToNodeType: Record<EPathType, NavigationTreeNodeType | undefined>
     [EPathType.EPathTypeView]: 'view',
 
     [EPathType.EPathTypeReplication]: 'async_replication',
+    [EPathType.EPathTypeTransfer]: 'transfer',
     [EPathType.EPathTypeResourcePool]: 'resource_pool',
 };
 
@@ -87,6 +88,7 @@ const pathTypeToEntityName: Record<EPathType, string | undefined> = {
     [EPathType.EPathTypeView]: 'View',
 
     [EPathType.EPathTypeReplication]: 'Async Replication',
+    [EPathType.EPathTypeTransfer]: 'Transfer',
     [EPathType.EPathTypeResourcePool]: 'Resource Pool',
 };
 
@@ -128,6 +130,7 @@ const pathTypeToIsTable: Record<EPathType, boolean> = {
     [EPathType.EPathTypePersQueueGroup]: false,
     [EPathType.EPathTypeExternalDataSource]: false,
     [EPathType.EPathTypeReplication]: false,
+    [EPathType.EPathTypeTransfer]: false,
     [EPathType.EPathTypeResourcePool]: false,
 };
 
@@ -169,6 +172,7 @@ const pathTypeToIsColumn: Record<EPathType, boolean> = {
     [EPathType.EPathTypeView]: false,
 
     [EPathType.EPathTypeReplication]: false,
+    [EPathType.EPathTypeTransfer]: false,
     [EPathType.EPathTypeResourcePool]: false,
 };
 
@@ -195,6 +199,7 @@ const pathTypeToIsDatabase: Record<EPathType, boolean> = {
     [EPathType.EPathTypeView]: false,
 
     [EPathType.EPathTypeReplication]: false,
+    [EPathType.EPathTypeTransfer]: false,
     [EPathType.EPathTypeResourcePool]: false,
 };
 
@@ -226,6 +231,7 @@ const pathTypeToEntityWithMergedImplementation: Record<EPathType, boolean> = {
     [EPathType.EPathTypeView]: false,
 
     [EPathType.EPathTypeReplication]: false,
+    [EPathType.EPathTypeTransfer]: false,
     [EPathType.EPathTypeResourcePool]: false,
 };
 
@@ -253,6 +259,7 @@ const pathTypeToChildless: Record<EPathType, boolean> = {
     [EPathType.EPathTypeResourcePool]: true,
 
     [EPathType.EPathTypeReplication]: true,
+    [EPathType.EPathTypeTransfer]: true,
 
     [EPathType.EPathTypeInvalid]: false,
     [EPathType.EPathTypeColumnStore]: false,
diff --git a/src/containers/Tenant/utils/schemaActions.tsx b/src/containers/Tenant/utils/schemaActions.tsx
index 60c0f4e18..52857ff2d 100644
--- a/src/containers/Tenant/utils/schemaActions.tsx
+++ b/src/containers/Tenant/utils/schemaActions.tsx
@@ -20,18 +20,21 @@ import {
     alterAsyncReplicationTemplate,
     alterTableTemplate,
     alterTopicTemplate,
+    alterTransferTemplate,
     createAsyncReplicationTemplate,
     createCdcStreamTemplate,
     createColumnTableTemplate,
     createExternalTableTemplate,
     createTableTemplate,
     createTopicTemplate,
+    createTransferTemplate,
     createViewTemplate,
     dropAsyncReplicationTemplate,
     dropExternalTableTemplate,
     dropTableIndex,
     dropTableTemplate,
     dropTopicTemplate,
+    dropTransferTemplate,
     dropViewTemplate,
     manageAutoPartitioningTemplate,
     selectQueryTemplate,
@@ -100,6 +103,9 @@ const bindActions = (
         createAsyncReplication: inputQuery(createAsyncReplicationTemplate),
         alterAsyncReplication: inputQuery(alterAsyncReplicationTemplate),
         dropAsyncReplication: inputQuery(dropAsyncReplicationTemplate),
+        createTransfer: inputQuery(createTransferTemplate),
+        alterTransfer: inputQuery(alterTransferTemplate),
+        dropTransfer: inputQuery(dropTransferTemplate),
         alterTable: inputQuery(alterTableTemplate),
         dropTable: inputQuery(dropTableTemplate),
         manageAutoPartitioning: inputQuery(manageAutoPartitioningTemplate),
@@ -181,6 +187,10 @@ export const getActions =
                 text: i18n('actions.createAsyncReplication'),
                 action: actions.createAsyncReplication,
             },
+            {
+                text: i18n('actions.createTransfer'),
+                action: actions.createTransfer,
+            },
             {text: i18n('actions.createTopic'), action: actions.createTopic},
             {text: i18n('actions.createView'), action: actions.createView},
         ];
@@ -276,6 +286,14 @@ export const getActions =
             ],
         ];
 
+        const TRANSFER_SET: ActionsSet = [
+            [copyItem],
+            [
+                {text: i18n('actions.alterTransfer'), action: actions.alterTransfer},
+                {text: i18n('actions.dropTransfer'), action: actions.dropTransfer},
+            ],
+        ];
+
         const INDEX_SET: ActionsSet = [
             [copyItem, {text: i18n('actions.dropIndex'), action: actions.dropIndex}],
         ];
@@ -286,6 +304,7 @@ export const getActions =
         // TS will error when a new type is added in the lib but is not mapped here
         const nodeTypeToActions: Record<NavigationTreeNodeType, ActionsSet> = {
             async_replication: ASYNC_REPLICATION_SET,
+            transfer: TRANSFER_SET,
 
             database: DB_SET,
 
diff --git a/src/containers/Tenant/utils/schemaQueryTemplates.ts b/src/containers/Tenant/utils/schemaQueryTemplates.ts
index 93e692a2e..b54796c80 100644
--- a/src/containers/Tenant/utils/schemaQueryTemplates.ts
+++ b/src/containers/Tenant/utils/schemaQueryTemplates.ts
@@ -78,6 +78,30 @@ WITH (
     -- PASSWORD_SECRET_NAME="your_password"
 );`;
 };
+export const createTransferTemplate = () => {
+    return `-- docs: https://ydb.tech/docs/en/yql/reference/syntax/create-transfer
+CREATE OBJECT secret_name (TYPE SECRET) WITH value="secret_value";
+
+$l = ($x) -> {
+    return [
+        <|
+            offset:$x._offset,
+            message:$x._data
+        |>
+    ];
+};
+
+CREATE TRANSFER my_transfer
+FROM \${1:<original_topic>} TO \${2:target_table} USING $l
+WITH (
+    CONNECTION_STRING="\${3:grpcs://mydb.ydb.tech:2135/?database=/remote_database}",
+    TOKEN_SECRET_NAME = "secret_name"
+    -- ENDPOINT="mydb.ydb.tech:2135",
+    -- DATABASE=\`/remote_database\`,
+    -- USER="user",
+    -- PASSWORD_SECRET_NAME="your_password"
+);`;
+};
 export const alterTableTemplate = (params?: SchemaQueryParams) => {
     const path = params?.relativePath
         ? `\`${normalizeParameter(params.relativePath)}\``
@@ -240,6 +264,13 @@ export const dropAsyncReplicationTemplate = (params?: SchemaQueryParams) => {
     return `DROP ASYNC REPLICATION ${path};`;
 };
 
+export const dropTransferTemplate = (params?: SchemaQueryParams) => {
+    const path = params?.relativePath
+        ? `\`${normalizeParameter(params.relativePath)}\``
+        : '${1:<my_transfer>}';
+    return `DROP TRANSFER ${path};`;
+};
+
 export const alterAsyncReplicationTemplate = (params?: SchemaQueryParams) => {
     const path = params?.relativePath
         ? `\`${normalizeParameter(params.relativePath)}\``
@@ -248,6 +279,25 @@ export const alterAsyncReplicationTemplate = (params?: SchemaQueryParams) => {
 ALTER ASYNC REPLICATION ${path} SET (STATE = "DONE", FAILOVER_MODE = "FORCE");`;
 };
 
+export const alterTransferTemplate = (params?: SchemaQueryParams) => {
+    const path = params?.relativePath
+        ? `\`${normalizeParameter(params.relativePath)}\``
+        : '${1:<my_transfer>}';
+    return `-- docs: https://ydb.tech/docs/en/yql/reference/syntax/alter-transfer
+
+$l = ($x) -> {
+    return [
+        <|
+            offset:$x._offset,
+            message:$x._data
+        |>
+    ];
+};
+
+ALTER TRANSFER ${path} 
+SET USING $l;`;
+};
+
 export const addTableIndex = (params?: SchemaQueryParams) => {
     const path = params?.relativePath
         ? `\`${normalizeParameter(params.relativePath)}\``
diff --git a/src/types/api/schema/schema.ts b/src/types/api/schema/schema.ts
index 0742d8601..4d50a15e0 100644
--- a/src/types/api/schema/schema.ts
+++ b/src/types/api/schema/schema.ts
@@ -297,6 +297,7 @@ export enum EPathType {
     EPathTypeView = 'EPathTypeView',
 
     EPathTypeReplication = 'EPathTypeReplication',
+    EPathTypeTransfer = 'EPathTypeTransfer',
     EPathTypeResourcePool = 'EPathTypeResourcePool',
 }
 

From 17dbc1b104c2567ab74411b7d0893c41d1678ac4 Mon Sep 17 00:00:00 2001
From: Nikolay Shestakov <tesseract@ydb.tech>
Date: Wed, 5 Mar 2025 15:52:00 +0500
Subject: [PATCH 02/10] WIP

---
 .../Overview/TransferInfo/TransferInfo.tsx    | 28 +++++++++++++++++--
 .../Overview/TransferInfo/i18n/en.json        |  5 +++-
 src/types/api/schema/replication.ts           | 11 ++++++++
 3 files changed, 41 insertions(+), 3 deletions(-)

diff --git a/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx
index fc2487281..d7846a5a0 100644
--- a/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx
+++ b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx
@@ -5,7 +5,6 @@ import {AsyncReplicationState} from '../../../../../components/AsyncReplicationS
 import {YDBDefinitionList} from '../../../../../components/YDBDefinitionList/YDBDefinitionList';
 import type {TEvDescribeSchemeResult} from '../../../../../types/api/schema';
 import {getEntityName} from '../../../utils';
-import {AsyncReplicationPaths} from '../AsyncReplicationPaths';
 
 import {Credentials} from './Credentials';
 import i18n from './i18n';
@@ -31,7 +30,6 @@ export function TransferInfo({data}: TransferProps) {
     return (
         <Flex direction="column" gap="4">
             <YDBDefinitionList title={entityName} items={transferItems} />
-            <AsyncReplicationPaths config={data.PathDescription?.ReplicationDescription?.Config} />
         </Flex>
     );
 }
@@ -41,6 +39,10 @@ function prepareTransferItems(data: TEvDescribeSchemeResult) {
     const state = transferDescription.State;
     const srcConnectionParams = transferDescription.Config?.SrcConnectionParams || {};
     const {Endpoint, Database} = srcConnectionParams;
+    const target = transferDescription.Config?.TransferSpecific?.Targets[0];
+    const srcPath = target?.SrcPath;
+    const dstPath = target?.DstPath;
+    const transformLambda = target?.TransformLambda;
 
     const info: DefinitionListItem[] = [];
 
@@ -74,5 +76,27 @@ function prepareTransferItems(data: TEvDescribeSchemeResult) {
         });
     }
 
+    info.push({
+        name: i18n('srcPath.label'),
+        copyText: srcPath,
+        content: <Text variant="code-inline-2">{srcPath}</Text>,
+    });
+
+    info.push({
+        name: i18n('dstPath.label'),
+        copyText: dstPath,
+        content: <Text variant="code-inline-2">{dstPath}</Text>,
+    });
+
+    info.push({
+        name: i18n('transformLambda.label'),
+        copyText: transformLambda,
+        content: (
+            <Text variant="code-2" whiteSpace="nowrap">
+                <pre>{transformLambda}</pre>
+            </Text>
+        ),
+    });
+
     return info;
 }
diff --git a/src/containers/Tenant/Diagnostics/Overview/TransferInfo/i18n/en.json b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/i18n/en.json
index ecd7f0044..bf3ff975c 100644
--- a/src/containers/Tenant/Diagnostics/Overview/TransferInfo/i18n/en.json
+++ b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/i18n/en.json
@@ -3,5 +3,8 @@
   "noData": "No data for entity:",
   "srcConnection.database.label": "Source Database Path",
   "srcConnection.endpoint.label": "Source Cluster Endpoint",
-  "state.label": "State"
+  "state.label": "State",
+  "srcPath.label": "Source Topic",
+  "dstPath.label": "Destination Table",
+  "transformLambda.label": "Transformation Lambda"
 }
diff --git a/src/types/api/schema/replication.ts b/src/types/api/schema/replication.ts
index 95c39a53e..480c796fd 100644
--- a/src/types/api/schema/replication.ts
+++ b/src/types/api/schema/replication.ts
@@ -33,6 +33,16 @@ interface TTargetSpecific {
     Targets: TTarget[];
 }
 
+export interface TTransformTarget {
+    SrcPath?: string;
+    DstPath?: string;
+    TransformLambda?: string;
+}
+
+interface TTransformTargetSpecific {
+    Targets: TTransformTarget[];
+}
+
 /**
  * source: https://github.com/ydb-platform/ydb/blob/main/ydb/core/protos/replication.proto
  */
@@ -40,6 +50,7 @@ export interface TReplicationConfig {
     SrcConnectionParams?: TConnectionParams;
     Everything?: TTargetEverything;
     Specific?: TTargetSpecific;
+    TransferSpecific?: TTransformTargetSpecific;
     InitialSync?: boolean;
 }
 

From dfa6436e7a0bab0604b49d33dbdc017609108900 Mon Sep 17 00:00:00 2001
From: Nikolay Shestakov <tesseract@ydb.tech>
Date: Wed, 5 Mar 2025 16:00:11 +0500
Subject: [PATCH 03/10] fix query templates

---
 .../Tenant/utils/schemaQueryTemplates.ts         | 16 ++++++++--------
 1 file changed, 8 insertions(+), 8 deletions(-)

diff --git a/src/containers/Tenant/utils/schemaQueryTemplates.ts b/src/containers/Tenant/utils/schemaQueryTemplates.ts
index b54796c80..bcbbded19 100644
--- a/src/containers/Tenant/utils/schemaQueryTemplates.ts
+++ b/src/containers/Tenant/utils/schemaQueryTemplates.ts
@@ -82,17 +82,17 @@ export const createTransferTemplate = () => {
     return `-- docs: https://ydb.tech/docs/en/yql/reference/syntax/create-transfer
 CREATE OBJECT secret_name (TYPE SECRET) WITH value="secret_value";
 
-$l = ($x) -> {
+\\$l = (\\$x) -> {
     return [
         <|
-            offset:$x._offset,
-            message:$x._data
+            offset:\\$x._offset,
+            message:\\$x._data
         |>
     ];
 };
 
 CREATE TRANSFER my_transfer
-FROM \${1:<original_topic>} TO \${2:target_table} USING $l
+FROM \${1:<original_topic>} TO \${2:<target_table>} USING \\$l
 WITH (
     CONNECTION_STRING="\${3:grpcs://mydb.ydb.tech:2135/?database=/remote_database}",
     TOKEN_SECRET_NAME = "secret_name"
@@ -285,17 +285,17 @@ export const alterTransferTemplate = (params?: SchemaQueryParams) => {
         : '${1:<my_transfer>}';
     return `-- docs: https://ydb.tech/docs/en/yql/reference/syntax/alter-transfer
 
-$l = ($x) -> {
+\\$l = (\\$x) -> {
     return [
         <|
-            offset:$x._offset,
-            message:$x._data
+            offset:\\$x._offset,
+            message:\\$x._data
         |>
     ];
 };
 
 ALTER TRANSFER ${path} 
-SET USING $l;`;
+SET USING \\$l;`;
 };
 
 export const addTableIndex = (params?: SchemaQueryParams) => {

From fca01358f7e2cfd9b975bdda07ca22df3cecd656 Mon Sep 17 00:00:00 2001
From: Nikolay Shestakov <tesseract@ydb.tech>
Date: Wed, 5 Mar 2025 16:06:03 +0500
Subject: [PATCH 04/10] revert packages-lock

---
 package-lock.json | 17 +++++++++++------
 1 file changed, 11 insertions(+), 6 deletions(-)

diff --git a/package-lock.json b/package-lock.json
index b17d7284b..bde8f1c13 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -1,12 +1,12 @@
 {
   "name": "ydb-embedded-ui",
-  "version": "8.12.0",
+  "version": "8.13.0",
   "lockfileVersion": 3,
   "requires": true,
   "packages": {
     "": {
       "name": "ydb-embedded-ui",
-      "version": "8.12.0",
+      "version": "8.13.0",
       "dependencies": {
         "@bem-react/classname": "^1.6.0",
         "@ebay/nice-modal-react": "^1.2.13",
@@ -997,7 +997,12 @@
       },
       "peerDependencies": {
         "@babel/core": "^7.0.0-0"
-      }
+      }⚠
+Error: Parse error on line 1000:
+...: "^7.0.0-0"      }
+----------------------^
+Expecting '}', ',', got 'EOF'
+
     },
     "node_modules/@babel/plugin-syntax-typescript": {
       "version": "7.25.9",
@@ -29305,9 +29310,9 @@
       }
     },
     "node_modules/ydb-ui-components": {
-      "version": "4.6.0",
-      "resolved": "https://registry.npmjs.org/ydb-ui-components/-/ydb-ui-components-4.6.0.tgz",
-      "integrity": "sha512-9gkD6KBda3Jo/zt8SpTgrmUckXP6gophDiL4E41CKjORCmi5umT5C7D/D98awzaB+ZltVL0L5ySdnZKizfD9Yw==",
+      "version": "4.5.0",
+      "resolved": "https://registry.npmjs.org/ydb-ui-components/-/ydb-ui-components-4.5.0.tgz",
+      "integrity": "sha512-XefBB6dWHXbtyZ9jaxkmCJM5tl19V6VYrJ0nB2WcERCbb8ZG0GYJgywi5BBqQgOL8wH1Z5iut8WX7sQPcj+WNA==",
       "license": "MIT",
       "dependencies": {
         "@bem-react/classname": "^1.6.0",

From f7b4efb1b9bb21d00e9e7d0fa22e4ea0e0776a83 Mon Sep 17 00:00:00 2001
From: Nikolay Shestakov <tesseract@ydb.tech>
Date: Wed, 5 Mar 2025 16:10:01 +0500
Subject: [PATCH 05/10] revert packages-lock

---
 package-lock.json | 9 ++-------
 1 file changed, 2 insertions(+), 7 deletions(-)

diff --git a/package-lock.json b/package-lock.json
index bde8f1c13..bbe276409 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -998,11 +998,6 @@
       "peerDependencies": {
         "@babel/core": "^7.0.0-0"
       }⚠
-Error: Parse error on line 1000:
-...: "^7.0.0-0"      }
-----------------------^
-Expecting '}', ',', got 'EOF'
-
     },
     "node_modules/@babel/plugin-syntax-typescript": {
       "version": "7.25.9",
@@ -29310,8 +29305,8 @@ Expecting '}', ',', got 'EOF'
       }
     },
     "node_modules/ydb-ui-components": {
-      "version": "4.5.0",
-      "resolved": "https://registry.npmjs.org/ydb-ui-components/-/ydb-ui-components-4.5.0.tgz",
+      "version": "4.6.0",
+      "resolved": "https://registry.npmjs.org/ydb-ui-components/-/ydb-ui-components-4.6.0.tgz",
       "integrity": "sha512-XefBB6dWHXbtyZ9jaxkmCJM5tl19V6VYrJ0nB2WcERCbb8ZG0GYJgywi5BBqQgOL8wH1Z5iut8WX7sQPcj+WNA==",
       "license": "MIT",
       "dependencies": {

From 5531c65d631cae9a403dea9b5af3741ef4ad7699 Mon Sep 17 00:00:00 2001
From: Nikolay Shestakov <tesseract@ydb.tech>
Date: Wed, 5 Mar 2025 16:11:12 +0500
Subject: [PATCH 06/10] revert packages-lock

---
 package-lock.json | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/package-lock.json b/package-lock.json
index bbe276409..4399ec6ac 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -997,7 +997,7 @@
       },
       "peerDependencies": {
         "@babel/core": "^7.0.0-0"
-      }⚠
+      }
     },
     "node_modules/@babel/plugin-syntax-typescript": {
       "version": "7.25.9",

From b2f10aeff60f94f3179bf39e02a54b664a75517d Mon Sep 17 00:00:00 2001
From: Nikolay Shestakov <tesseract@ydb.tech>
Date: Wed, 5 Mar 2025 16:26:47 +0500
Subject: [PATCH 07/10] revert packages-lock

---
 package-lock.json | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/package-lock.json b/package-lock.json
index 4399ec6ac..aa5fc93c2 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -29307,7 +29307,7 @@
     "node_modules/ydb-ui-components": {
       "version": "4.6.0",
       "resolved": "https://registry.npmjs.org/ydb-ui-components/-/ydb-ui-components-4.6.0.tgz",
-      "integrity": "sha512-XefBB6dWHXbtyZ9jaxkmCJM5tl19V6VYrJ0nB2WcERCbb8ZG0GYJgywi5BBqQgOL8wH1Z5iut8WX7sQPcj+WNA==",
+      "integrity": "sha512-9gkD6KBda3Jo/zt8SpTgrmUckXP6gophDiL4E41CKjORCmi5umT5C7D/D98awzaB+ZltVL0L5ySdnZKizfD9Yw==",
       "license": "MIT",
       "dependencies": {
         "@bem-react/classname": "^1.6.0",

From fa71cb9df6195ace77a38713d9d4da1e9efaab75 Mon Sep 17 00:00:00 2001
From: Nikolay Shestakov <tesseract@ydb.tech>
Date: Wed, 5 Mar 2025 17:06:48 +0500
Subject: [PATCH 08/10] for rebuild

---
 .../Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx    | 1 +
 1 file changed, 1 insertion(+)

diff --git a/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx
index d7846a5a0..42730d477 100644
--- a/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx
+++ b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx
@@ -88,6 +88,7 @@ function prepareTransferItems(data: TEvDescribeSchemeResult) {
         content: <Text variant="code-inline-2">{dstPath}</Text>,
     });
 
+    // TODO use true pre
     info.push({
         name: i18n('transformLambda.label'),
         copyText: transformLambda,

From d6b914504b80073bc290c5eba6cfe6d6f52347a6 Mon Sep 17 00:00:00 2001
From: Nikolay Shestakov <tesseract@ydb.tech>
Date: Wed, 5 Mar 2025 20:54:01 +0500
Subject: [PATCH 09/10] use YqlHighlighter

---
 .../Diagnostics/Overview/TransferInfo/TransferInfo.tsx    | 8 ++------
 1 file changed, 2 insertions(+), 6 deletions(-)

diff --git a/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx
index 42730d477..dd49ec4d0 100644
--- a/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx
+++ b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx
@@ -3,6 +3,7 @@ import {Flex, Text} from '@gravity-ui/uikit';
 
 import {AsyncReplicationState} from '../../../../../components/AsyncReplicationState';
 import {YDBDefinitionList} from '../../../../../components/YDBDefinitionList/YDBDefinitionList';
+import {YqlHighlighter} from '../../../../../components/YqlHighlighter/YqlHighlighter';
 import type {TEvDescribeSchemeResult} from '../../../../../types/api/schema';
 import {getEntityName} from '../../../utils';
 
@@ -88,15 +89,10 @@ function prepareTransferItems(data: TEvDescribeSchemeResult) {
         content: <Text variant="code-inline-2">{dstPath}</Text>,
     });
 
-    // TODO use true pre
     info.push({
         name: i18n('transformLambda.label'),
         copyText: transformLambda,
-        content: (
-            <Text variant="code-2" whiteSpace="nowrap">
-                <pre>{transformLambda}</pre>
-            </Text>
-        ),
+        content: <YqlHighlighter>{transformLambda}</YqlHighlighter>,
     });
 
     return info;

From a067c7b20db5886a921b96422ad617463512a63a Mon Sep 17 00:00:00 2001
From: Nikolay Shestakov <tesseract@ydb.tech>
Date: Wed, 5 Mar 2025 21:30:44 +0500
Subject: [PATCH 10/10] use YqlHighlighter 2

---
 .../Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx   | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx
index dd49ec4d0..6981941f1 100644
--- a/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx
+++ b/src/containers/Tenant/Diagnostics/Overview/TransferInfo/TransferInfo.tsx
@@ -92,7 +92,7 @@ function prepareTransferItems(data: TEvDescribeSchemeResult) {
     info.push({
         name: i18n('transformLambda.label'),
         copyText: transformLambda,
-        content: <YqlHighlighter>{transformLambda}</YqlHighlighter>,
+        content: transformLambda ? <YqlHighlighter>{transformLambda}</YqlHighlighter> : null,
     });
 
     return info;