@alauda-fe/common
Version:
Alauda frontend team common codes.
173 lines • 31.7 kB
JavaScript
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"]}