diff options
| author | Hazelnoot <acomputerdog@gmail.com> | 2025-05-24 18:28:43 -0400 |
|---|---|---|
| committer | Hazelnoot <acomputerdog@gmail.com> | 2025-05-28 21:31:40 -0400 |
| commit | 373c60b52173e6ed8bcc01b3c4f1c6ac4734398a (patch) | |
| tree | 4c454bf94808b2b33dc4011e2614c283618a8efa /packages/backend/src/misc | |
| parent | add instance properties for persisted block data (diff) | |
| download | sharkey-373c60b52173e6ed8bcc01b3c4f1c6ac4734398a.tar.gz sharkey-373c60b52173e6ed8bcc01b3c4f1c6ac4734398a.tar.bz2 sharkey-373c60b52173e6ed8bcc01b3c4f1c6ac4734398a.zip | |
add diff-arrays utility for efficient array diffs
Diffstat (limited to 'packages/backend/src/misc')
| -rw-r--r-- | packages/backend/src/misc/diff-arrays.ts | 56 |
1 files changed, 56 insertions, 0 deletions
diff --git a/packages/backend/src/misc/diff-arrays.ts b/packages/backend/src/misc/diff-arrays.ts new file mode 100644 index 0000000000..1f6820481b --- /dev/null +++ b/packages/backend/src/misc/diff-arrays.ts @@ -0,0 +1,56 @@ +/* + * SPDX-FileCopyrightText: hazelnoot and other Sharkey contributors + * SPDX-License-Identifier: AGPL-3.0-only + */ + +export interface DiffResult<T> { + added: T[]; + removed: T[]; +} + +/** + * Calculates the difference between two snapshots of data. + * Null, undefined, and empty arrays are supported, and values do not have to be unique. + * Result sets are de-duplicated, and will be empty if no data was added or removed (respectively). + * @param dataBefore Array containing data before the change + * @param dataAfter Array containing data after the change + */ +export function diffArrays<T>(dataBefore: T[] | null | undefined, dataAfter: T[] | null | undefined): DiffResult<T> { + const before = dataBefore ? new Set(dataBefore) : null; + const after = dataAfter ? new Set(dataAfter) : null; + + // data before AND after => changed + if (before?.size && after?.size) { + const added: T[] = []; + const removed: T[] = []; + + for (const host of before) { + // before and NOT after => removed + if (!after.has(host)) { + removed.push(host); + } + } + + for (const host of after) { + // after and NOT before => added + if (!before.has(host)) { + added.push(host); + } + } + + return { added, removed }; + } + + // data ONLY before => all removed + if (before?.size) { + return { added: [], removed: Array.from(before) }; + } + + // data ONLY after => all added + if (after?.size) { + return { added: Array.from(after), removed: [] }; + } + + // data NEITHER before nor after => no change + return { added: [], removed: [] }; +} |