From 81257ba42279cb1ca9abf050936182651d2c4263 Mon Sep 17 00:00:00 2001 From: Riqwan Thamir Date: Tue, 17 Dec 2024 12:56:02 +0100 Subject: [PATCH 1/2] chore: address previous PR comments --- .../utils/src/common/filter-object-by-keys.ts | 25 +++++++++++ .../flatten-object-to-key-value-pairs.ts | 44 +++++++++++++++---- 2 files changed, 60 insertions(+), 9 deletions(-) diff --git a/packages/core/utils/src/common/filter-object-by-keys.ts b/packages/core/utils/src/common/filter-object-by-keys.ts index 63fed4c8e888f..245bd1fa45089 100644 --- a/packages/core/utils/src/common/filter-object-by-keys.ts +++ b/packages/core/utils/src/common/filter-object-by-keys.ts @@ -1,5 +1,30 @@ import { isDefined } from "./is-defined" +/* + Given an array of keys that are object paths (like product.categories.id), this function will + return an object that contains keys from those object paths + + Given an object: testObject = { + product: { + id: "test-product", + name: "Test Product", + categories: [{ + id: "test-category", + name: "Test Category" + }] + } + } + + filterObjectByKeys(testObject, ['product.categories.id']) will return + + { + product: { + categories: [{ + id: "test-category" + }] + } + } +*/ export function filterObjectByKeys(obj, paths) { function buildObject(paths) { const result = {} diff --git a/packages/core/utils/src/common/flatten-object-to-key-value-pairs.ts b/packages/core/utils/src/common/flatten-object-to-key-value-pairs.ts index cfa50f4e6f8f7..6976168a32a2e 100644 --- a/packages/core/utils/src/common/flatten-object-to-key-value-pairs.ts +++ b/packages/core/utils/src/common/flatten-object-to-key-value-pairs.ts @@ -1,7 +1,33 @@ +import { isObject } from "./is-object" + type NestedObject = { [key: string]: any } +/* + Given a deeply nested object that can be arrays or objects, this function will flatten + it to an object that is 1 level deep. + + Given an object: testObject = { + product: { + id: "test-product", + name: "Test Product", + categories: [{ + id: "test-category", + name: "Test Category" + }] + } + } + + flattenObjectToKeyValuePairs(testObject) will return + + { + "product.id": "test-product", + "product.name": "Test Product", + "product.categories.id": ["test-category"], + "product.categories.name": ["Test Category"] + } +*/ export function flattenObjectToKeyValuePairs(obj: NestedObject): NestedObject { const result: NestedObject = {} @@ -12,17 +38,17 @@ export function flattenObjectToKeyValuePairs(obj: NestedObject): NestedObject { ): string[][] { const paths: string[][] = [] - if (!obj || typeof obj !== "object") { + if (!obj || !isObject(obj)) { return paths } // If it's an array of objects, add this path - if (Array.isArray(obj) && obj.length > 0 && typeof obj[0] === "object") { + if (Array.isArray(obj) && obj.length > 0 && isObject(obj[0])) { paths.push(currentPath) } // Check all properties - if (typeof obj === "object") { + if (isObject(obj)) { Object.entries(obj as Record).forEach(([key, value]) => { const newPath = [...currentPath, key] paths.push(...findArrayPaths(value, newPath)) @@ -35,7 +61,7 @@ export function flattenObjectToKeyValuePairs(obj: NestedObject): NestedObject { // Extract array values at a specific path function getArrayValues(obj: unknown, path: string[]): unknown[] { const arrayObj = path.reduce((acc: unknown, key: string) => { - if (acc && typeof acc === "object") { + if (isObject(acc)) { return (acc as Record)[key] } return undefined @@ -48,7 +74,7 @@ export function flattenObjectToKeyValuePairs(obj: NestedObject): NestedObject { // Process non-array paths function processRegularPaths(obj: unknown, prefix = ""): void { - if (!obj || typeof obj !== "object") { + if (!obj || !isObject(obj)) { result[prefix] = obj return } @@ -57,7 +83,7 @@ export function flattenObjectToKeyValuePairs(obj: NestedObject): NestedObject { Object.entries(obj as Record).forEach(([key, value]) => { const newPrefix = prefix ? `${prefix}.${key}` : key - if (value && typeof value === "object" && !Array.isArray(value)) { + if (isObject(obj) && !Array.isArray(value)) { processRegularPaths(value, newPrefix) } else if (!Array.isArray(value)) { result[newPrefix] = value @@ -78,7 +104,7 @@ export function flattenObjectToKeyValuePairs(obj: NestedObject): NestedObject { // Get all possible keys from the array objects const keys = new Set() arrayObjects.forEach((item) => { - if (item && typeof item === "object") { + if (isObject(item)) { Object.keys(item as object).forEach((k) => keys.add(k)) } }) @@ -87,7 +113,7 @@ export function flattenObjectToKeyValuePairs(obj: NestedObject): NestedObject { keys.forEach((key) => { const values = arrayObjects .map((item) => { - if (item && typeof item === "object") { + if (isObject(item)) { return (item as Record)[key] } return undefined @@ -96,7 +122,7 @@ export function flattenObjectToKeyValuePairs(obj: NestedObject): NestedObject { if (values.length > 0) { const newPath = `${pathStr}.${key}` - if (values.every((v) => typeof v === "object" && !Array.isArray(v))) { + if (values.every((v) => isObject(v) && !Array.isArray(v))) { // If these are all objects, recursively process them const subObj = { [key]: values } const subResult = flattenObjectToKeyValuePairs(subObj) From e5c3b19363a84b8cb20ff62bceb1a52056865bbc Mon Sep 17 00:00:00 2001 From: Riqwan Thamir Date: Tue, 17 Dec 2024 15:03:36 +0100 Subject: [PATCH 2/2] chore: fix specs --- .../src/common/flatten-object-to-key-value-pairs.ts | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/packages/core/utils/src/common/flatten-object-to-key-value-pairs.ts b/packages/core/utils/src/common/flatten-object-to-key-value-pairs.ts index 6976168a32a2e..8a05cae81ad57 100644 --- a/packages/core/utils/src/common/flatten-object-to-key-value-pairs.ts +++ b/packages/core/utils/src/common/flatten-object-to-key-value-pairs.ts @@ -38,7 +38,7 @@ export function flattenObjectToKeyValuePairs(obj: NestedObject): NestedObject { ): string[][] { const paths: string[][] = [] - if (!obj || !isObject(obj)) { + if (!obj || typeof obj !== "object") { return paths } @@ -61,7 +61,7 @@ export function flattenObjectToKeyValuePairs(obj: NestedObject): NestedObject { // Extract array values at a specific path function getArrayValues(obj: unknown, path: string[]): unknown[] { const arrayObj = path.reduce((acc: unknown, key: string) => { - if (isObject(acc)) { + if (acc && isObject(acc)) { return (acc as Record)[key] } return undefined @@ -74,7 +74,7 @@ export function flattenObjectToKeyValuePairs(obj: NestedObject): NestedObject { // Process non-array paths function processRegularPaths(obj: unknown, prefix = ""): void { - if (!obj || !isObject(obj)) { + if (!obj || typeof obj !== "object") { result[prefix] = obj return } @@ -83,7 +83,7 @@ export function flattenObjectToKeyValuePairs(obj: NestedObject): NestedObject { Object.entries(obj as Record).forEach(([key, value]) => { const newPrefix = prefix ? `${prefix}.${key}` : key - if (isObject(obj) && !Array.isArray(value)) { + if (value && isObject(value) && !Array.isArray(value)) { processRegularPaths(value, newPrefix) } else if (!Array.isArray(value)) { result[newPrefix] = value @@ -104,7 +104,7 @@ export function flattenObjectToKeyValuePairs(obj: NestedObject): NestedObject { // Get all possible keys from the array objects const keys = new Set() arrayObjects.forEach((item) => { - if (isObject(item)) { + if (item && isObject(item)) { Object.keys(item as object).forEach((k) => keys.add(k)) } }) @@ -113,7 +113,7 @@ export function flattenObjectToKeyValuePairs(obj: NestedObject): NestedObject { keys.forEach((key) => { const values = arrayObjects .map((item) => { - if (isObject(item)) { + if (item && isObject(item)) { return (item as Record)[key] } return undefined