@dxos/util 0.8.4-main.c1de068 → 0.8.4-main.dedc0f3

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.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@dxos/util",
3
- "version": "0.8.4-main.c1de068",
3
+ "version": "0.8.4-main.dedc0f3",
4
4
  "description": "Temporary bucket for misc functions, which should graduate into separate packages.",
5
5
  "homepage": "https://dxos.org",
6
6
  "bugs": "https://github.com/dxos/dxos/issues",
@@ -10,12 +10,12 @@
10
10
  "type": "module",
11
11
  "exports": {
12
12
  ".": {
13
+ "types": "./dist/types/src/index.d.ts",
13
14
  "browser": "./dist/lib/browser/index.mjs",
14
15
  "node": {
15
16
  "require": "./dist/lib/node/index.cjs",
16
17
  "default": "./dist/lib/node-esm/index.mjs"
17
- },
18
- "types": "./dist/types/src/index.d.ts"
18
+ }
19
19
  }
20
20
  },
21
21
  "types": "dist/types/src/index.d.ts",
@@ -29,15 +29,15 @@
29
29
  "dependencies": {
30
30
  "lodash.get": "^4.4.2",
31
31
  "lodash.set": "^4.3.2",
32
- "@dxos/debug": "0.8.4-main.c1de068",
33
- "@dxos/invariant": "0.8.4-main.c1de068",
34
- "@dxos/keys": "0.8.4-main.c1de068",
35
- "@dxos/node-std": "0.8.4-main.c1de068"
32
+ "@dxos/debug": "0.8.4-main.dedc0f3",
33
+ "@dxos/node-std": "0.8.4-main.dedc0f3",
34
+ "@dxos/invariant": "0.8.4-main.dedc0f3",
35
+ "@dxos/keys": "0.8.4-main.dedc0f3"
36
36
  },
37
37
  "devDependencies": {
38
38
  "@types/lodash.get": "^4.4.9",
39
39
  "@types/lodash.set": "^4.3.9",
40
- "@dxos/crypto": "0.8.4-main.c1de068"
40
+ "@dxos/crypto": "0.8.4-main.dedc0f3"
41
41
  },
42
42
  "publishConfig": {
43
43
  "access": "public"
package/src/array.test.ts CHANGED
@@ -4,7 +4,7 @@
4
4
 
5
5
  import { describe, expect, test } from 'vitest';
6
6
 
7
- import { diff, intersection, distinctBy, partition } from './array';
7
+ import { diff, distinctBy, intersection, partition } from './array';
8
8
 
9
9
  describe('diff', () => {
10
10
  test('returns the difference between two sets', () => {
@@ -2,7 +2,7 @@
2
2
  // Copyright 2024 DXOS.org
3
3
  //
4
4
 
5
- import { describe, test, expect } from 'vitest';
5
+ import { describe, expect, test } from 'vitest';
6
6
 
7
7
  import { CircularBuffer } from './circular-buffer';
8
8
 
@@ -6,7 +6,7 @@ import { expect, test } from 'vitest';
6
6
 
7
7
  import { PublicKey } from '@dxos/keys';
8
8
 
9
- import { makeSet, makeMap } from './complex';
9
+ import { makeMap, makeSet } from './complex';
10
10
 
11
11
  const PulicKeySet = makeSet<PublicKey>(PublicKey.hash);
12
12
 
@@ -4,7 +4,7 @@
4
4
 
5
5
  import { expect, test } from 'vitest';
6
6
 
7
- import { createKeyPair, createId } from '@dxos/crypto';
7
+ import { createId, createKeyPair } from '@dxos/crypto';
8
8
  import { PublicKey } from '@dxos/keys';
9
9
 
10
10
  import { humanize } from './human-hash';
package/src/index.ts CHANGED
@@ -51,5 +51,6 @@ export * from './tracer';
51
51
  export * from './tree';
52
52
  export * from './types';
53
53
  export * from './uint8array';
54
+ export * from './unit';
54
55
  export * from './url';
55
56
  export * from './weak';
@@ -2,9 +2,9 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- import { describe, it, expect } from 'vitest';
5
+ import { describe, expect, it } from 'vitest';
6
6
 
7
- import { byPosition, type Position } from './position';
7
+ import { type Position, byPosition } from './position';
8
8
 
9
9
  type TestItem = {
10
10
  id: number;
package/src/safe-parse.ts CHANGED
@@ -6,7 +6,7 @@ export const safeParseInt = (value: string | undefined, defaultValue?: number) =
6
6
  try {
7
7
  const n = parseInt(value ?? '');
8
8
  return isNaN(n) ? defaultValue : n;
9
- } catch (err) {
9
+ } catch {
10
10
  return defaultValue;
11
11
  }
12
12
  };
@@ -22,13 +22,14 @@ export const safeParseFloat = (str: string, defaultValue?: number): number | und
22
22
  export const safeParseJson: {
23
23
  <T extends object>(data: string | undefined | null, defaultValue: T): T;
24
24
  <T extends object>(data: string | undefined | null): T | undefined;
25
- } = <T extends object>(data: string | undefined | null, defaultValue?: T) => {
25
+ } = <T extends object = any>(data: string | undefined | null, defaultValue?: T): T | undefined => {
26
26
  if (data && data.length > 0) {
27
27
  try {
28
28
  return JSON.parse(data);
29
- } catch (err) {
29
+ } catch {
30
30
  // no-op.
31
31
  }
32
32
  }
33
+
33
34
  return defaultValue;
34
35
  };
package/src/sort.test.ts CHANGED
@@ -4,7 +4,7 @@
4
4
 
5
5
  import { describe, expect, test } from 'vitest';
6
6
 
7
- import { compareMulti, compareScalar, compareObject, compareString } from './sort';
7
+ import { compareMulti, compareObject, compareScalar, compareString } from './sort';
8
8
 
9
9
  const data = [
10
10
  { i: 0, idx: 1, label: 'apple' },
@@ -14,6 +14,6 @@ describe('string', () => {
14
14
  - 3
15
15
  `;
16
16
 
17
- expect(text).to.eq('- 1\n- 2\n - 3\n');
17
+ expect(text).to.eq('- 1\n- 2\n - 3');
18
18
  });
19
19
  });
package/src/string.ts CHANGED
@@ -16,9 +16,28 @@ export const capitalize = (str: string): string => {
16
16
  /**
17
17
  * Remove leading space from multi-line strings.
18
18
  */
19
- export const trim = (strings: TemplateStringsArray, ...values: any[]) => {
20
- const full = String.raw(strings, ...values);
21
- const lines = full.replace(/^\n/, '').split('\n');
22
- const indent = Math.min(...lines.filter((l) => l.trim()).map((l) => l.match(/^ */)![0].length));
23
- return lines.map((l) => l.slice(indent)).join('\n');
24
- };
19
+ export function trim(strings: TemplateStringsArray, ...values: any[]) {
20
+ // First, build the raw result with relative indentation.
21
+ const raw = strings.reduce((out, str, i) => {
22
+ out += str;
23
+ if (i < values.length) {
24
+ const match = str.match(/(^|\n)([ \t]*)$/);
25
+ const baseIndent = match ? match[2] : '';
26
+ const val = String(values[i]).replace(/\r?\n/g, '\n' + baseIndent);
27
+ out += val;
28
+ }
29
+ return out;
30
+ }, '');
31
+
32
+ // Split into lines and trim leading/trailing blank lines.
33
+ const lines = raw.split('\n');
34
+
35
+ while (lines.length && !lines[0].trim()) lines.shift();
36
+ while (lines.length && !lines[lines.length - 1].trim()) lines.pop();
37
+
38
+ // Find smallest indent across all non-blank lines.
39
+ const minIndent = Math.min(...lines.filter((l) => l.trim()).map((l) => l.match(/^[ \t]*/)?.[0].length ?? 0));
40
+
41
+ // Remove that indent from all lines.
42
+ return lines.map((l) => l.slice(minIndent)).join('\n');
43
+ }
package/src/tree.test.ts CHANGED
@@ -6,7 +6,7 @@ import { describe, test } from 'vitest';
6
6
 
7
7
  import { PublicKey } from '@dxos/keys';
8
8
 
9
- import { stringifyTree, type TreeNode } from './tree';
9
+ import { type TreeNode, stringifyTree } from './tree';
10
10
 
11
11
  describe('Tree logging', () => {
12
12
  test('simple', () => {
package/src/types.ts CHANGED
@@ -47,6 +47,7 @@ export type Falsy = false | 0 | '' | null | undefined;
47
47
  * Use with filter chaining instead of filter(Boolean) to preserve type.
48
48
  * NOTE: To filter by type:
49
49
  * items.filter((item: any): item is RangeSet<Decoration> => item instanceof RangeSet)
50
+ * @deprecated Replace with Predicate.isTruthy
50
51
  */
51
52
  export const isNotFalsy = <T>(value: T): value is Exclude<T, Falsy> => !!value;
52
53
  export const isNonNullable = <T>(value: T | null | undefined): value is T => value != null;
@@ -2,7 +2,7 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import { test, expect, describe } from 'vitest';
5
+ import { describe, expect, test } from 'vitest';
6
6
 
7
7
  import { arrayToBuffer, arrayToString, bufferToArray, stringToArray } from './uint8array';
8
8
 
@@ -0,0 +1,25 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { describe, it } from 'vitest';
6
+
7
+ import { Unit } from './unit';
8
+
9
+ describe('url', () => {
10
+ it('duration', ({ expect }) => {
11
+ const tests: [number, string][] = [
12
+ [0, '0ms'],
13
+ [1, '1ms'],
14
+ [1_000, '1.0s'],
15
+ [2_887, '2.9s'],
16
+ [1_000 * 60, '1m'],
17
+ [1_000 * 60 + 1_000 * 5, '1m 5s'],
18
+ [1_000 * 60 * 60 + 1_000 * 60 * 5, '1h 5m'],
19
+ ];
20
+
21
+ tests.forEach(([input, output]) => {
22
+ expect(Unit.Duration(input)).toBe(output);
23
+ });
24
+ });
25
+ });
package/src/unit.ts ADDED
@@ -0,0 +1,52 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ type Unit = {
6
+ symbol: string;
7
+ quotient: number;
8
+ };
9
+
10
+ type Format = (n: number, precision?: number) => string;
11
+
12
+ const Formatter = (unit: Unit): Format => {
13
+ return (n: number, precision = 2) => {
14
+ const value = n / unit.quotient;
15
+ return `${value.toFixed(precision)}${unit.symbol}`;
16
+ };
17
+ };
18
+
19
+ export const Unit = {
20
+ // ms.
21
+ Hour: Formatter({ symbol: 'h', quotient: 60 * 60 * 1_000 }),
22
+ Minute: Formatter({ symbol: 'm', quotient: 60 * 1_000 }),
23
+ Second: Formatter({ symbol: 's', quotient: 1_000 }),
24
+ Millisecond: Formatter({ symbol: 'ms', quotient: 1 }),
25
+ Duration: (n: number) => {
26
+ const hours = Math.floor(n / (60 * 60 * 1_000));
27
+ const minutes = Math.floor((n % (60 * 60 * 1_000)) / (60 * 1_000));
28
+ if (hours) {
29
+ return minutes ? `${hours}h ${minutes}m` : `${hours}h`;
30
+ }
31
+
32
+ const seconds = Math.floor((n % (60 * 1_000)) / 1_000);
33
+ if (minutes) {
34
+ return seconds ? `${minutes}m ${seconds}s` : `${minutes}m`;
35
+ }
36
+
37
+ if (seconds) {
38
+ return `${(n / 1_000).toFixed(1)}s`;
39
+ }
40
+
41
+ return `${n}ms`;
42
+ },
43
+
44
+ // bytes (note KB via KiB).
45
+ Gigabyte: Formatter({ symbol: 'GB', quotient: 1_000 * 1_000 * 1_000 }),
46
+ Megabyte: Formatter({ symbol: 'MB', quotient: 1_000 * 1_000 }),
47
+ Kilobyte: Formatter({ symbol: 'KB', quotient: 1_000 }),
48
+
49
+ // general.
50
+ Thousand: Formatter({ symbol: 'k', quotient: 1_000 }),
51
+ Percent: Formatter({ symbol: '%', quotient: 1 / 100 }),
52
+ };
package/src/weak.test.ts CHANGED
@@ -2,7 +2,7 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import { test, expect, describe } from 'vitest';
5
+ import { describe, expect, test } from 'vitest';
6
6
 
7
7
  import { WeakDictionary } from './weak';
8
8