Skip to content

Commit

Permalink
fix: implicitly handle nested keys that are an array of values again
Browse files Browse the repository at this point in the history
  • Loading branch information
mart-jansink committed Dec 28, 2020
1 parent 1db161b commit 895c231
Show file tree
Hide file tree
Showing 2 changed files with 69 additions and 11 deletions.
45 changes: 45 additions & 0 deletions src/__tests__/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -198,6 +198,21 @@ const tests: Record<string, TestCase> = {
{favoriteIceCream: ['mint', 'chocolate']},
],
},
'can handle nested keys that are an array of values': {
input: [
[
{favorite: {iceCream: ['mint', 'chocolate']}},
{favorite: {iceCream: ['candy cane', 'brownie']}},
{favorite: {iceCream: ['birthday cake', 'rocky road', 'strawberry']}},
],
'cc',
{keys: ['favorite.iceCream']},
],
output: [
{favorite: {iceCream: ['candy cane', 'brownie']}},
{favorite: {iceCream: ['mint', 'chocolate']}},
],
},
'can handle nested keys that are an array of values with a wildcard': {
input: [
[
Expand All @@ -213,6 +228,36 @@ const tests: Record<string, TestCase> = {
{favorite: {iceCream: ['mint', 'chocolate']}},
],
},
'can handle nested keys that are an array of objects with a single wildcard': {
input: [
[
{favorite: {iceCream: [{tastes: ['vanilla', 'mint']}, {tastes: ['vanilla', 'chocolate']}]}},
{favorite: {iceCream: [{tastes: ['vanilla', 'candy cane']}, {tastes: ['vanilla', 'brownie']}]}},
{favorite: {iceCream: [{tastes: ['vanilla', 'birthday cake']}, {tastes: ['vanilla', 'rocky road']}, {tastes: ['strawberry']}]}},
],
'cc',
{keys: ['favorite.iceCream.*.tastes']},
],
output: [
{favorite: {iceCream: [{tastes:['vanilla', 'candy cane']}, {tastes:['vanilla', 'brownie']}]}},
{favorite: {iceCream: [{tastes:['vanilla', 'mint']}, {tastes:['vanilla', 'chocolate']}]}},
],
},
'can handle nested keys that are an array of objects with two wildcards': {
input: [
[
{favorite: {iceCream: [{tastes: ['vanilla', 'mint']}, {tastes: ['vanilla', 'chocolate']}]}},
{favorite: {iceCream: [{tastes: ['vanilla', 'candy cane']}, {tastes: ['vanilla', 'brownie']}]}},
{favorite: {iceCream: [{tastes: ['vanilla', 'birthday cake']}, {tastes: ['vanilla', 'rocky road']}, {tastes: ['strawberry']}]}},
],
'cc',
{keys: ['favorite.iceCream.*.tastes.*']},
],
output: [
{favorite: {iceCream: [{tastes:['vanilla', 'candy cane']}, {tastes:['vanilla', 'brownie']}]}},
{favorite: {iceCream: [{tastes:['vanilla', 'mint']}, {tastes:['vanilla', 'chocolate']}]}},
],
},
'can handle keys with a maxRanking': {
input: [
[
Expand Down
35 changes: 24 additions & 11 deletions src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -362,8 +362,12 @@ function getItemValues<ItemType>(
// @ts-expect-error just like below...
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
value = item[key];
} else {
value = getNestedValue<ItemType>(key, item)
} else if (key.includes(".")) {
// eslint-disable-next-line @typescript-eslint/no-unsafe-call
return getNestedValue<ItemType>(key, item)
}
else {
value = null
}

// because `value` can also be undefined
Expand All @@ -377,27 +381,23 @@ function getItemValues<ItemType>(
}

/**
* Given key: "foo.bar.baz"
* Given path: "foo.bar.baz"
* And item: {foo: {bar: {baz: 'buzz'}}}
* -> 'buzz'
* @param key a dot-separated set of keys
* @param path a dot-separated set of keys
* @param item the item to get the value from
*/
function getNestedValue<ItemType>(
key: string,
path: string,
item: ItemType,
): string | Array<string> | null {
): Array<string> {
// @ts-expect-error really have no idea how to type this properly...
return key.split('.').reduce((nestedItems: Array<object | string | null>, nestedKey: string): Array<object | string> => {
const result = path.split('.').reduce((nestedItems: Array<object | string | null>, nestedKey: string): Array<object | string> => {
return nestedItems.reduce((values: Array<object | string>, nestedItem: Array<object | string> | object | string | null): Array<object | string> => {
if (nestedItem == null) {
return values
}

if (nestedKey === "*") {
return values.concat(nestedItem)
}

if (Object.hasOwnProperty.call(nestedItem,nestedKey)) {
// @ts-expect-error and here again...
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
Expand All @@ -406,10 +406,23 @@ function getNestedValue<ItemType>(
values.push(nestedValue)
}
}
else if (nestedKey === "*") {
// ensure that values is an array
values = values.concat(nestedItem)
}

return values
}, [])
}, [item])

if (Array.isArray(result[0])) {
// @ts-expect-error just like above because we need to flatten the result to
// keep allowing the implicit wildcard for an array of strings at the end of
// the path; don't use `.flat()` because that's not available in node.js v10
return [].concat(...result)
}
// @ts-expect-error because somehow this is not a string[]
return result
}

/**
Expand Down

0 comments on commit 895c231

Please sign in to comment.