UNPKG

@alauda-fe/common

Version:

Alauda frontend team common codes.

173 lines 31.7 kB
import { inject } from '@angular/core'; import { toObservable, toSignal } from '@angular/core/rxjs-interop'; import { ActivatedRoute, Router } from '@angular/router'; import { NEVER, Subject, catchError, filter, interval, isObservable, map, of, startWith, switchMap, switchScan, take, withLatestFrom, } from 'rxjs'; import { WatchEvent } from '../api/public-api'; import { DOWNGRADE_WATCH_ENABLED, DOWNGRADE_WATCH_POLLING, publishRef, } from '../core/public-api'; export class K8SResourcePagedList { constructor(config) { this.config = config; this.reload$$ = new Subject(); this.addedUid = new Set(); this.deletedUid = new Set(); this.params$ = isObservable(this.config.params) ? this.config.params : toObservable(this.config.params); this.loadState$ = this.config.watcher && !DOWNGRADE_WATCH_ENABLED ? this.loadStateWithWatcher() : this.config.watcher || this.config.polling ? this.loadStateWithPolling() : this.sourceLoadState(); this.loading$ = this.loadState$.pipe(map(state => state.loading)); this.items$ = this.loadState$.pipe(map(state => state.items)); this.totalItems$ = this.loadState$.pipe(map(state => state.totalItems)); this.loadError$ = this.loadState$.pipe(map(state => state.loadError)); this.$loading = toSignal(this.loading$); this.$items = toSignal(this.items$); this.$totalItems = toSignal(this.totalItems$); this.$loadError = toSignal(this.loadError$); } reload() { this.reload$$.next(); } sourceLoadState() { return this.params$.pipe(map(parseListParams), switchMap(({ listParams, extraParams }) => { return this.reload$$.pipe(startWith(null), switchMap(() => { return this.config.fetcher(listParams, extraParams).pipe(map(list => ({ loading: false, loadSuccess: true, items: list.items ?? [], totalItems: list.metadata.totalItems ?? 0, })), startWith({ loading: true, loadSuccess: false, items: [], totalItems: 0, }), catchError((err) => of({ loading: false, loadSuccess: false, loadError: err, items: [], totalItems: 0, }))); })); }), publishRef()); } loadStateWithPolling() { return this.sourceLoadState().pipe(switchMap(sourceState => sourceState.loadSuccess ? interval(this.config.polling || DOWNGRADE_WATCH_POLLING).pipe(withLatestFrom(this.params$), map(([_, params]) => parseListParams(params)), switchMap(({ listParams, extraParams }) => this.config.fetcher(listParams, extraParams).pipe(map(list => ({ ...sourceState, items: list.items, totalItems: list.metadata.totalItems, })), catchError(() => NEVER))), startWith(sourceState)) : of(sourceState)), publishRef()); } loadStateWithWatcher() { return this.sourceLoadState().pipe(switchMap(sourceState => sourceState.loadSuccess ? this.params$.pipe(take(1), switchMap(params => { const watchParams = parseListParams({ keyword: params.keyword, extra: params.extra, }); const fetchParams = parseListParams(params); return this.config .watcher(watchParams.listParams, watchParams.extraParams) .pipe(filter(({ type, object }) => { switch (type) { case WatchEvent.Added: { if (this.addedUid.has(object.metadata.uid)) { return false; } this.addedUid.add(object.metadata.uid); return true; } case WatchEvent.Deleted: { if (this.deletedUid.has(object.metadata.uid)) { return false; } this.deletedUid.add(object.metadata.uid); return true; } default: { return true; } } }), switchScan((state, { type, object }) => { switch (type) { case WatchEvent.Modified: { return of({ ...state, items: state.items.map(item => item.metadata.uid === object.metadata.uid ? object : item), }); } case WatchEvent.Added: case WatchEvent.Deleted: { return this.config .fetcher(fetchParams.listParams, fetchParams.extraParams) .pipe(map(list => ({ ...state, items: list.items, totalItems: list.metadata.totalItems, })), catchError(() => NEVER)); } default: { return NEVER; } } }, sourceState)); }), startWith(sourceState)) : of(sourceState)), publishRef()); } } export function extractPagedListParams(defaultParams = {}, extraParams) { const route = inject(ActivatedRoute); const params$ = route.queryParams.pipe(map(params => ({ ...defaultParams, ...{ pageIndex: params.pageIndex || defaultParams.pageIndex || 0, pageSize: params.pageSize || defaultParams.pageSize || 20, }, ...(params.keyword ? { keyword: params.keyword } : {}), ...(params.sortField ? { sortField: params.sortField } : {}), ...(params.sortOrder ? { sortOrder: params.sortOrder } : {}), ...(params.fieldSelector ? { fieldSelector: params.fieldSelector } : {}), ...(params.labelSelector ? { labelSelector: params.labelSelector } : {}), ...(extraParams ? { extra: extraParams(params) } : {}), }))); return { stream: () => params$, signal: () => toSignal(params$), }; } export function queryListParams() { const router = inject(Router); const route = inject(ActivatedRoute); return (queryParams) => { router.navigate(['./'], { relativeTo: route, queryParamsHandling: 'merge', queryParams, }); }; } function parseListParams({ pageIndex, pageSize, sortField, sortOrder, keyword, fieldSelector: originFieldSelector, labelSelector, extra: extraParams, }) { const search = keyword ? `search=${keyword}` : ''; const sortBy = sortField ? `sortby=${sortOrder === 'desc' ? '-' : ''}${sortField}` : ''; const fieldSelector = [search, sortBy, originFieldSelector] .filter(v => !!v) .join(','); return { listParams: { limit: (pageSize || 20) + '', continue: +(pageIndex || 0) * +(pageSize || 20) + '', ...(labelSelector ? { labelSelector } : {}), ...(fieldSelector ? { fieldSelector } : {}), }, extraParams, }; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"k8s-resource-paged-list.js","sourceRoot":"","sources":["../../../../../libs/common/src/k8s-resource-list/k8s-resource-paged-list.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpErrorResponse } from '@angular/common/http';\nimport { Signal, inject } from '@angular/core';\nimport { toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { ActivatedRoute, Params, Router } from '@angular/router';\nimport {\n  NEVER,\n  Observable,\n  Subject,\n  catchError,\n  filter,\n  interval,\n  isObservable,\n  map,\n  of,\n  startWith,\n  switchMap,\n  switchScan,\n  take,\n  withLatestFrom,\n} from 'rxjs';\n\nimport { WatchAPIResponse, WatchEvent } from '../api/public-api';\nimport {\n  DOWNGRADE_WATCH_ENABLED,\n  DOWNGRADE_WATCH_POLLING,\n  KubernetesResource,\n  KubernetesResourceList,\n  Status,\n  publishRef,\n} from '../core/public-api';\n\nexport class K8SResourcePagedList<\n  R extends KubernetesResource = KubernetesResource,\n  P = unknown,\n> {\n  private readonly reload$$ = new Subject<void>();\n  private readonly addedUid = new Set<string>();\n  private readonly deletedUid = new Set<string>();\n\n  params$ = isObservable(this.config.params)\n    ? this.config.params\n    : toObservable(this.config.params);\n\n  loadState$ =\n    this.config.watcher && !DOWNGRADE_WATCH_ENABLED\n      ? this.loadStateWithWatcher()\n      : this.config.watcher || this.config.polling\n        ? this.loadStateWithPolling()\n        : this.sourceLoadState();\n\n  loading$ = this.loadState$.pipe(map(state => state.loading));\n  items$ = this.loadState$.pipe(map(state => state.items));\n  totalItems$ = this.loadState$.pipe(map(state => state.totalItems));\n  loadError$ = this.loadState$.pipe(map(state => state.loadError));\n\n  $loading = toSignal(this.loading$);\n  $items = toSignal(this.items$);\n  $totalItems = toSignal(this.totalItems$);\n  $loadError = toSignal(this.loadError$);\n\n  constructor(private readonly config: PagedListConfig<R, P>) {}\n\n  reload() {\n    this.reload$$.next();\n  }\n\n  private sourceLoadState() {\n    return this.params$.pipe(\n      map(parseListParams),\n      switchMap(({ listParams, extraParams }) => {\n        return this.reload$$.pipe(\n          startWith(null),\n          switchMap((): Observable<LoadState<R>> => {\n            return this.config.fetcher(listParams, extraParams).pipe(\n              map(list => ({\n                loading: false,\n                loadSuccess: true,\n                items: list.items ?? [],\n                totalItems: list.metadata.totalItems ?? 0,\n              })),\n              startWith({\n                loading: true,\n                loadSuccess: false,\n                items: [],\n                totalItems: 0,\n              }),\n              catchError((err: Status | HttpErrorResponse) =>\n                of({\n                  loading: false,\n                  loadSuccess: false,\n                  loadError: err,\n                  items: [],\n                  totalItems: 0,\n                }),\n              ),\n            );\n          }),\n        );\n      }),\n      publishRef(),\n    );\n  }\n\n  private loadStateWithPolling(): Observable<LoadState<R>> {\n    return this.sourceLoadState().pipe(\n      switchMap(sourceState =>\n        sourceState.loadSuccess\n          ? interval(this.config.polling || DOWNGRADE_WATCH_POLLING).pipe(\n              withLatestFrom(this.params$),\n              map(([_, params]) => parseListParams(params)),\n              switchMap(({ listParams, extraParams }) =>\n                this.config.fetcher(listParams, extraParams).pipe(\n                  map(list => ({\n                    ...sourceState,\n                    items: list.items,\n                    totalItems: list.metadata.totalItems,\n                  })),\n                  catchError(() => NEVER),\n                ),\n              ),\n              startWith(sourceState),\n            )\n          : of(sourceState),\n      ),\n      publishRef(),\n    );\n  }\n\n  private loadStateWithWatcher(): Observable<LoadState<R>> {\n    return this.sourceLoadState().pipe(\n      switchMap(sourceState =>\n        sourceState.loadSuccess\n          ? this.params$.pipe(\n              take(1),\n              switchMap(params => {\n                const watchParams = parseListParams({\n                  keyword: params.keyword,\n                  extra: params.extra,\n                });\n                const fetchParams = parseListParams(params);\n                return this.config\n                  .watcher(watchParams.listParams, watchParams.extraParams)\n                  .pipe(\n                    filter(({ type, object }) => {\n                      switch (type) {\n                        case WatchEvent.Added: {\n                          if (this.addedUid.has(object.metadata.uid)) {\n                            return false;\n                          }\n                          this.addedUid.add(object.metadata.uid);\n                          return true;\n                        }\n                        case WatchEvent.Deleted: {\n                          if (this.deletedUid.has(object.metadata.uid)) {\n                            return false;\n                          }\n                          this.deletedUid.add(object.metadata.uid);\n                          return true;\n                        }\n                        default: {\n                          return true;\n                        }\n                      }\n                    }),\n                    switchScan((state, { type, object }) => {\n                      switch (type) {\n                        case WatchEvent.Modified: {\n                          return of({\n                            ...state,\n                            items: state.items.map(item =>\n                              item.metadata.uid === object.metadata.uid\n                                ? object\n                                : item,\n                            ),\n                          });\n                        }\n                        case WatchEvent.Added:\n                        case WatchEvent.Deleted: {\n                          return this.config\n                            .fetcher(\n                              fetchParams.listParams,\n                              fetchParams.extraParams,\n                            )\n                            .pipe(\n                              map(list => ({\n                                ...state,\n                                items: list.items,\n                                totalItems: list.metadata.totalItems,\n                              })),\n                              catchError(() => NEVER),\n                            );\n                        }\n                        default: {\n                          return NEVER;\n                        }\n                      }\n                    }, sourceState),\n                  );\n              }),\n              startWith(sourceState),\n            )\n          : of(sourceState),\n      ),\n      publishRef(),\n    );\n  }\n}\n\nexport interface PagedListConfig<R, P> {\n  params: Signal<ListFetchParams<P>> | Observable<ListFetchParams<P>>;\n  fetcher: (\n    listParams: FetchSeed,\n    extraParams: P,\n  ) => Observable<KubernetesResourceList<R>>;\n  watcher?: (\n    listParams: FetchSeed,\n    extraParams: P,\n  ) => Observable<WatchAPIResponse<R>>;\n  polling?: number;\n}\n\nexport interface ListFetchParams<P = never> {\n  pageIndex?: number | string;\n  pageSize?: number | string;\n  sortField?: string;\n  sortOrder?: 'asc' | 'desc';\n  keyword?: string;\n  fieldSelector?: string;\n  labelSelector?: string;\n  extra?: P;\n}\n\nexport interface FetchSeed extends Record<string, string> {\n  limit?: string;\n  continue?: string;\n  fieldSelector?: string;\n}\n\ninterface LoadState<R> {\n  items: R[];\n  totalItems: number;\n  loading: boolean;\n  loadSuccess: boolean;\n  loadError?: HttpErrorResponse | Status;\n}\n\nexport function extractPagedListParams<P>(\n  defaultParams: ListFetchParams = {},\n  extraParams?: (queryParams: Params) => P,\n): Readonly<{\n  stream: () => Observable<ListFetchParams<P>>;\n  signal: () => Signal<ListFetchParams<P>>;\n}> {\n  const route = inject(ActivatedRoute);\n  const params$ = route.queryParams.pipe(\n    map(params => ({\n      ...defaultParams,\n      ...{\n        pageIndex: params.pageIndex || defaultParams.pageIndex || 0,\n        pageSize: params.pageSize || defaultParams.pageSize || 20,\n      },\n      ...(params.keyword ? { keyword: params.keyword } : {}),\n      ...(params.sortField ? { sortField: params.sortField } : {}),\n      ...(params.sortOrder ? { sortOrder: params.sortOrder } : {}),\n      ...(params.fieldSelector ? { fieldSelector: params.fieldSelector } : {}),\n      ...(params.labelSelector ? { labelSelector: params.labelSelector } : {}),\n      ...(extraParams ? { extra: extraParams(params) } : {}),\n    })),\n  );\n\n  return {\n    stream: () => params$,\n    signal: () => toSignal(params$),\n  };\n}\n\nexport function queryListParams() {\n  const router = inject(Router);\n  const route = inject(ActivatedRoute);\n\n  return (queryParams: Params) => {\n    router.navigate(['./'], {\n      relativeTo: route,\n      queryParamsHandling: 'merge',\n      queryParams,\n    });\n  };\n}\n\nfunction parseListParams<P>({\n  pageIndex,\n  pageSize,\n  sortField,\n  sortOrder,\n  keyword,\n  fieldSelector: originFieldSelector,\n  labelSelector,\n  extra: extraParams,\n}: ListFetchParams<P>) {\n  const search = keyword ? `search=${keyword}` : '';\n  const sortBy = sortField\n    ? `sortby=${sortOrder === 'desc' ? '-' : ''}${sortField}`\n    : '';\n  const fieldSelector = [search, sortBy, originFieldSelector]\n    .filter(v => !!v)\n    .join(',');\n  return {\n    listParams: {\n      limit: (pageSize || 20) + '',\n      continue: +(pageIndex || 0) * +(pageSize || 20) + '',\n      ...(labelSelector ? { labelSelector } : {}),\n      ...(fieldSelector ? { fieldSelector } : {}),\n    } as FetchSeed,\n    extraParams,\n  };\n}\n"]}