@ibdop/platform-kit 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (51) hide show
  1. package/README.md +63 -0
  2. package/dist/components/ErrorBoundary.d.ts +68 -0
  3. package/dist/components/ErrorBoundary.d.ts.map +1 -0
  4. package/dist/components/Notification.d.ts +46 -0
  5. package/dist/components/Notification.d.ts.map +1 -0
  6. package/dist/components/VersionInfo.d.ts +18 -0
  7. package/dist/components/VersionInfo.d.ts.map +1 -0
  8. package/dist/components/index.d.ts +8 -0
  9. package/dist/components/index.d.ts.map +1 -0
  10. package/dist/hooks/index.d.ts +9 -0
  11. package/dist/hooks/index.d.ts.map +1 -0
  12. package/dist/hooks/useApi.d.ts +65 -0
  13. package/dist/hooks/useApi.d.ts.map +1 -0
  14. package/dist/hooks/useInfoData.d.ts +35 -0
  15. package/dist/hooks/useInfoData.d.ts.map +1 -0
  16. package/dist/hooks/usePermissions.d.ts +23 -0
  17. package/dist/hooks/usePermissions.d.ts.map +1 -0
  18. package/dist/hooks/useShellAuth.d.ts +23 -0
  19. package/dist/hooks/useShellAuth.d.ts.map +1 -0
  20. package/dist/hooks/useV1Config.d.ts +27 -0
  21. package/dist/hooks/useV1Config.d.ts.map +1 -0
  22. package/dist/index.cjs.js +18 -0
  23. package/dist/index.d.ts +10 -0
  24. package/dist/index.d.ts.map +1 -0
  25. package/dist/index.es.js +1500 -0
  26. package/dist/index.umd.js +18 -0
  27. package/dist/services/api.d.ts +59 -0
  28. package/dist/services/api.d.ts.map +1 -0
  29. package/dist/services/index.d.ts +8 -0
  30. package/dist/services/index.d.ts.map +1 -0
  31. package/dist/services/logger.d.ts +41 -0
  32. package/dist/services/logger.d.ts.map +1 -0
  33. package/dist/types/index.d.ts +178 -0
  34. package/dist/types/index.d.ts.map +1 -0
  35. package/package.json +90 -0
  36. package/src/components/ErrorBoundary.tsx +292 -0
  37. package/src/components/Notification.tsx +297 -0
  38. package/src/components/VersionInfo.tsx +271 -0
  39. package/src/components/index.ts +14 -0
  40. package/src/global.d.ts +40 -0
  41. package/src/hooks/index.ts +13 -0
  42. package/src/hooks/useApi.ts +314 -0
  43. package/src/hooks/useInfoData.ts +124 -0
  44. package/src/hooks/usePermissions.ts +88 -0
  45. package/src/hooks/useShellAuth.ts +145 -0
  46. package/src/hooks/useV1Config.ts +112 -0
  47. package/src/index.ts +17 -0
  48. package/src/services/api.ts +290 -0
  49. package/src/services/index.ts +9 -0
  50. package/src/services/logger.ts +71 -0
  51. package/src/types/index.ts +215 -0
package/README.md ADDED
@@ -0,0 +1,63 @@
1
+ # Platform Kit
2
+
3
+ > Единый пакет для микрофронтендов IngoBank DevOps Platform
4
+
5
+ ## Установка
6
+
7
+ ```bash
8
+ npm install @ib-dop/platform-kit
9
+ ```
10
+
11
+ ### Peer Dependencies
12
+
13
+ ```bash
14
+ npm install react "^18.0.0"
15
+ npm install react-dom "^18.0.0"
16
+ npm install axios "^1.0.0"
17
+ ```
18
+
19
+ ## Быстрый старт
20
+
21
+ ```typescript
22
+ import { ErrorBoundary, useShellAuth, createApiClient } from '@ib-dop/platform-kit'
23
+
24
+ export function App() {
25
+ const auth = useShellAuth()
26
+ const api = createApiClient()
27
+
28
+ return (
29
+ <ErrorBoundary mfeName="@ib-dop/mf-example">
30
+ <Dashboard auth={auth} />
31
+ </ErrorBoundary>
32
+ )
33
+ }
34
+ ```
35
+
36
+ ## Что внутри
37
+
38
+ | Модуль | Содержимое |
39
+ |--------|------------|
40
+ | **Hooks** | `useShellAuth`, `useInfoData`, `useV1Config`, `useApi`, `usePermissions` |
41
+ | **Components** | `ErrorBoundary`, `VersionInfo` |
42
+ | **Services** | API клиент, Logger |
43
+
44
+ ## Документация
45
+
46
+ - **[`docs/USAGE.md`](docs/USAGE.md)** - Подключение и использование
47
+ - **[`docs/BUILD_AND_PUBLISH.md`](docs/BUILD_AND_PUBLISH.md)** - Сборка и публикация
48
+ - **[`PLAN.md`](PLAN.md)** - План разработки
49
+
50
+ ## Миграция
51
+
52
+ ### Было
53
+
54
+ ```typescript
55
+ import { useShellAuth } from './hooks/useShellAuth'
56
+ import ErrorBoundary from './components/ErrorBoundary'
57
+ ```
58
+
59
+ ### Стало
60
+
61
+ ```typescript
62
+ import { useShellAuth, ErrorBoundary } from '@ib-dop/platform-kit'
63
+ ```
@@ -0,0 +1,68 @@
1
+ import { Component, ReactNode, ErrorInfo } from 'react';
2
+ /**
3
+ * Props для ErrorBoundary
4
+ */
5
+ interface ErrorBoundaryProps {
6
+ children: ReactNode;
7
+ mfeName?: string;
8
+ showDetails?: boolean;
9
+ }
10
+ /**
11
+ * State для ErrorBoundary
12
+ */
13
+ interface ErrorBoundaryState {
14
+ hasError: boolean;
15
+ error?: Error;
16
+ }
17
+ /**
18
+ * ErrorBoundary - компонент для перехвата и обработки ошибок
19
+ *
20
+ * @example
21
+ * ```tsx
22
+ * <ErrorBoundary mfeName="@ib-dop/mf-example">
23
+ * <MyComponent />
24
+ * </ErrorBoundary>
25
+ * ```
26
+ */
27
+ export default class ErrorBoundary extends Component<ErrorBoundaryProps, ErrorBoundaryState> {
28
+ private hasDispatched;
29
+ constructor(props: ErrorBoundaryProps);
30
+ /**
31
+ * Получить имя MF из props или window
32
+ */
33
+ private getMfeName;
34
+ /**
35
+ * Определить нужно ли показывать детали ошибки
36
+ */
37
+ private shouldShowDetails;
38
+ /**
39
+ * Dispatch ошибки в shell
40
+ */
41
+ private dispatchError;
42
+ /**
43
+ * Получить derived state из ошибки
44
+ */
45
+ static getDerivedStateFromError(error: Error): ErrorBoundaryState;
46
+ /**
47
+ * Обработать ошибку
48
+ */
49
+ componentDidCatch(error: Error, errorInfo: ErrorInfo): void;
50
+ /**
51
+ * Копировать ошибку в буфер обмена
52
+ */
53
+ private handleCopy;
54
+ /**
55
+ * Повторить рендер
56
+ */
57
+ private handleRetry;
58
+ /**
59
+ * Перейти на главную
60
+ */
61
+ private handleGoHome;
62
+ /**
63
+ * Рендер
64
+ */
65
+ render(): ReactNode;
66
+ }
67
+ export {};
68
+ //# sourceMappingURL=ErrorBoundary.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ErrorBoundary.d.ts","sourceRoot":"","sources":["../../src/components/ErrorBoundary.tsx"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAc,EAAE,SAAS,EAAE,SAAS,EAAE,SAAS,EAAE,MAAM,OAAO,CAAA;AAuB9D;;GAEG;AACH,UAAU,kBAAkB;IAC1B,QAAQ,EAAE,SAAS,CAAA;IACnB,OAAO,CAAC,EAAE,MAAM,CAAA;IAChB,WAAW,CAAC,EAAE,OAAO,CAAA;CACtB;AAED;;GAEG;AACH,UAAU,kBAAkB;IAC1B,QAAQ,EAAE,OAAO,CAAA;IACjB,KAAK,CAAC,EAAE,KAAK,CAAA;CACd;AAED;;;;;;;;;GASG;AACH,MAAM,CAAC,OAAO,OAAO,aAAc,SAAQ,SAAS,CAAC,kBAAkB,EAAE,kBAAkB,CAAC;IAC1F,OAAO,CAAC,aAAa,CAAQ;gBAEjB,KAAK,EAAE,kBAAkB;IAKrC;;OAEG;IACH,OAAO,CAAC,UAAU;IAWlB;;OAEG;IACH,OAAO,CAAC,iBAAiB;IAyBzB;;OAEG;IACH,OAAO,CAAC,aAAa;IAwBrB;;OAEG;IACH,MAAM,CAAC,wBAAwB,CAAC,KAAK,EAAE,KAAK,GAAG,kBAAkB;IAIjE;;OAEG;IACH,iBAAiB,CAAC,KAAK,EAAE,KAAK,EAAE,SAAS,EAAE,SAAS,GAAG,IAAI;IAK3D;;OAEG;IACH,OAAO,CAAC,UAAU,CAUjB;IAED;;OAEG;IACH,OAAO,CAAC,WAAW,CAOlB;IAED;;OAEG;IACH,OAAO,CAAC,YAAY,CAInB;IAED;;OAEG;IACH,MAAM,IAAI,SAAS;CAwGpB"}
@@ -0,0 +1,46 @@
1
+ import { default as React, ReactNode } from 'react';
2
+ import { NotificationPayload } from '../types';
3
+ /**
4
+ * Context for notifications
5
+ */
6
+ interface NotificationContextValue {
7
+ notify: (payload: Omit<NotificationPayload, 'timestamp' | 'mfeName'>) => void;
8
+ notifySuccess: (message: string, title?: string) => void;
9
+ notifyError: (message: string, title?: string) => void;
10
+ notifyWarning: (message: string, title?: string) => void;
11
+ notifyInfo: (message: string, title?: string) => void;
12
+ removeNotification: (id: string) => void;
13
+ }
14
+ /**
15
+ * Provider for notifications
16
+ */
17
+ interface NotificationProviderProps {
18
+ children: ReactNode;
19
+ }
20
+ export declare function NotificationProvider({ children }: NotificationProviderProps): React.ReactElement;
21
+ /**
22
+ * Хук для использования уведомлений
23
+ *
24
+ * @returns NotificationContextValue
25
+ *
26
+ * @example
27
+ * ```tsx
28
+ * const { notifySuccess, notifyError } = useNotification()
29
+ *
30
+ * const handleSave = async () => {
31
+ * try {
32
+ * await saveData()
33
+ * notifySuccess('Данные сохранены')
34
+ * } catch (error) {
35
+ * notifyError('Ошибка сохранения')
36
+ * }
37
+ * }
38
+ * ```
39
+ */
40
+ export declare function useNotification(): NotificationContextValue;
41
+ /**
42
+ * Dispatch notification to shell (standalone function)
43
+ */
44
+ export declare function dispatchNotification(payload: Omit<NotificationPayload, 'timestamp' | 'mfeName'>): void;
45
+ export {};
46
+ //# sourceMappingURL=Notification.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Notification.d.ts","sourceRoot":"","sources":["../../src/components/Notification.tsx"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,KAAK,EAAE,EAA+D,SAAS,EAAE,MAAM,OAAO,CAAA;AACrG,OAAO,KAAK,EAAE,mBAAmB,EAAoB,MAAM,UAAU,CAAA;AA8BrE;;GAEG;AACH,UAAU,wBAAwB;IAChC,MAAM,EAAE,CAAC,OAAO,EAAE,IAAI,CAAC,mBAAmB,EAAE,WAAW,GAAG,SAAS,CAAC,KAAK,IAAI,CAAA;IAC7E,aAAa,EAAE,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,MAAM,KAAK,IAAI,CAAA;IACxD,WAAW,EAAE,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,MAAM,KAAK,IAAI,CAAA;IACtD,aAAa,EAAE,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,MAAM,KAAK,IAAI,CAAA;IACxD,UAAU,EAAE,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,MAAM,KAAK,IAAI,CAAA;IACrD,kBAAkB,EAAE,CAAC,EAAE,EAAE,MAAM,KAAK,IAAI,CAAA;CACzC;AAOD;;GAEG;AACH,UAAU,yBAAyB;IACjC,QAAQ,EAAE,SAAS,CAAA;CACpB;AAED,wBAAgB,oBAAoB,CAAC,EAAE,QAAQ,EAAE,EAAE,yBAAyB,GAAG,KAAK,CAAC,YAAY,CA8KhG;AAED;;;;;;;;;;;;;;;;;;GAkBG;AACH,wBAAgB,eAAe,IAAI,wBAAwB,CAkB1D;AAED;;GAEG;AACH,wBAAgB,oBAAoB,CAAC,OAAO,EAAE,IAAI,CAAC,mBAAmB,EAAE,WAAW,GAAG,SAAS,CAAC,GAAG,IAAI,CAiBtG"}
@@ -0,0 +1,18 @@
1
+ import { default as React } from 'react';
2
+ /**
3
+ * Props для VersionInfo
4
+ */
5
+ interface VersionInfoProps {
6
+ mfeName: string;
7
+ }
8
+ /**
9
+ * VersionInfo - компонент для отображения информации о версии
10
+ *
11
+ * @example
12
+ * ```tsx
13
+ * <VersionInfo mfeName="@ib-dop/mf-home" />
14
+ * ```
15
+ */
16
+ export default function VersionInfo({ mfeName }: VersionInfoProps): React.ReactElement;
17
+ export {};
18
+ //# sourceMappingURL=VersionInfo.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"VersionInfo.d.ts","sourceRoot":"","sources":["../../src/components/VersionInfo.tsx"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,KAAmB,MAAM,OAAO,CAAA;AAsCvC;;GAEG;AACH,UAAU,gBAAgB;IACxB,OAAO,EAAE,MAAM,CAAA;CAChB;AAED;;;;;;;GAOG;AACH,MAAM,CAAC,OAAO,UAAU,WAAW,CAAC,EAAE,OAAO,EAAE,EAAE,gBAAgB,GAAG,KAAK,CAAC,YAAY,CAmNrF"}
@@ -0,0 +1,8 @@
1
+ /**
2
+ * Components barrel export
3
+ */
4
+ export { default as ErrorBoundary } from './ErrorBoundary';
5
+ export { default as VersionInfo } from './VersionInfo';
6
+ export { NotificationProvider, useNotification, dispatchNotification } from './Notification';
7
+ export type { NotificationPayload, NotificationType, } from '../types';
8
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/components/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,OAAO,IAAI,aAAa,EAAE,MAAM,iBAAiB,CAAA;AAE1D,OAAO,EAAE,OAAO,IAAI,WAAW,EAAE,MAAM,eAAe,CAAA;AAEtD,OAAO,EAAE,oBAAoB,EAAE,eAAe,EAAE,oBAAoB,EAAE,MAAM,gBAAgB,CAAA;AAE5F,YAAY,EACV,mBAAmB,EACnB,gBAAgB,GACjB,MAAM,UAAU,CAAA"}
@@ -0,0 +1,9 @@
1
+ /**
2
+ * Hooks barrel export
3
+ */
4
+ export { useShellAuth, getAuth } from './useShellAuth';
5
+ export { useInfoData } from './useInfoData';
6
+ export { useV1Config } from './useV1Config';
7
+ export { useApi } from './useApi';
8
+ export { usePermissions } from './usePermissions';
9
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/hooks/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,YAAY,EAAE,OAAO,EAAE,MAAM,gBAAgB,CAAA;AAEtD,OAAO,EAAE,WAAW,EAAE,MAAM,eAAe,CAAA;AAE3C,OAAO,EAAE,WAAW,EAAE,MAAM,eAAe,CAAA;AAE3C,OAAO,EAAE,MAAM,EAAE,MAAM,UAAU,CAAA;AAEjC,OAAO,EAAE,cAAc,EAAE,MAAM,kBAAkB,CAAA"}
@@ -0,0 +1,65 @@
1
+ import { ApiError, ApiResponse, NotificationPayload } from '../types';
2
+ /**
3
+ * Dispatch notification to shell
4
+ */
5
+ export declare function dispatchNotification(notification: Omit<NotificationPayload, 'timestamp' | 'mfeName'>): void;
6
+ /**
7
+ * Dispatch error notification to shell
8
+ */
9
+ export declare function dispatchErrorNotification(error: ApiError, context?: string): void;
10
+ /**
11
+ * Interface для конфигурации useApi
12
+ */
13
+ export interface UseApiConfig<T = never> {
14
+ notifyOnError?: boolean;
15
+ notifyOnSuccess?: boolean;
16
+ successMessage?: string;
17
+ errorContext?: string;
18
+ onSuccess?: (data: T) => void;
19
+ onError?: (error: ApiError) => void;
20
+ }
21
+ /**
22
+ * Interface для результата useApi
23
+ */
24
+ export interface UseApiResult<T> {
25
+ data: T | null;
26
+ error: ApiError | null;
27
+ isLoading: boolean;
28
+ isError: boolean;
29
+ isSuccess: boolean;
30
+ execute: () => Promise<T | null>;
31
+ reset: () => void;
32
+ }
33
+ /**
34
+ * Централизованный хук для API запросов
35
+ *
36
+ * @param request - функция, возвращающая Promise<ApiResponse<T>>
37
+ * @param config - конфигурация
38
+ * @returns UseApiResult<T>
39
+ *
40
+ * @example
41
+ * ```tsx
42
+ * const { data, isLoading, isError, execute } = useApi<User[]>(
43
+ * () => fetch('/api/users').then(r => r.json()),
44
+ * { notifyOnError: true, errorContext: 'загрузка пользователей' }
45
+ * )
46
+ * ```
47
+ */
48
+ export declare function useApi<T>(request: () => Promise<ApiResponse<T>>, config?: UseApiConfig<T>): UseApiResult<T>;
49
+ /**
50
+ * GET запрос
51
+ */
52
+ export declare function useGet<T>(url: string, params?: Record<string, unknown>, config?: UseApiConfig<T>): UseApiResult<T>;
53
+ /**
54
+ * POST запрос
55
+ */
56
+ export declare function usePost<T>(url: string, data?: unknown, config?: UseApiConfig<T>): UseApiResult<T>;
57
+ /**
58
+ * PUT запрос
59
+ */
60
+ export declare function usePut<T>(url: string, data?: unknown, config?: UseApiConfig<T>): UseApiResult<T>;
61
+ /**
62
+ * DELETE запрос
63
+ */
64
+ export declare function useDel<T>(url: string, config?: UseApiConfig<T>): UseApiResult<T>;
65
+ //# sourceMappingURL=useApi.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useApi.d.ts","sourceRoot":"","sources":["../../src/hooks/useApi.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAGH,OAAO,KAAK,EAAE,QAAQ,EAAE,WAAW,EAAE,mBAAmB,EAAE,MAAM,UAAU,CAAA;AAsB1E;;GAEG;AACH,wBAAgB,oBAAoB,CAAC,YAAY,EAAE,IAAI,CAAC,mBAAmB,EAAE,WAAW,GAAG,SAAS,CAAC,GAAG,IAAI,CAiB3G;AAED;;GAEG;AACH,wBAAgB,yBAAyB,CAAC,KAAK,EAAE,QAAQ,EAAE,OAAO,CAAC,EAAE,MAAM,GAAG,IAAI,CAuCjF;AAED;;GAEG;AACH,MAAM,WAAW,YAAY,CAAC,CAAC,GAAG,KAAK;IACrC,aAAa,CAAC,EAAE,OAAO,CAAA;IACvB,eAAe,CAAC,EAAE,OAAO,CAAA;IACzB,cAAc,CAAC,EAAE,MAAM,CAAA;IACvB,YAAY,CAAC,EAAE,MAAM,CAAA;IACrB,SAAS,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC,KAAK,IAAI,CAAA;IAC7B,OAAO,CAAC,EAAE,CAAC,KAAK,EAAE,QAAQ,KAAK,IAAI,CAAA;CACpC;AAED;;GAEG;AACH,MAAM,WAAW,YAAY,CAAC,CAAC;IAC7B,IAAI,EAAE,CAAC,GAAG,IAAI,CAAA;IACd,KAAK,EAAE,QAAQ,GAAG,IAAI,CAAA;IACtB,SAAS,EAAE,OAAO,CAAA;IAClB,OAAO,EAAE,OAAO,CAAA;IAChB,SAAS,EAAE,OAAO,CAAA;IAClB,OAAO,EAAE,MAAM,OAAO,CAAC,CAAC,GAAG,IAAI,CAAC,CAAA;IAChC,KAAK,EAAE,MAAM,IAAI,CAAA;CAClB;AAED;;;;;;;;;;;;;;GAcG;AACH,wBAAgB,MAAM,CAAC,CAAC,EACtB,OAAO,EAAE,MAAM,OAAO,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,EACtC,MAAM,GAAE,YAAY,CAAC,CAAC,CAAM,GAC3B,YAAY,CAAC,CAAC,CAAC,CAsFjB;AAID;;GAEG;AACH,wBAAgB,MAAM,CAAC,CAAC,EACtB,GAAG,EAAE,MAAM,EACX,MAAM,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAChC,MAAM,GAAE,YAAY,CAAC,CAAC,CAAM,GAC3B,YAAY,CAAC,CAAC,CAAC,CAYjB;AAED;;GAEG;AACH,wBAAgB,OAAO,CAAC,CAAC,EACvB,GAAG,EAAE,MAAM,EACX,IAAI,CAAC,EAAE,OAAO,EACd,MAAM,GAAE,YAAY,CAAC,CAAC,CAAM,GAC3B,YAAY,CAAC,CAAC,CAAC,CAajB;AAED;;GAEG;AACH,wBAAgB,MAAM,CAAC,CAAC,EACtB,GAAG,EAAE,MAAM,EACX,IAAI,CAAC,EAAE,OAAO,EACd,MAAM,GAAE,YAAY,CAAC,CAAC,CAAM,GAC3B,YAAY,CAAC,CAAC,CAAC,CAajB;AAED;;GAEG;AACH,wBAAgB,MAAM,CAAC,CAAC,EACtB,GAAG,EAAE,MAAM,EACX,MAAM,GAAE,YAAY,CAAC,CAAC,CAAM,GAC3B,YAAY,CAAC,CAAC,CAAC,CAYjB"}
@@ -0,0 +1,35 @@
1
+ import { InfoData } from '../types';
2
+ /**
3
+ * Interface для props useInfoData
4
+ */
5
+ interface UseInfoDataProps {
6
+ mfeName?: string;
7
+ }
8
+ /**
9
+ * Interface для результата useInfoData
10
+ */
11
+ interface UseInfoDataResult {
12
+ data: InfoData | null;
13
+ isLoading: boolean;
14
+ error: string | null;
15
+ refetch: () => void;
16
+ }
17
+ /**
18
+ * Централизованный хук для загрузки информации о версии MF
19
+ *
20
+ * @param props - { mfeName?: string } - опционально имя MF
21
+ * @returns { data, isLoading, error, refetch }
22
+ *
23
+ * @example
24
+ * ```tsx
25
+ * const { data, isLoading, error } = useInfoData({ mfeName: '@ib-dop/mf-home' })
26
+ *
27
+ * if (isLoading) return <Loading />
28
+ * if (error) return <Error>{error}</Error>
29
+ *
30
+ * return <Badge>{data?.BUILD_VERSION}</Badge>
31
+ * ```
32
+ */
33
+ export declare function useInfoData(props?: UseInfoDataProps): UseInfoDataResult;
34
+ export type { UseInfoDataProps, UseInfoDataResult };
35
+ //# sourceMappingURL=useInfoData.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useInfoData.d.ts","sourceRoot":"","sources":["../../src/hooks/useInfoData.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAGH,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,UAAU,CAAA;AAoBxC;;GAEG;AACH,UAAU,gBAAgB;IACxB,OAAO,CAAC,EAAE,MAAM,CAAA;CACjB;AAED;;GAEG;AACH,UAAU,iBAAiB;IACzB,IAAI,EAAE,QAAQ,GAAG,IAAI,CAAA;IACrB,SAAS,EAAE,OAAO,CAAA;IAClB,KAAK,EAAE,MAAM,GAAG,IAAI,CAAA;IACpB,OAAO,EAAE,MAAM,IAAI,CAAA;CACpB;AAiBD;;;;;;;;;;;;;;;GAeG;AACH,wBAAgB,WAAW,CAAC,KAAK,CAAC,EAAE,gBAAgB,GAAG,iBAAiB,CA6CvE;AAGD,YAAY,EAAE,gBAAgB,EAAE,iBAAiB,EAAE,CAAA"}
@@ -0,0 +1,23 @@
1
+ import { MfPermissionsConfig, MfPermissions } from '../types';
2
+ /**
3
+ * Default permissions configuration
4
+ */
5
+ export declare const DEFAULT_PERMISSIONS_CONFIG: MfPermissionsConfig;
6
+ /**
7
+ * Централизованный хук для проверки прав доступа
8
+ *
9
+ * @param config - конфигурация прав доступа (опционально)
10
+ * @returns MfPermissions - объект с правами доступа
11
+ *
12
+ * @example
13
+ * ```tsx
14
+ * const permissions = usePermissions()
15
+ *
16
+ * if (permissions.canAdmin) {
17
+ * return <AdminPanel />
18
+ * }
19
+ * ```
20
+ */
21
+ export declare function usePermissions(config?: Partial<MfPermissionsConfig>): MfPermissions;
22
+ export type { MfPermissionsConfig };
23
+ //# sourceMappingURL=usePermissions.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"usePermissions.d.ts","sourceRoot":"","sources":["../../src/hooks/usePermissions.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAIH,OAAO,KAAK,EAAE,mBAAmB,EAAE,aAAa,EAAE,MAAM,UAAU,CAAA;AAiBlE;;GAEG;AACH,eAAO,MAAM,0BAA0B,EAAE,mBAQxC,CAAA;AAED;;;;;;;;;;;;;;GAcG;AACH,wBAAgB,cAAc,CAAC,MAAM,GAAE,OAAO,CAAC,mBAAmB,CAAM,GAAG,aAAa,CA+BvF;AAGD,YAAY,EAAE,mBAAmB,EAAE,CAAA"}
@@ -0,0 +1,23 @@
1
+ import { ShellAuth } from '../types';
2
+ /**
3
+ * Получить auth из window globals
4
+ */
5
+ declare function getAuth(): ShellAuth | null;
6
+ /**
7
+ * Централизованный хук для получения состояния авторизации из shell
8
+ *
9
+ * @returns ShellAuth - объект с состоянием авторизации
10
+ *
11
+ * @example
12
+ * ```tsx
13
+ * const auth = useShellAuth()
14
+ *
15
+ * if (auth.isLoading) return <Loading />
16
+ * if (!auth.isAuthenticated) return <Login />
17
+ *
18
+ * return <div>{auth.user?.profile?.preferred_username}</div>
19
+ * ```
20
+ */
21
+ export declare function useShellAuth(): ShellAuth;
22
+ export { getAuth };
23
+ //# sourceMappingURL=useShellAuth.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useShellAuth.d.ts","sourceRoot":"","sources":["../../src/hooks/useShellAuth.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAGH,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,UAAU,CAAA;AAiBzC;;GAEG;AACH,iBAAS,OAAO,IAAI,SAAS,GAAG,IAAI,CAgBnC;AAED;;;;;;;;;;;;;;GAcG;AACH,wBAAgB,YAAY,IAAI,SAAS,CAgFxC;AAGD,OAAO,EAAE,OAAO,EAAE,CAAA"}
@@ -0,0 +1,27 @@
1
+ import { V1ConfigData } from '../types';
2
+ /**
3
+ * Interface для результата useV1Config
4
+ */
5
+ interface UseV1ConfigResult {
6
+ data: V1ConfigData | null;
7
+ isLoading: boolean;
8
+ error: string | null;
9
+ }
10
+ /**
11
+ * Централизованный хук для загрузки конфигурации из sessionStorage
12
+ *
13
+ * @returns { data, isLoading, error }
14
+ *
15
+ * @example
16
+ * ```tsx
17
+ * const { data, isLoading, error } = useV1Config()
18
+ *
19
+ * if (isLoading) return null
20
+ * if (error || !data) return <Error>{error}</Error>
21
+ *
22
+ * return <ConfigDisplay config={data} />
23
+ * ```
24
+ */
25
+ export declare function useV1Config(): UseV1ConfigResult;
26
+ export type { UseV1ConfigResult };
27
+ //# sourceMappingURL=useV1Config.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useV1Config.d.ts","sourceRoot":"","sources":["../../src/hooks/useV1Config.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAGH,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,UAAU,CAAA;AA6B5C;;GAEG;AACH,UAAU,iBAAiB;IACzB,IAAI,EAAE,YAAY,GAAG,IAAI,CAAA;IACzB,SAAS,EAAE,OAAO,CAAA;IAClB,KAAK,EAAE,MAAM,GAAG,IAAI,CAAA;CACrB;AAED;;;;;;;;;;;;;;GAcG;AACH,wBAAgB,WAAW,IAAI,iBAAiB,CAgD/C;AAGD,YAAY,EAAE,iBAAiB,EAAE,CAAA"}
@@ -0,0 +1,18 @@
1
+ "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const h=require("react"),kt=require("axios"),Le={log:(...n)=>{},warn:(...n)=>{}};function Q(){if(typeof window>"u")return null;const n=window;if(n.__SHELL_AUTH_INSTANCE__)return n.__SHELL_AUTH_INSTANCE__;const r=window;return r.__SHELL_AUTH__?.authInstance?r.__SHELL_AUTH__.authInstance:null}function Be(){const[n,r]=h.useState(null),[a,l]=h.useState(!0),s=h.useRef(0),d=20,i=h.useCallback(()=>Q(),[]);h.useEffect(()=>{const E=i();if(E){r(E),l(!1);return}const g=u=>{r(u.detail),l(!1)},S=setInterval(()=>{s.current++;const u=i();u?(Le.log("Auth found via polling, attempts:",s.current),r(u),l(!1),clearInterval(S)):s.current>=d&&(l(!1),clearInterval(S))},500);return window.addEventListener("shell-auth-ready",g),()=>{clearInterval(S),window.removeEventListener("shell-auth-ready",g)}},[i]);const p=n||{user:null,isAuthenticated:!1,isLoading:a,signinRedirect:async()=>{const E=Q();E?.signinRedirect?await E.signinRedirect():typeof window<"u"&&(window.location.href="/")},removeUser:async()=>{const E=Q();E?.removeUser&&await E.removeUser()}};return Le.log("Auth result:",{isAuthenticated:p.isAuthenticated,isLoading:p.isLoading}),p}const Tt={log:(...n)=>{},warn:(...n)=>{},error:(...n)=>{console.error("[useInfoData]",...n)}};function Ot(n){if(n)return n;if(typeof window<"u"){const r=window;if(r.__MF_NAME__)return r.__MF_NAME__}return"unknown-mfe"}function We(n){const[r,a]=h.useState(null),[l,s]=h.useState(!0),[d,i]=h.useState(null),x=h.useCallback(()=>{const E=Ot(n?.mfeName).replace("@ib-dop/","");s(!0),i(null),fetch(`/svc/${E}/info.json`).then(g=>{if(!g.ok)throw new Error(`HTTP ${g.status}: ${g.statusText}`);return g.json()}).then(g=>{a(g)}).catch(g=>{Tt.error("Failed to load info:",g),i(g instanceof Error?g.message:String(g))}).finally(()=>{s(!1)})},[n?.mfeName]);return h.useEffect(()=>{x()},[x]),{data:r,isLoading:l,error:d,refetch:x}}const ue={log:(...n)=>{},warn:(...n)=>{},error:(...n)=>{console.error("[useV1Config]",...n)}},X={authority:"",client_id:"",environment:"development"};function Mt(){const[n,r]=h.useState(null),[a,l]=h.useState(!0),[s,d]=h.useState(null);return h.useEffect(()=>{(()=>{if(typeof sessionStorage>"u"){d("sessionStorage not available"),r(X),l(!1);return}try{const x=sessionStorage.getItem("config");if(x){const p=JSON.parse(x);r({...X,...p}),d(null),ue.log("Config loaded successfully")}else r(X),d("Config not found in sessionStorage"),ue.warn("Config not found in sessionStorage")}catch(x){ue.error("Error parsing config:",x),r(X),d("Error parsing config")}finally{l(!1)}})()},[]),{data:n,isLoading:a,error:s}}function He(n){if(typeof window>"u")return;const r=window.__MF_NAME__||"unknown",a={...n,mfeName:r,timestamp:Date.now()};window.dispatchEvent(new CustomEvent("mfe-notification",{detail:a,bubbles:!0}))}function $e(n,r){const a={network:{title:"Нет подключения",message:"Сервер недоступен. Проверьте подключение к интернету."},unauthorized:{title:"Требуется вход",message:"Ваша сессия истекла. Войдите в систему снова."},forbidden:{title:"Доступ запрещён",message:"У вас нет прав для выполнения этого действия."},not_found:{title:"Не найдено",message:"Запрошенный ресурс не найден."},server:{title:"Ошибка сервера",message:"Произошла ошибка на сервере. Попробуйте позже."},client:{title:"Ошибка",message:n.message||"Произошла ошибка при выполнении запроса."},unknown:{title:"Неизвестная ошибка",message:n.message||"Произошла неизвестная ошибка."}},l=a[n.type]||a.unknown;He({type:n.type==="network"?"warning":"error",title:l.title,message:r?`${l.message} (${r})`:l.message})}function Nt(n,r={}){const{notifyOnError:a=!0,notifyOnSuccess:l=!1,successMessage:s,errorContext:d,onSuccess:i,onError:x}=r,[p,E]=h.useState(null),[g,S]=h.useState(null),[u,_]=h.useState(!1),A=g!==null,M=p!==null&&!u&&!A,ee=h.useCallback(async()=>{_(!0),S(null);try{const w=await n();if(w.ok)return E(w.data),l&&s&&He({type:"success",title:"Успешно",message:s}),i?.(w.data),w.data;{const N={message:w.data||"Request failed",status:w.status,type:"client",timestamp:Date.now()};return S(N),a&&$e(N,d),x?.(N),null}}catch(w){const N=w;return S(N),a&&$e(N,d),x?.(N),null}finally{_(!1)}},[n,a,l,s,d,i,x]),j=h.useCallback(()=>{E(null),S(null),_(!1)},[]);return{data:p,error:g,isLoading:u,isError:A,isSuccess:M,execute:ee,reset:j}}const jt={canView:["all"],canEdit:["ibdop-user","ibdop-admin","ibdop-devops"],canDelete:["ibdop-admin","ibdop-devops"],canAdmin:["ibdop-admin"],canViewSensitiveData:["ibdop-admin","ibdop-devops"],canExportData:["ibdop-user"],canManageUsers:["ibdop-admin"]};function Pt(n={}){const r=Be(),a=h.useMemo(()=>({...jt,...n}),[n]),l=h.useMemo(()=>{const d=r.user?.profile?.realm_roles||r.user?.profile?.roles||[];return Array.isArray(d)?d:[d]},[r.user]),s=d=>d.includes("all")?!0:d.some(i=>l.includes(i));return{canView:s(a.canView),canEdit:s(a.canEdit),canDelete:s(a.canDelete),canAdmin:s(a.canAdmin),canViewSensitiveData:s(a.canViewSensitiveData),canExportData:s(a.canExportData),canManageUsers:s(a.canManageUsers)}}var Z={exports:{}},B={};var Fe;function Dt(){if(Fe)return B;Fe=1;var n=h,r=Symbol.for("react.element"),a=Symbol.for("react.fragment"),l=Object.prototype.hasOwnProperty,s=n.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED.ReactCurrentOwner,d={key:!0,ref:!0,__self:!0,__source:!0};function i(x,p,E){var g,S={},u=null,_=null;E!==void 0&&(u=""+E),p.key!==void 0&&(u=""+p.key),p.ref!==void 0&&(_=p.ref);for(g in p)l.call(p,g)&&!d.hasOwnProperty(g)&&(S[g]=p[g]);if(x&&x.defaultProps)for(g in p=x.defaultProps,p)S[g]===void 0&&(S[g]=p[g]);return{$$typeof:r,type:x,key:u,ref:_,props:S,_owner:s.current}}return B.Fragment=a,B.jsx=i,B.jsxs=i,B}var W={};var Ue;function Lt(){return Ue||(Ue=1,process.env.NODE_ENV!=="production"&&(function(){var n=h,r=Symbol.for("react.element"),a=Symbol.for("react.portal"),l=Symbol.for("react.fragment"),s=Symbol.for("react.strict_mode"),d=Symbol.for("react.profiler"),i=Symbol.for("react.provider"),x=Symbol.for("react.context"),p=Symbol.for("react.forward_ref"),E=Symbol.for("react.suspense"),g=Symbol.for("react.suspense_list"),S=Symbol.for("react.memo"),u=Symbol.for("react.lazy"),_=Symbol.for("react.offscreen"),A=Symbol.iterator,M="@@iterator";function ee(e){if(e===null||typeof e!="object")return null;var t=A&&e[A]||e[M];return typeof t=="function"?t:null}var j=n.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED;function w(e){{for(var t=arguments.length,o=new Array(t>1?t-1:0),c=1;c<t;c++)o[c-1]=arguments[c];N("error",e,o)}}function N(e,t,o){{var c=j.ReactDebugCurrentFrame,y=c.getStackAddendum();y!==""&&(t+="%s",o=o.concat([y]));var b=o.map(function(v){return String(v)});b.unshift("Warning: "+t),Function.prototype.apply.call(console[e],console,b)}}var qe=!1,Ke=!1,Xe=!1,Ze=!1,Qe=!1,de;de=Symbol.for("react.module.reference");function et(e){return!!(typeof e=="string"||typeof e=="function"||e===l||e===d||Qe||e===s||e===E||e===g||Ze||e===_||qe||Ke||Xe||typeof e=="object"&&e!==null&&(e.$$typeof===u||e.$$typeof===S||e.$$typeof===i||e.$$typeof===x||e.$$typeof===p||e.$$typeof===de||e.getModuleId!==void 0))}function tt(e,t,o){var c=e.displayName;if(c)return c;var y=t.displayName||t.name||"";return y!==""?o+"("+y+")":o}function pe(e){return e.displayName||"Context"}function P(e){if(e==null)return null;if(typeof e.tag=="number"&&w("Received an unexpected object in getComponentNameFromType(). This is likely a bug in React. Please file an issue."),typeof e=="function")return e.displayName||e.name||null;if(typeof e=="string")return e;switch(e){case l:return"Fragment";case a:return"Portal";case d:return"Profiler";case s:return"StrictMode";case E:return"Suspense";case g:return"SuspenseList"}if(typeof e=="object")switch(e.$$typeof){case x:var t=e;return pe(t)+".Consumer";case i:var o=e;return pe(o._context)+".Provider";case p:return tt(e,e.render,"ForwardRef");case S:var c=e.displayName||null;return c!==null?c:P(e.type)||"Memo";case u:{var y=e,b=y._payload,v=y._init;try{return P(v(b))}catch{return null}}}return null}var D=Object.assign,U=0,ge,he,me,ve,_e,ye,Ee;function be(){}be.__reactDisabledLog=!0;function rt(){{if(U===0){ge=console.log,he=console.info,me=console.warn,ve=console.error,_e=console.group,ye=console.groupCollapsed,Ee=console.groupEnd;var e={configurable:!0,enumerable:!0,value:be,writable:!0};Object.defineProperties(console,{info:e,log:e,warn:e,error:e,group:e,groupCollapsed:e,groupEnd:e})}U++}}function nt(){{if(U--,U===0){var e={configurable:!0,enumerable:!0,writable:!0};Object.defineProperties(console,{log:D({},e,{value:ge}),info:D({},e,{value:he}),warn:D({},e,{value:me}),error:D({},e,{value:ve}),group:D({},e,{value:_e}),groupCollapsed:D({},e,{value:ye}),groupEnd:D({},e,{value:Ee})})}U<0&&w("disabledDepth fell below zero. This is a bug in React. Please file an issue.")}}var te=j.ReactCurrentDispatcher,re;function Y(e,t,o){{if(re===void 0)try{throw Error()}catch(y){var c=y.stack.trim().match(/\n( *(at )?)/);re=c&&c[1]||""}return`
2
+ `+re+e}}var ne=!1,J;{var at=typeof WeakMap=="function"?WeakMap:Map;J=new at}function xe(e,t){if(!e||ne)return"";{var o=J.get(e);if(o!==void 0)return o}var c;ne=!0;var y=Error.prepareStackTrace;Error.prepareStackTrace=void 0;var b;b=te.current,te.current=null,rt();try{if(t){var v=function(){throw Error()};if(Object.defineProperty(v.prototype,"props",{set:function(){throw Error()}}),typeof Reflect=="object"&&Reflect.construct){try{Reflect.construct(v,[])}catch(k){c=k}Reflect.construct(e,[],v)}else{try{v.call()}catch(k){c=k}e.call(v.prototype)}}else{try{throw Error()}catch(k){c=k}e()}}catch(k){if(k&&c&&typeof k.stack=="string"){for(var m=k.stack.split(`
3
+ `),I=c.stack.split(`
4
+ `),R=m.length-1,C=I.length-1;R>=1&&C>=0&&m[R]!==I[C];)C--;for(;R>=1&&C>=0;R--,C--)if(m[R]!==I[C]){if(R!==1||C!==1)do if(R--,C--,C<0||m[R]!==I[C]){var T=`
5
+ `+m[R].replace(" at new "," at ");return e.displayName&&T.includes("<anonymous>")&&(T=T.replace("<anonymous>",e.displayName)),typeof e=="function"&&J.set(e,T),T}while(R>=1&&C>=0);break}}}finally{ne=!1,te.current=b,nt(),Error.prepareStackTrace=y}var F=e?e.displayName||e.name:"",L=F?Y(F):"";return typeof e=="function"&&J.set(e,L),L}function ot(e,t,o){return xe(e,!1)}function st(e){var t=e.prototype;return!!(t&&t.isReactComponent)}function q(e,t,o){if(e==null)return"";if(typeof e=="function")return xe(e,st(e));if(typeof e=="string")return Y(e);switch(e){case E:return Y("Suspense");case g:return Y("SuspenseList")}if(typeof e=="object")switch(e.$$typeof){case p:return ot(e.render);case S:return q(e.type,t,o);case u:{var c=e,y=c._payload,b=c._init;try{return q(b(y),t,o)}catch{}}}return""}var V=Object.prototype.hasOwnProperty,we={},Se=j.ReactDebugCurrentFrame;function K(e){if(e){var t=e._owner,o=q(e.type,e._source,t?t.type:null);Se.setExtraStackFrame(o)}else Se.setExtraStackFrame(null)}function it(e,t,o,c,y){{var b=Function.call.bind(V);for(var v in e)if(b(e,v)){var m=void 0;try{if(typeof e[v]!="function"){var I=Error((c||"React class")+": "+o+" type `"+v+"` is invalid; it must be a function, usually from the `prop-types` package, but received `"+typeof e[v]+"`.This often happens because of typos such as `PropTypes.function` instead of `PropTypes.func`.");throw I.name="Invariant Violation",I}m=e[v](t,v,c,o,null,"SECRET_DO_NOT_PASS_THIS_OR_YOU_WILL_BE_FIRED")}catch(R){m=R}m&&!(m instanceof Error)&&(K(y),w("%s: type specification of %s `%s` is invalid; the type checker function must return `null` or an `Error` but returned a %s. You may have forgotten to pass an argument to the type checker creator (arrayOf, instanceOf, objectOf, oneOf, oneOfType, and shape all require an argument).",c||"React class",o,v,typeof m),K(null)),m instanceof Error&&!(m.message in we)&&(we[m.message]=!0,K(y),w("Failed %s type: %s",o,m.message),K(null))}}}var lt=Array.isArray;function ae(e){return lt(e)}function ut(e){{var t=typeof Symbol=="function"&&Symbol.toStringTag,o=t&&e[Symbol.toStringTag]||e.constructor.name||"Object";return o}}function ct(e){try{return Re(e),!1}catch{return!0}}function Re(e){return""+e}function Ce(e){if(ct(e))return w("The provided key is an unsupported type %s. This value must be coerced to a string before before using it here.",ut(e)),Re(e)}var Ae=j.ReactCurrentOwner,ft={key:!0,ref:!0,__self:!0,__source:!0},Ie,ke;function dt(e){if(V.call(e,"ref")){var t=Object.getOwnPropertyDescriptor(e,"ref").get;if(t&&t.isReactWarning)return!1}return e.ref!==void 0}function pt(e){if(V.call(e,"key")){var t=Object.getOwnPropertyDescriptor(e,"key").get;if(t&&t.isReactWarning)return!1}return e.key!==void 0}function gt(e,t){typeof e.ref=="string"&&Ae.current}function ht(e,t){{var o=function(){Ie||(Ie=!0,w("%s: `key` is not a prop. Trying to access it will result in `undefined` being returned. If you need to access the same value within the child component, you should pass it as a different prop. (https://reactjs.org/link/special-props)",t))};o.isReactWarning=!0,Object.defineProperty(e,"key",{get:o,configurable:!0})}}function mt(e,t){{var o=function(){ke||(ke=!0,w("%s: `ref` is not a prop. Trying to access it will result in `undefined` being returned. If you need to access the same value within the child component, you should pass it as a different prop. (https://reactjs.org/link/special-props)",t))};o.isReactWarning=!0,Object.defineProperty(e,"ref",{get:o,configurable:!0})}}var vt=function(e,t,o,c,y,b,v){var m={$$typeof:r,type:e,key:t,ref:o,props:v,_owner:b};return m._store={},Object.defineProperty(m._store,"validated",{configurable:!1,enumerable:!1,writable:!0,value:!1}),Object.defineProperty(m,"_self",{configurable:!1,enumerable:!1,writable:!1,value:c}),Object.defineProperty(m,"_source",{configurable:!1,enumerable:!1,writable:!1,value:y}),Object.freeze&&(Object.freeze(m.props),Object.freeze(m)),m};function _t(e,t,o,c,y){{var b,v={},m=null,I=null;o!==void 0&&(Ce(o),m=""+o),pt(t)&&(Ce(t.key),m=""+t.key),dt(t)&&(I=t.ref,gt(t,y));for(b in t)V.call(t,b)&&!ft.hasOwnProperty(b)&&(v[b]=t[b]);if(e&&e.defaultProps){var R=e.defaultProps;for(b in R)v[b]===void 0&&(v[b]=R[b])}if(m||I){var C=typeof e=="function"?e.displayName||e.name||"Unknown":e;m&&ht(v,C),I&&mt(v,C)}return vt(e,m,I,y,c,Ae.current,v)}}var oe=j.ReactCurrentOwner,Te=j.ReactDebugCurrentFrame;function $(e){if(e){var t=e._owner,o=q(e.type,e._source,t?t.type:null);Te.setExtraStackFrame(o)}else Te.setExtraStackFrame(null)}var se;se=!1;function ie(e){return typeof e=="object"&&e!==null&&e.$$typeof===r}function Oe(){{if(oe.current){var e=P(oe.current.type);if(e)return`
6
+
7
+ Check the render method of \``+e+"`."}return""}}function yt(e){return""}var Me={};function Et(e){{var t=Oe();if(!t){var o=typeof e=="string"?e:e.displayName||e.name;o&&(t=`
8
+
9
+ Check the top-level render call using <`+o+">.")}return t}}function Ne(e,t){{if(!e._store||e._store.validated||e.key!=null)return;e._store.validated=!0;var o=Et(t);if(Me[o])return;Me[o]=!0;var c="";e&&e._owner&&e._owner!==oe.current&&(c=" It was passed a child from "+P(e._owner.type)+"."),$(e),w('Each child in a list should have a unique "key" prop.%s%s See https://reactjs.org/link/warning-keys for more information.',o,c),$(null)}}function je(e,t){{if(typeof e!="object")return;if(ae(e))for(var o=0;o<e.length;o++){var c=e[o];ie(c)&&Ne(c,t)}else if(ie(e))e._store&&(e._store.validated=!0);else if(e){var y=ee(e);if(typeof y=="function"&&y!==e.entries)for(var b=y.call(e),v;!(v=b.next()).done;)ie(v.value)&&Ne(v.value,t)}}}function bt(e){{var t=e.type;if(t==null||typeof t=="string")return;var o;if(typeof t=="function")o=t.propTypes;else if(typeof t=="object"&&(t.$$typeof===p||t.$$typeof===S))o=t.propTypes;else return;if(o){var c=P(t);it(o,e.props,"prop",c,e)}else if(t.PropTypes!==void 0&&!se){se=!0;var y=P(t);w("Component %s declared `PropTypes` instead of `propTypes`. Did you misspell the property assignment?",y||"Unknown")}typeof t.getDefaultProps=="function"&&!t.getDefaultProps.isReactClassApproved&&w("getDefaultProps is only used on classic React.createClass definitions. Use a static property named `defaultProps` instead.")}}function xt(e){{for(var t=Object.keys(e.props),o=0;o<t.length;o++){var c=t[o];if(c!=="children"&&c!=="key"){$(e),w("Invalid prop `%s` supplied to `React.Fragment`. React.Fragment can only have `key` and `children` props.",c),$(null);break}}e.ref!==null&&($(e),w("Invalid attribute `ref` supplied to `React.Fragment`."),$(null))}}var Pe={};function De(e,t,o,c,y,b){{var v=et(e);if(!v){var m="";(e===void 0||typeof e=="object"&&e!==null&&Object.keys(e).length===0)&&(m+=" You likely forgot to export your component from the file it's defined in, or you might have mixed up default and named imports.");var I=yt();I?m+=I:m+=Oe();var R;e===null?R="null":ae(e)?R="array":e!==void 0&&e.$$typeof===r?(R="<"+(P(e.type)||"Unknown")+" />",m=" Did you accidentally export a JSX literal instead of a component?"):R=typeof e,w("React.jsx: type is invalid -- expected a string (for built-in components) or a class/function (for composite components) but got: %s.%s",R,m)}var C=_t(e,t,o,y,b);if(C==null)return C;if(v){var T=t.children;if(T!==void 0)if(c)if(ae(T)){for(var F=0;F<T.length;F++)je(T[F],e);Object.freeze&&Object.freeze(T)}else w("React.jsx: Static children should always be an array. You are likely explicitly calling React.jsxs or React.jsxDEV. Use the Babel transform instead.");else je(T,e)}if(V.call(t,"key")){var L=P(e),k=Object.keys(t).filter(function(It){return It!=="key"}),le=k.length>0?"{key: someKey, "+k.join(": ..., ")+": ...}":"{key: someKey}";if(!Pe[L+le]){var At=k.length>0?"{"+k.join(": ..., ")+": ...}":"{}";w(`A props object containing a "key" prop is being spread into JSX:
10
+ let props = %s;
11
+ <%s {...props} />
12
+ React keys must be passed directly to JSX without using spread:
13
+ let props = %s;
14
+ <%s key={someKey} {...props} />`,le,L,At,L),Pe[L+le]=!0}}return e===l?xt(C):bt(C),C}}function wt(e,t,o){return De(e,t,o,!0)}function St(e,t,o){return De(e,t,o,!1)}var Rt=St,Ct=wt;W.Fragment=l,W.jsx=Rt,W.jsxs=Ct})()),W}var Ve;function $t(){return Ve||(Ve=1,process.env.NODE_ENV==="production"?Z.exports=Dt():Z.exports=Lt()),Z.exports}var f=$t();const Ge="platform-kit",Ft=!1,ce={log:(...n)=>{},warn:(...n)=>{},error:(...n)=>{console.error(`[${Ge}]`,...n)}};class Ut extends h.Component{hasDispatched=!1;constructor(r){super(r),this.state={hasError:!1}}getMfeName(){if(this.props.mfeName)return this.props.mfeName;if(typeof window<"u"){const r=window;if(r.__MF_NAME__)return r.__MF_NAME__}return Ge}shouldShowDetails(){if(this.props.showDetails!==void 0)return this.props.showDetails;if(typeof sessionStorage<"u")try{const r=sessionStorage.getItem("config");if(r){const a=JSON.parse(r);if(a.showErrorDetails!==void 0)return a.showErrorDetails}}catch{}return Ft}dispatchError(r,a){if(this.hasDispatched||typeof window>"u")return;this.hasDispatched=!0;const l=this.getMfeName();ce.error("ErrorBoundary caught:",r);try{window.dispatchEvent(new CustomEvent("mfe-error",{detail:{mfeName:l,error:r.message||String(r),stack:r.stack,componentStack:a?.componentStack,timestamp:Date.now()},bubbles:!0}))}catch(s){ce.error("Failed to dispatch mfe-error event:",s)}}static getDerivedStateFromError(r){return{hasError:!0,error:r}}componentDidCatch(r,a){this.dispatchError(r,a),ce.error("Error info:",a.componentStack)}handleCopy=()=>{const r=`Error in ${this.getMfeName()}:
15
+ ${this.state.error?.message}
16
+ ${this.state.error?.stack}`;typeof navigator<"u"&&navigator.clipboard&&navigator.clipboard.writeText(r).then(()=>{alert("Ошибка скопирована в буфер обмена")}).catch(()=>{prompt("Скопируйте ошибку:",r)})};handleRetry=()=>{this.setState({hasError:!1,error:void 0}),this.hasDispatched=!1,typeof window<"u"&&window.location.reload()};handleGoHome=()=>{typeof window<"u"&&(window.location.href="/")};render(){if(this.state.hasError){const r=this.state.error?.message||"Unknown error",a=this.state.error?.stack||"",l=this.shouldShowDetails(),s=this.getMfeName();return f.jsxs("div",{style:{padding:"20px",textAlign:"center",color:"#d32f2f",fontFamily:"monospace",background:"#ffebee",border:"1px solid #ef5350",borderRadius:"4px",margin:"10px"},children:[f.jsxs("h2",{style:{fontSize:"16px",margin:"0 0 8px 0"},children:["⚠️ Ошибка в ",s]}),f.jsx("p",{style:{fontSize:"12px",margin:0},children:"Произошла ошибка в микрофронтенде. Сообщение отправлено в shell."}),l&&f.jsxs("details",{style:{whiteSpace:"pre-wrap",textAlign:"left",marginTop:"10px",background:"#fff",padding:"8px",borderRadius:"4px"},children:[f.jsx("summary",{style:{cursor:"pointer",fontWeight:"bold"},children:"Детали ошибки"}),f.jsxs("pre",{style:{fontSize:"11px",overflow:"auto",maxHeight:"150px",margin:"8px 0 0 0",padding:"8px",background:"#f5f5f5",borderRadius:"4px"},children:[r,a&&`
17
+
18
+ ${a}`]})]}),f.jsxs("div",{style:{marginTop:"12px",display:"flex",gap:"8px",justifyContent:"center"},children:[f.jsx("button",{onClick:this.handleCopy,style:{padding:"8px 12px",background:"#666",color:"white",border:"none",borderRadius:"4px",cursor:"pointer"},children:"📋 Копировать"}),f.jsx("button",{onClick:this.handleRetry,style:{padding:"8px 12px",background:"#d32f2f",color:"white",border:"none",borderRadius:"4px",cursor:"pointer"},children:"🔄 Обновить"}),f.jsx("button",{onClick:this.handleGoHome,style:{padding:"8px 12px",background:"#1976d2",color:"white",border:"none",borderRadius:"4px",cursor:"pointer"},children:"🏠 На главную"})]})]})}return this.props.children==null?null:this.props.children}}const O={info:"ℹ️",code:"💻",link:"🔗",user:"👤",clock:"🕐",apps:"📦",storage:"💾",tags:"🏷️",spreadsheet:"📊"};function Vt({mfeName:n}){const{data:r,isLoading:a,error:l}=We({mfeName:n}),[s,d]=h.useState(!1);if(a)return f.jsxs("span",{className:"text-muted small me-2",children:[f.jsx("span",{className:"me-1",children:"⏳"}),"Загрузка..."]});if(l||!r)return f.jsxs("span",{className:"text-muted small me-2",title:`Ошибка: ${l||"нет данных"}`,children:[f.jsx("span",{className:"me-1",children:"ℹ️"}),"N/A"]});const i=r,x=i.BUILD_VERSION||i.IMAGE_VERSION||i.APP_NAME||"N/A",p=[];return i.APP_NAME&&p.push({key:"APP_NAME",value:i.APP_NAME,label:"Приложение",icon:O.apps}),i.BUILD_VERSION&&p.push({key:"BUILD_VERSION",value:i.BUILD_VERSION,label:"Версия",icon:O.tags}),i.IMAGE_VERSION&&p.push({key:"IMAGE_VERSION",value:i.IMAGE_VERSION,label:"Образ",icon:O.storage}),i.GIT_COMMIT&&p.push({key:"GIT_COMMIT",value:i.GIT_COMMIT,label:"Commit",icon:O.spreadsheet}),i.GIT_BRANCH&&p.push({key:"GIT_BRANCH",value:i.GIT_BRANCH,label:"Ветка",icon:O.spreadsheet}),i.CI_COMMIT_AUTHOR&&p.push({key:"CI_COMMIT_AUTHOR",value:i.CI_COMMIT_AUTHOR,label:"Автор",icon:O.user}),i.CI_COMMIT_TIMESTAMP&&p.push({key:"CI_COMMIT_TIMESTAMP",value:i.CI_COMMIT_TIMESTAMP,label:"Дата",icon:O.clock}),i.CI_JOB_URL&&p.push({key:"CI_JOB_URL",value:i.CI_JOB_URL,label:"CI Job",icon:O.link}),i.CI_PIPELINE_URL&&p.push({key:"CI_PIPELINE_URL",value:i.CI_PIPELINE_URL,label:"Pipeline",icon:O.link}),i.CI_COMMIT_MESSAGE&&(i.CI_COMMIT_MESSAGE.length>60?i.CI_COMMIT_MESSAGE.substring(0,57)+"":i.CI_COMMIT_MESSAGE,p.push({key:"CI_COMMIT_MESSAGE",value:i.CI_COMMIT_MESSAGE,label:"Сообщение",icon:O.code})),f.jsxs("div",{style:{display:"inline-block",position:"relative"},children:[f.jsxs("button",{onClick:()=>d(!s),style:{background:"transparent",border:"none",cursor:"pointer",padding:"4px 8px",fontSize:"14px",display:"inline-flex",alignItems:"center"},title:"Информация о версии",children:[f.jsx("span",{className:"me-1",children:O.info}),f.jsx("span",{className:"text-dark",children:x})]}),s&&f.jsxs("div",{style:{position:"absolute",top:"100%",right:0,zIndex:1e3,minWidth:"300px",background:"white",border:"1px solid #dee2e6",borderRadius:"8px",boxShadow:"0 4px 12px rgba(0,0,0,0.15)",padding:"16px",marginTop:"4px"},children:[f.jsxs("div",{style:{marginBottom:"12px",paddingBottom:"8px",borderBottom:"1px solid #dee2e6",display:"flex",alignItems:"center"},children:[f.jsx("span",{className:"me-2",children:O.apps}),f.jsx("strong",{children:i.APP_NAME||n})]}),f.jsxs("div",{style:{marginBottom:"12px"},children:[f.jsx("span",{className:"text-muted",children:"Версия: "}),f.jsx("strong",{children:x})]}),p.length>0&&f.jsx("div",{style:{fontSize:"13px"},children:p.map(({key:E,value:g,label:S,icon:u})=>{const _=g.length>40&&(E.includes("URL")||E.includes("MESSAGE"))?`${g.substring(0,37)}...`:g;return f.jsxs("div",{style:{marginBottom:"8px",display:"flex",alignItems:"flex-start"},children:[f.jsx("span",{className:"me-2",style:{flexShrink:0},children:u}),f.jsxs("div",{style:{flex:1,minWidth:0},children:[f.jsx("div",{className:"small text-muted",children:S}),f.jsx("div",{className:"font-monospace small text-truncate",style:{maxWidth:"100%"},title:g,children:E.includes("URL")?f.jsx("a",{href:g,target:"_blank",rel:"noopener noreferrer",style:{color:"#007bff"},children:_}):_})]})]},E)})}),p.length===0&&f.jsx("div",{className:"text-center text-muted py-2 small",children:"Нет информации о версии"}),f.jsx("div",{style:{marginTop:"12px",paddingTop:"8px",borderTop:"1px solid #dee2e6",textAlign:"center",fontSize:"12px",color:"#6c757d"},children:"IngoBank DevOps Platform"}),f.jsx("button",{onClick:()=>d(!1),style:{position:"absolute",top:"8px",right:"8px",background:"transparent",border:"none",cursor:"pointer",fontSize:"16px",color:"#6c757d"},children:"×"})]}),s&&f.jsx("div",{onClick:()=>d(!1),style:{position:"fixed",top:0,left:0,right:0,bottom:0,zIndex:999}})]})}const fe="platform-kit",ze=h.createContext(null);function Bt({children:n}){const[r,a]=h.useState([]),l=typeof window<"u"&&window.__MF_NAME__||fe,s=h.useCallback((u,_,A,M)=>({id:`${Date.now()}-${Math.random().toString(36).substr(2,9)}`,type:u,title:_,message:A,mfeName:l,timestamp:Date.now(),duration:M}),[l]),d=h.useCallback(u=>{a(A=>[...A,u].slice(0,5));const _=u.duration||5e3;_>0&&setTimeout(()=>{a(A=>A.filter(M=>M.id!==u.id))},_)},[]),i=h.useCallback(u=>{const _=s(u.type,u.title,u.message,u.duration);d(_),typeof window<"u"&&window.dispatchEvent(new CustomEvent("mfe-notification",{detail:_,bubbles:!0}))},[s,d]),x=h.useCallback((u,_="Успешно")=>{i({type:"success",title:_,message:u})},[i]),p=h.useCallback((u,_="Ошибка")=>{i({type:"error",title:_,message:u})},[i]),E=h.useCallback((u,_="Предупреждение")=>{i({type:"warning",title:_,message:u})},[i]),g=h.useCallback((u,_="Информация")=>{i({type:"info",title:_,message:u})},[i]),S=h.useCallback(u=>{a(_=>_.filter(A=>A.id!==u))},[]);return h.useEffect(()=>{if(typeof window>"u")return;const u=_=>{const A=_.detail;if(A&&A.type&&A.title&&A.message){const M={...A,id:`${Date.now()}-${Math.random().toString(36).substr(2,9)}`};d(M)}};return window.addEventListener("mfe-notification",u),()=>{window.removeEventListener("mfe-notification",u)}},[d]),f.jsxs(ze.Provider,{value:{notify:i,notifySuccess:x,notifyError:p,notifyWarning:E,notifyInfo:g,removeNotification:S},children:[n,r.length>0&&f.jsx("div",{style:{position:"fixed",top:"80px",right:"20px",zIndex:9998,maxWidth:"400px",width:"100%",display:"flex",flexDirection:"column",gap:"8px",pointerEvents:"none"},children:r.map(u=>f.jsxs("div",{className:`notification notification-${u.type}`,style:{pointerEvents:"auto",padding:"12px 16px",borderRadius:"8px",background:u.type==="success"?"#d4edda":u.type==="error"?"#f8d7da":u.type==="warning"?"#fff3cd":"#d1ecf1",color:u.type==="success"?"#155724":u.type==="error"?"#721c24":u.type==="warning"?"#856404":"#0c5460",boxShadow:"0 4px 12px rgba(0,0,0,0.15)",display:"flex",alignItems:"flex-start",gap:"12px"},children:[f.jsxs("div",{style:{flex:1},children:[f.jsx("strong",{style:{display:"block",marginBottom:"4px"},children:u.title}),f.jsx("p",{style:{margin:0,fontSize:"14px"},children:u.message}),f.jsx("small",{style:{opacity:.7,fontSize:"12px"},children:u.mfeName})]}),f.jsx("button",{onClick:()=>S(u.id),style:{background:"transparent",border:"none",cursor:"pointer",fontSize:"18px",lineHeight:1,opacity:.5},children:"×"})]},u.id))})]})}function Wt(){const n=h.useContext(ze);return n||{notify:()=>{},notifySuccess:()=>{},notifyError:()=>{},notifyWarning:()=>{},notifyInfo:()=>{},removeNotification:()=>{}}}function Ht(n){if(typeof window>"u")return;const r=window.__MF_NAME__||fe,a={...n,mfeName:r,timestamp:Date.now()};window.dispatchEvent(new CustomEvent("mfe-notification",{detail:a,bubbles:!0}))}const Ye="platform-kit",H={log:(...n)=>{},warn:(...n)=>{},error:(...n)=>{console.error(`[${Ye}]`,...n)},info:(...n)=>{}};function Gt(){if(typeof window>"u")return{isAuthenticated:!1};const n=window.__SHELL_AUTH_INSTANCE__;if(n)return n;const r=window.__SHELL_AUTH__;return r?.authInstance?r.authInstance:{isAuthenticated:!1}}function zt(){if(typeof window>"u")return;const n=window.__SHELL_AUTH_INSTANCE__;n?.signinRedirect?n.signinRedirect().catch(r=>{H.error("Failed to redirect to login:",r)}):window.location.href="/"}function Yt(n){const r=n.response;let a="unknown";return r?r.status===401?a="unauthorized":r.status===403?a="forbidden":r.status===404?a="not_found":r.status>=500&&(a="server"):a="network",{message:r?.data?.message??n.message??"Unknown error",code:r?.data?.code,status:r?.status,type:a,timestamp:Date.now(),url:n.config?.url}}function Je(n={}){const r=n.name||Ye,a={log:(...s)=>H.log(`[API:${r}]`,...s),warn:(...s)=>H.warn(`[API:${r}]`,...s),error:(...s)=>H.error(`[API:${r}]`,...s),info:(...s)=>H.info(`[API:${r}]`,...s)},l=kt.create({timeout:n.timeout??1e4,baseURL:n.baseURL??"",headers:{"Content-Type":"application/json"}});return l.interceptors.request.use(s=>{const d=Gt();if(d?.isAuthenticated){const i=d.user?.profile?.access_token;i&&s.headers&&(s.headers.Authorization=`Bearer ${i}`,a.info("Auth token attached"))}else a.info("User not authenticated - request without token");return a.log(`${s.method?.toUpperCase()} ${s.url}`),s},s=>(a.error("Request interceptor error:",s.message),Promise.reject(s))),l.interceptors.response.use(s=>(a.log(`Response ${s.status}:`,s.config.url),s),s=>{const d=s.response?.status,i=s.config?.url;return d===401?(a.warn("401 Unauthorized - triggering re-auth"),zt()):d===403?a.warn("403 Forbidden - insufficient permissions"):d===404?a.warn("404 Not found:",i):d===429?a.warn("429 Rate limited"):d===500?a.error("500 Server error:",i):s.response?a.warn(`Error ${d}:`,s.message):a.error("Network error - backend may be unavailable:",i),Promise.reject(Yt(s))}),l}const G=Je();async function Jt(n,r,a=G){const l=await a.get(n,{params:r});return{data:l.data,status:l.status,ok:l.status>=200&&l.status<300}}async function qt(n,r,a=G){const l=await a.post(n,r);return{data:l.data,status:l.status,ok:l.status>=200&&l.status<300}}async function Kt(n,r,a=G){const l=await a.put(n,r);return{data:l.data,status:l.status,ok:l.status>=200&&l.status<300}}async function Xt(n,r=G){const a=await r.delete(n);return{data:a.data,status:a.status,ok:a.status>=200&&a.status<300}}function z(n,r){const a=r?.prefix?`[${r.prefix}]`:`[${n}]`;return{log:(...l)=>{},warn:(...l)=>{},error:(...l)=>{console.error(a,...l)},info:(...l)=>{}}}const Zt=z("platform-kit",{prefix:"AUTH"}),Qt=z("platform-kit",{prefix:"API"}),er=z("platform-kit",{prefix:"ERROR"}),tr=z("platform-kit",{prefix:"INFO"});exports.ErrorBoundary=Ut;exports.NotificationProvider=Bt;exports.VersionInfo=Vt;exports.api=G;exports.apiLogger=Qt;exports.authLogger=Zt;exports.createApiClient=Je;exports.createMfLogger=z;exports.del=Xt;exports.dispatchNotification=Ht;exports.errorLogger=er;exports.get=Jt;exports.getAuth=Q;exports.infoLogger=tr;exports.post=qt;exports.put=Kt;exports.useApi=Nt;exports.useInfoData=We;exports.useNotification=Wt;exports.usePermissions=Pt;exports.useShellAuth=Be;exports.useV1Config=Mt;
@@ -0,0 +1,10 @@
1
+ /**
2
+ * @ib-dop/platform-kit - Platform Development Kit for Microfrontends
3
+ *
4
+ * Unified toolkit for all microfrontends in the IB DOP Platform
5
+ */
6
+ export * from './types';
7
+ export * from './hooks';
8
+ export * from './components';
9
+ export * from './services';
10
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAGH,cAAc,SAAS,CAAA;AAGvB,cAAc,SAAS,CAAA;AAGvB,cAAc,cAAc,CAAA;AAG5B,cAAc,YAAY,CAAA"}