Skip to content

Commit

Permalink
docs: clarify that css "class" is "depType"
Browse files Browse the repository at this point in the history
  • Loading branch information
wraithgar committed Jun 22, 2022
1 parent 06e91a8 commit 8f51279
Show file tree
Hide file tree
Showing 3 changed files with 57 additions and 49 deletions.
13 changes: 11 additions & 2 deletions docs/content/using-npm/dependency-selectors.md
Original file line number Diff line number Diff line change
Expand Up @@ -26,11 +26,20 @@ The `npm query` commmand exposes a new dependency selector syntax (informed by &
- `#<name>` dependency selector (equivalent to `[name="..."]`)
- `#<name>@<version>` (equivalent to `[name=<name>]:semver(<version>)`)
- `,` selector list delimiter
- `.` class selector
- `:` pseudo class selector
- `.` dependency type selector
- `:` pseudo selector
- `>` direct decendent/child selector
- `~` sibling selector

#### Dependency Type Selectors

- `.prod` dependency found in the `dependencies` section of `package.json`, or is a child of said dependency
- `.dev` dependency found in the `devDependencies` section of `package.json`, or is a child of said dependency
- `.optional` dependency found in the `optionalDependencies` section of `package.json`, or has `"optional": true` set in its entry in the `peerDependenciesMeta` section of `package.json`, or a child of said dependency
- `.peer` dependency found in the `peerDependencies` section of `package.json`
- `.workspace` dependency found in the `workspaces` section of `package.json`
- `.bundled` dependency found in the `bundleDependencies` section of `package.json`, or is a child of said dependency

#### Pseudo Selectors
- [`:not(<selector>)`](https://developer.mozilla.org/en-US/docs/Web/CSS/:not)
- [`:has(<selector>)`](https://developer.mozilla.org/en-US/docs/Web/CSS/:has)
Expand Down
63 changes: 31 additions & 32 deletions workspaces/arborist/lib/query-selector-all.js
Original file line number Diff line number Diff line change
Expand Up @@ -7,10 +7,10 @@ const npa = require('npm-package-arg')
const minimatch = require('minimatch')
const semver = require('semver')

// handle results for parsed query asts, results are stored in a map that
// has a key that points to each ast selector node and stores the resulting
// array of arborist nodes as its value, that is essential to how we handle
// multiple query selectors, e.g: `#a, #b, #c` <- 3 diff ast selector nodes
// handle results for parsed query asts, results are stored in a map that has a
// key that points to each ast selector node and stores the resulting array of
// arborist nodes as its value, that is essential to how we handle multiple
// query selectors, e.g: `#a, #b, #c` <- 3 diff ast selector nodes
class Results {
#results = null
#currentAstSelector = null
Expand All @@ -33,9 +33,9 @@ class Results {
this.#results.set(this.#currentAstSelector, value)
}

// when collecting results to a root astNode, we traverse the list of
// child selector nodes and collect all of their resulting arborist nodes
// into a single/flat Set of items, this ensures we also deduplicate items
// when collecting results to a root astNode, we traverse the list of child
// selector nodes and collect all of their resulting arborist nodes into a
// single/flat Set of items, this ensures we also deduplicate items
collect (rootAstNode) {
const acc = new Set()
for (const n of rootAstNode.nodes) {
Expand Down Expand Up @@ -90,7 +90,7 @@ const retrieveNodesFromParsedAst = async ({
return String(pkg[attribute] || '').endsWith(value)
},
}))
const classesMap = new Map(Object.entries({
const depTypesMap = new Map(Object.entries({
'.prod' (prevResults) {
return Promise.resolve(prevResults.filter(node =>
[...node.edgesIn].some(edge => edge.prod)))
Expand Down Expand Up @@ -355,15 +355,15 @@ const retrieveNodesFromParsedAst = async ({
}))

// retrieves the initial items to which start the filtering / matching
// for most of the different types of recognized ast nodes, e.g: class,
// id, *, etc in different contexts we need to start with the current list
// of filtered results, for example a query for `.workspace` actually
// means the same as `*.workspace` so we want to start with the full
// inventory if that's the first ast node we're reading but if it appears
// in the middle of a query it should respect the previous filtered
// results, combinators are a special case in which we always want to
// have the complete inventory list in order to use the left-hand side
// ast node as a filter combined with the element on its right-hand side
// for most of the different types of recognized ast nodes, e.g: class (aka
// depType), id, *, etc in different contexts we need to start with the
// current list of filtered results, for example a query for `.workspace`
// actually means the same as `*.workspace` so we want to start with the full
// inventory if that's the first ast node we're reading but if it appears in
// the middle of a query it should respect the previous filtered results,
// combinators are a special case in which we always want to have the
// complete inventory list in order to use the left-hand side ast node as a
// filter combined with the element on its right-hand side
const getInitialItems = () => {
const firstParsed = currentAstNode.parent.nodes[0] === currentAstNode &&
currentAstNode.parent.parent.type === 'root'
Expand Down Expand Up @@ -419,16 +419,16 @@ const retrieveNodesFromParsedAst = async ({
results.currentResult =
await processPendingCombinator(prevResults, nextResults)
}
const classType = async () => {
const classFn = classesMap.get(String(currentAstNode))
if (!classFn) {
const depType = async () => {
const depTypeFn = depTypesMap.get(String(currentAstNode))
if (!depTypeFn) {
throw Object.assign(
new Error(`\`${String(currentAstNode)}\` is not a supported class.`),
{ code: 'EQUERYNOCLASS' }
new Error(`\`${String(currentAstNode)}\` is not a supported dependency type.`),
{ code: 'EQUERYNODEPTYPE' }
)
}
const prevResults = results.currentResult
const nextResults = await classFn(getInitialItems())
const nextResults = await depTypeFn(getInitialItems())
results.currentResult =
await processPendingCombinator(prevResults, nextResults)
}
Expand All @@ -449,8 +449,8 @@ const retrieveNodesFromParsedAst = async ({
if (!pseudoFn) {
throw Object.assign(
new Error(`\`${currentAstNode.value
}\` is not a supported pseudo-class.`),
{ code: 'EQUERYNOPSEUDOCLASS' }
}\` is not a supported pseudo selector.`),
{ code: 'EQUERYNOPSEUDO' }
)
}
const prevResults = results.currentResult
Expand All @@ -473,20 +473,19 @@ const retrieveNodesFromParsedAst = async ({
await processPendingCombinator(prevResults, nextResults)
}

// maps each of the recognized css selectors
// to a function that parses it
// maps each of the recognized css selectors to a function that parses it
const retrieveByType = new Map(Object.entries({
attribute,
class: classType,
class: depType,
combinator,
id,
pseudo,
selector,
universal,
}))

// walks through the parsed css query and update the
// current result after parsing / executing each ast node
// walks through the parsed css query and update the current result after
// parsing / executing each ast node
const astNodeQueue = new Set()
rootAstNode.walk((nextAstNode) => {
astNodeQueue.add(nextAstNode)
Expand All @@ -506,8 +505,8 @@ const retrieveNodesFromParsedAst = async ({
const querySelectorAll = async (targetNode, query) => {
const rootAstNode = parser(query)

// results is going to be a Map in which its values are the
// resulting items returned for each parsed css ast selector
// results is going to be a Map in which its values are the resulting items
// returned for each parsed css ast selector
const inventory = [...targetNode.root.inventory.values()]
const res = await retrieveNodesFromParsedAst({
initialItems: inventory,
Expand Down
30 changes: 15 additions & 15 deletions workspaces/arborist/test/query-selector-all.js
Original file line number Diff line number Diff line change
Expand Up @@ -178,28 +178,28 @@ t.test('query-selector-all', async t => {
const emptyRes = await q(tree, '')
t.same(emptyRes, [], 'empty query')

// missing pseudo-class
// missing pseudo selector
t.rejects(
q(tree, ':foo'),
{ code: 'EQUERYNOPSEUDOCLASS' },
'should throw on missing pseudo-class'
{ code: 'EQUERYNOPSEUDO' },
'should throw on missing pseudo selector'
)

// missing class
// missing depType
t.rejects(
q(tree, '.foo'),
{ code: 'EQUERYNOCLASS' },
'should throw on missing class'
{ code: 'EQUERYNODEPTYPE' },
'should throw on missing dep type'
)

// missing attribute matcher on :attr
t.rejects(
q(tree, ':attr(foo, bar)'),
{ code: 'EQUERYATTR' },
'should throw on missing attribute matcher on :attr pseudo-class'
'should throw on missing attribute matcher on :attr pseudo'
)

// :scope pseudo-class
// :scope pseudo
const [nodeFoo] = await q(tree, '#foo')
const scopeRes = await querySelectorAll(nodeFoo, ':scope')
t.same(scopeRes, ['foo@2.2.2'], ':scope')
Expand Down Expand Up @@ -302,7 +302,7 @@ t.test('query-selector-all', async t => {
[':missing', ['missing-dep@^1.0.0']],
[':private', ['b@1.0.0']],

// :not pseudo-class
// :not pseudo
[':not(#foo)', [
'query-selector-all-tests@1.0.0',
'a@1.0.0',
Expand Down Expand Up @@ -330,14 +330,14 @@ t.test('query-selector-all', async t => {
'b@1.0.0',
]],

// has pseudo-class
// has pseudo
[':root > *:has(* > #bar@1.4.0)', ['foo@2.2.2']],
['*:has(* > #bar@1.4.0)', ['foo@2.2.2']],
['*:has(> #bar@1.4.0)', ['foo@2.2.2']],
['.workspace:has(> * > #lorem)', ['a@1.0.0']],
['.workspace:has(* #lorem, ~ #b)', ['a@1.0.0']],

// is pseudo-class
// is pseudo
[':is(#a, #b) > *', ['bar@2.0.0', 'baz@1.0.0']],
// TODO: ipsum is not empty but it's child is missing
// so it doesn't return a result here
Expand All @@ -350,7 +350,7 @@ t.test('query-selector-all', async t => {
'dasher@2.0.0',
]],

// type pseudo-class
// type pseudo
[':type()', [
'query-selector-all-tests@1.0.0',
'a@1.0.0',
Expand Down Expand Up @@ -378,7 +378,7 @@ t.test('query-selector-all', async t => {
]],
[':type(git)', []],

// path pseudo-class
// path pseudo
[':path(node_modules/*)', [
'abbrev@1.1.1',
'bar@2.0.0',
Expand Down Expand Up @@ -410,7 +410,7 @@ t.test('query-selector-all', async t => {
'lorem@1.0.0',
]],

// semver pseudo-class
// semver pseudo
[':semver()', [
'query-selector-all-tests@1.0.0',
'a@1.0.0',
Expand Down Expand Up @@ -461,7 +461,7 @@ t.test('query-selector-all', async t => {
[':semver(=1.4.0)', ['bar@1.4.0']],
[':semver(1.4.0 || 2.2.2)', ['foo@2.2.2', 'bar@1.4.0']],

// attr pseudo-class
// attr pseudo
[':attr([name=dasher])', ['dasher@2.0.0']],
[':attr(dependencies, [bar="^1.0.0"])', ['foo@2.2.2']],
[':attr(dependencies, :attr([bar="^1.0.0"]))', ['foo@2.2.2']],
Expand Down

0 comments on commit 8f51279

Please sign in to comment.