mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
Add several more util functions and tests (#4106)
This commit is contained in:
parent
0aabc9a5de
commit
5abc1c618c
@ -20,7 +20,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import * as uuid from "uuid";
|
import * as uuid from "uuid";
|
||||||
import type { Tuple } from "./utils";
|
import { tuple, Tuple } from "./utils";
|
||||||
|
|
||||||
export interface HotbarItem {
|
export interface HotbarItem {
|
||||||
entity: {
|
entity: {
|
||||||
@ -46,7 +46,7 @@ export const defaultHotbarCells = 12; // Number is chosen to easy hit any item w
|
|||||||
export function getEmptyHotbar(name: string, id: string = uuid.v4()): Hotbar {
|
export function getEmptyHotbar(name: string, id: string = uuid.v4()): Hotbar {
|
||||||
return {
|
return {
|
||||||
id,
|
id,
|
||||||
items: Array(defaultHotbarCells).fill(null) as Tuple<HotbarItem | null, typeof defaultHotbarCells>,
|
items: tuple.filled(defaultHotbarCells, null),
|
||||||
name,
|
name,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
59
src/common/utils/__tests__/tuple.test.ts
Normal file
59
src/common/utils/__tests__/tuple.test.ts
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
/**
|
||||||
|
* Copyright (c) 2021 OpenLens Authors
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy of
|
||||||
|
* this software and associated documentation files (the "Software"), to deal in
|
||||||
|
* the Software without restriction, including without limitation the rights to
|
||||||
|
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
||||||
|
* the Software, and to permit persons to whom the Software is furnished to do so,
|
||||||
|
* subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in all
|
||||||
|
* copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
||||||
|
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
||||||
|
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
||||||
|
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
||||||
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { tuple } from "../../utils";
|
||||||
|
|
||||||
|
describe("tuple tests", () => {
|
||||||
|
describe("zip()", () => {
|
||||||
|
it("should yield 0 times and return 1 tuple of empty arrays when given empty array", () => {
|
||||||
|
expect(tuple.zip([]).next()).toEqual({
|
||||||
|
done: true,
|
||||||
|
value: [[]],
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should yield 1 times and return 2 tuple of empty arrays when given one element array tuples", () => {
|
||||||
|
const i = tuple.zip([1], [2]);
|
||||||
|
|
||||||
|
expect(i.next()).toEqual({
|
||||||
|
done: false,
|
||||||
|
value: [1, 2]
|
||||||
|
});
|
||||||
|
expect(i.next()).toEqual({
|
||||||
|
done: true,
|
||||||
|
value: [[], []],
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should yield 1 times and return 2 tuple of partial arrays when given one element array tuples", () => {
|
||||||
|
const i = tuple.zip([1], [2, 3]);
|
||||||
|
|
||||||
|
expect(i.next()).toEqual({
|
||||||
|
done: false,
|
||||||
|
value: [1, 2]
|
||||||
|
});
|
||||||
|
expect(i.next()).toEqual({
|
||||||
|
done: true,
|
||||||
|
value: [[], [3]],
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@ -19,26 +19,11 @@
|
|||||||
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
export type Tuple<T, N extends number> = N extends N ? number extends N ? T[] : _TupleOf<T, N, []> : never;
|
|
||||||
type _TupleOf<T, N extends number, R extends unknown[]> = R["length"] extends N ? R : _TupleOf<T, N, [T, ...R]>;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
* A inference typed version of `Array(length).fill(value)`
|
||||||
* @param sources The source arrays
|
* @param length The number of entries
|
||||||
* @yields A tuple of the next element from each of the sources
|
* @param value The value of each of the indices
|
||||||
* @returns The tuple of all the sources as soon as at least one of the sources is exausted
|
|
||||||
*/
|
*/
|
||||||
export function* zipStrict<T, N extends number>(...sources: Tuple<T[], N>): Iterator<Tuple<T, N>, Tuple<T[], N>> {
|
export function filled<T>(length: number, value: T): T[] {
|
||||||
const maxSafeLength = sources.reduce((prev, cur) => Math.min(prev, cur.length), Number.POSITIVE_INFINITY);
|
return Array(length).fill(value);
|
||||||
|
|
||||||
if (!isFinite(maxSafeLength)) {
|
|
||||||
// There are no sources, thus just return
|
|
||||||
return [] as Tuple<T[], N>;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (let i = 0; i < maxSafeLength; i += 1) {
|
|
||||||
yield sources.map(source => source[i]) as Tuple<T, N>;
|
|
||||||
}
|
|
||||||
|
|
||||||
return sources.map(source => source.slice(maxSafeLength)) as Tuple<T[], N>;
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -61,5 +61,10 @@ export * from "./convertCpu";
|
|||||||
|
|
||||||
import * as iter from "./iter";
|
import * as iter from "./iter";
|
||||||
import * as array from "./array";
|
import * as array from "./array";
|
||||||
|
import * as tuple from "./tuple";
|
||||||
|
|
||||||
export { iter, array };
|
export {
|
||||||
|
iter,
|
||||||
|
array,
|
||||||
|
tuple,
|
||||||
|
};
|
||||||
|
|||||||
54
src/common/utils/tuple.ts
Normal file
54
src/common/utils/tuple.ts
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
/**
|
||||||
|
* Copyright (c) 2021 OpenLens Authors
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy of
|
||||||
|
* this software and associated documentation files (the "Software"), to deal in
|
||||||
|
* the Software without restriction, including without limitation the rights to
|
||||||
|
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
||||||
|
* the Software, and to permit persons to whom the Software is furnished to do so,
|
||||||
|
* subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in all
|
||||||
|
* copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
||||||
|
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
||||||
|
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
||||||
|
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
||||||
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { array } from "../utils";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A strict N-tuple of type T
|
||||||
|
*/
|
||||||
|
export type Tuple<T, N extends number> = N extends N ? number extends N ? T[] : _TupleOf<T, N, []> : never;
|
||||||
|
type _TupleOf<T, N extends number, R extends unknown[]> = R["length"] extends N ? R : _TupleOf<T, N, [T, ...R]>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Iterates over `sources` yielding full tuples until one of the tuple arrays
|
||||||
|
* is empty. Then it returns a tuple with the rest of each of tuples
|
||||||
|
* @param sources The source arrays
|
||||||
|
* @yields A tuple of the next element from each of the sources
|
||||||
|
* @returns The tuple of all the sources as soon as at least one of the sources is exausted
|
||||||
|
*/
|
||||||
|
export function* zip<T, N extends number>(...sources: Tuple<T[], N>): Iterator<Tuple<T, N>, Tuple<T[], N>> {
|
||||||
|
const maxSafeLength = Math.min(...sources.map(source => source.length));
|
||||||
|
|
||||||
|
for (let i = 0; i < maxSafeLength; i += 1) {
|
||||||
|
yield sources.map(source => source[i]) as Tuple<T, N>;
|
||||||
|
}
|
||||||
|
|
||||||
|
return sources.map(source => source.slice(maxSafeLength)) as Tuple<T[], N>;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a `length` tuple filled with copies of `value`
|
||||||
|
* @param length The size of the tuple
|
||||||
|
* @param value The value for each of the tuple entries
|
||||||
|
*/
|
||||||
|
export function filled<T, L extends number>(length: L, value: T): Tuple<T, L> {
|
||||||
|
return array.filled(length, value) as Tuple<T, L>;
|
||||||
|
}
|
||||||
@ -19,7 +19,6 @@
|
|||||||
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import _ from "lodash";
|
|
||||||
import type { LensApiRequest } from "../router";
|
import type { LensApiRequest } from "../router";
|
||||||
import { respondJson } from "../utils/http-responses";
|
import { respondJson } from "../utils/http-responses";
|
||||||
import type { Cluster } from "../cluster";
|
import type { Cluster } from "../cluster";
|
||||||
@ -33,8 +32,7 @@ export type IMetricsQuery = string | string[] | {
|
|||||||
};
|
};
|
||||||
|
|
||||||
// This is used for backoff retry tracking.
|
// This is used for backoff retry tracking.
|
||||||
const MAX_ATTEMPTS = 5;
|
const ATTEMPTS = [false, false, false, false, true];
|
||||||
const ATTEMPTS = [...(_.fill(Array(MAX_ATTEMPTS - 1), false)), true];
|
|
||||||
|
|
||||||
// prometheus metrics loader
|
// prometheus metrics loader
|
||||||
async function loadMetrics(promQueries: string[], cluster: Cluster, prometheusPath: string, queryParams: Record<string, string>): Promise<any[]> {
|
async function loadMetrics(promQueries: string[], cluster: Cluster, prometheusPath: string, queryParams: Record<string, string>): Promise<any[]> {
|
||||||
|
|||||||
@ -49,7 +49,7 @@ export class ServiceAccountsSecret extends React.Component<Props, State> {
|
|||||||
<>
|
<>
|
||||||
{!showToken && (
|
{!showToken && (
|
||||||
<>
|
<>
|
||||||
<span className="asterisks">{Array(16).fill("•").join("")}</span>
|
<span className="asterisks">{"•".repeat(16)}</span>
|
||||||
<Icon
|
<Icon
|
||||||
small material="lock_open"
|
small material="lock_open"
|
||||||
tooltip="Show value"
|
tooltip="Show value"
|
||||||
|
|||||||
@ -32,7 +32,7 @@ import type { Align, ListOnScrollProps } from "react-window";
|
|||||||
|
|
||||||
import { SearchStore, searchStore } from "../../../common/search-store";
|
import { SearchStore, searchStore } from "../../../common/search-store";
|
||||||
import { UserStore } from "../../../common/user-store";
|
import { UserStore } from "../../../common/user-store";
|
||||||
import { boundMethod, cssNames } from "../../utils";
|
import { array, boundMethod, cssNames } from "../../utils";
|
||||||
import { Spinner } from "../spinner";
|
import { Spinner } from "../spinner";
|
||||||
import { VirtualList } from "../virtual-list";
|
import { VirtualList } from "../virtual-list";
|
||||||
import { logStore } from "./log.store";
|
import { logStore } from "./log.store";
|
||||||
@ -234,7 +234,7 @@ export class LogList extends React.Component<Props> {
|
|||||||
render() {
|
render() {
|
||||||
const { isLoading } = this.props;
|
const { isLoading } = this.props;
|
||||||
const isInitLoading = isLoading && !this.logs.length;
|
const isInitLoading = isLoading && !this.logs.length;
|
||||||
const rowHeights = new Array(this.logs.length).fill(this.lineHeight);
|
const rowHeights = array.filled(this.logs.length, this.lineHeight);
|
||||||
|
|
||||||
if (isInitLoading) {
|
if (isInitLoading) {
|
||||||
return (
|
return (
|
||||||
|
|||||||
@ -20,7 +20,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import type { TableSortCallback } from "./table";
|
import type { TableSortCallback } from "./table";
|
||||||
import { array, Ordering, rectifyOrdering, sortCompare } from "../../utils";
|
import { Ordering, rectifyOrdering, sortCompare, tuple } from "../../utils";
|
||||||
|
|
||||||
export function getSorted<T>(rawItems: T[], sortingCallback: TableSortCallback<T> | undefined, orderByRaw: string): T[] {
|
export function getSorted<T>(rawItems: T[], sortingCallback: TableSortCallback<T> | undefined, orderByRaw: string): T[] {
|
||||||
if (typeof sortingCallback !== "function") {
|
if (typeof sortingCallback !== "function") {
|
||||||
@ -40,7 +40,7 @@ export function getSorted<T>(rawItems: T[], sortingCallback: TableSortCallback<T
|
|||||||
|
|
||||||
const leftSortBy = [left.sortBy].flat();
|
const leftSortBy = [left.sortBy].flat();
|
||||||
const rightSortBy = [right.sortBy].flat();
|
const rightSortBy = [right.sortBy].flat();
|
||||||
const zipIter = array.zipStrict(leftSortBy, rightSortBy);
|
const zipIter = tuple.zip(leftSortBy, rightSortBy);
|
||||||
let r = zipIter.next();
|
let r = zipIter.next();
|
||||||
|
|
||||||
for (; r.done === false; r = zipIter.next()) {
|
for (; r.done === false; r = zipIter.next()) {
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user