-
-
Notifications
You must be signed in to change notification settings - Fork 212
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
fix: incorrect implementation of filterNodesByGroups (#326)
* fix by adding script and ts-nocheck * chore: format * remove ts-nocheck
- Loading branch information
Showing
3 changed files
with
196 additions
and
9 deletions.
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,192 @@ | ||
// Copyright © 2024 Ory Corp | ||
// SPDX-License-Identifier: Apache-2.0 | ||
import { | ||
UiNode, | ||
UiNodeAnchorAttributes, | ||
UiNodeAttributes, | ||
UiNodeGroupEnum, | ||
UiNodeImageAttributes, | ||
UiNodeInputAttributes, | ||
UiNodeInputAttributesTypeEnum, | ||
UiNodeScriptAttributes, | ||
UiNodeTextAttributes, | ||
} from "@ory/client" | ||
|
||
/** | ||
* Returns the node's label. | ||
* | ||
* @param node | ||
* @return label | ||
*/ | ||
export const getNodeLabel = (node: UiNode): string => { | ||
const attributes = node.attributes | ||
if (isUiNodeAnchorAttributes(attributes)) { | ||
return attributes.title.text | ||
} | ||
|
||
if (isUiNodeImageAttributes(attributes)) { | ||
return node.meta.label?.text || "" | ||
} | ||
|
||
if (isUiNodeInputAttributes(attributes)) { | ||
if (attributes.label?.text) { | ||
return attributes.label.text | ||
} | ||
} | ||
|
||
return node.meta.label?.text || "" | ||
} | ||
|
||
/** | ||
* A TypeScript type guard for nodes of the type <a> | ||
* | ||
* @param attrs | ||
*/ | ||
export function isUiNodeAnchorAttributes( | ||
attrs: UiNodeAttributes, | ||
): attrs is UiNodeAnchorAttributes & { node_type: "a" } { | ||
return attrs.node_type === "a" | ||
} | ||
|
||
/** | ||
* A TypeScript type guard for nodes of the type <img> | ||
* | ||
* @param attrs | ||
*/ | ||
export function isUiNodeImageAttributes( | ||
attrs: UiNodeAttributes, | ||
): attrs is UiNodeImageAttributes & { node_type: "img" } { | ||
return attrs.node_type === "img" | ||
} | ||
|
||
/** | ||
* A TypeScript type guard for nodes of the type <input> | ||
* | ||
* @param attrs | ||
*/ | ||
export function isUiNodeInputAttributes( | ||
attrs: UiNodeAttributes, | ||
): attrs is UiNodeInputAttributes & { node_type: "input" } { | ||
return attrs.node_type === "input" | ||
} | ||
|
||
/** | ||
* A TypeScript type guard for nodes of the type <span>{text}</span> | ||
* | ||
* @param attrs | ||
*/ | ||
export function isUiNodeTextAttributes( | ||
attrs: UiNodeAttributes, | ||
): attrs is UiNodeTextAttributes & { node_type: "text" } { | ||
return attrs.node_type === "text" | ||
} | ||
|
||
/** | ||
* A TypeScript type guard for nodes of the type <script> | ||
* | ||
* @param attrs | ||
*/ | ||
export function isUiNodeScriptAttributes( | ||
attrs: UiNodeAttributes, | ||
): attrs is UiNodeScriptAttributes & { node_type: "script" } { | ||
return attrs.node_type === "script" | ||
} | ||
|
||
/** | ||
* Returns a node's ID. | ||
* | ||
* @param attributes | ||
*/ | ||
export function getNodeId({ attributes }: UiNode) { | ||
if (isUiNodeInputAttributes(attributes)) { | ||
return attributes.name | ||
} else { | ||
return attributes.id | ||
} | ||
} | ||
|
||
/** | ||
* Return the node input attribute type | ||
* In <input> elements we have a variety of types, such as text, password, email, etc. | ||
* When the attribute is null or the `type` attribute is not present, we assume it has no defined type. | ||
* @param attr | ||
* @returns type of node | ||
*/ | ||
export const getNodeInputType = (attr: any): string => attr?.["type"] ?? "" | ||
|
||
export type FilterNodesByGroups = { | ||
nodes: Array<UiNode> | ||
groups?: Array<UiNodeGroupEnum | string> | UiNodeGroupEnum | string | ||
withoutDefaultGroup?: boolean | ||
attributes?: | ||
| Array<UiNodeInputAttributesTypeEnum | string> | ||
| UiNodeInputAttributesTypeEnum | ||
| string | ||
withoutDefaultAttributes?: boolean | ||
excludeAttributes?: | ||
| Array<UiNodeInputAttributesTypeEnum | string> | ||
| UiNodeInputAttributesTypeEnum | ||
| string | ||
} | ||
|
||
/** | ||
* Filters nodes by their groups and attributes. | ||
* If no filtering options are specified, all nodes are returned. | ||
* Will always add default nodes unless `withoutDefaultGroup` is true. | ||
* Will always add default attributes unless `withoutDefaultAttributes` is true. | ||
* @param {Object} filterNodesByGroups - An object containing the nodes and the filtering options. | ||
* @param {Array<UiNode>} filterNodesByGroups.nodes - An array of nodes. | ||
* @param {Array<UiNodeGroupEnum | string> | string} filterNodesByGroups.groups - An array or comma seperated strings of groups to filter by. | ||
* @param {boolean} filterNodesByGroups.withoutDefaultGroup - If true, will not add default nodes under the 'default' category. | ||
* @param {Array<UiNodeInputAttributesTypeEnum | string> | string} filterNodesByGroups.attributes - An array or comma seperated strings of attributes to filter by. | ||
* @param {boolean} filterNodesByGroups.withoutDefaultAttributes - If true, will not add default attributes such as 'hidden' and 'script'. | ||
*/ | ||
export const filterNodesByGroups = ({ | ||
nodes, | ||
groups, | ||
withoutDefaultGroup, | ||
attributes, | ||
withoutDefaultAttributes, | ||
excludeAttributes, | ||
}: FilterNodesByGroups) => { | ||
const search = (s: Array<string> | string | undefined) => { | ||
if (!s) return [] | ||
return typeof s === "string" ? s.split(",") : s | ||
} | ||
|
||
return nodes.filter(({ group, attributes: attr }) => { | ||
// if we have not specified any group or attribute filters, return all nodes | ||
if (!groups && !attributes && !excludeAttributes) return true | ||
|
||
const g = search(groups) || [] | ||
if (!withoutDefaultGroup) { | ||
g.push("default") | ||
} | ||
|
||
// filter the attributes | ||
const a = search(attributes) || [] | ||
if (!withoutDefaultAttributes) { | ||
// always add hidden fields e.g. csrf | ||
if (group.includes("default")) { | ||
a.push("hidden") | ||
} | ||
// automatically add the necessary fields for webauthn and totp | ||
if (group.includes("webauthn") || group.includes("totp")) { | ||
a.push("input", "script") | ||
} | ||
} | ||
|
||
// filter the attributes to exclude | ||
const ea = search(excludeAttributes) || [] | ||
|
||
const filterGroup = groups ? g.includes(group) : true | ||
const filterAttributes = attributes | ||
? a.includes(getNodeInputType(attr)) | ||
: true | ||
const filterExcludeAttributes = excludeAttributes | ||
? !ea.includes(getNodeInputType(attr)) | ||
: true | ||
|
||
return filterGroup && filterAttributes && filterExcludeAttributes | ||
}) | ||
} |