Important: This documentation covers Yarn 1 (Classic).
For Yarn 2+ docs and migration guide, see yarnpkg.com.

Package detail

broccoli-funnel

broccolijs2.2mMIT3.0.8TypeScript support: definitely-typed

Broccoli plugin that allows you to filter files selected from an input node down based on regular expressions.

broccoli-plugin, javascript

readme

Broccoli Funnel

Build Status Build status

Given an input node, the Broccoli Funnel plugin returns a new node with only a subset of the files from the input node. The files can be moved to different paths. You can use regular expressions to select which files to include or exclude.

Documentation

funnel(inputNode, options)

inputNode {Single node}

A Broccoli node (formerly: "tree"). A node in Broccoli can be either a string that references a directory in your project or a node object returned from running another Broccoli plugin.

If your project has the following file structure:

.
├── Brocfile.js
└── src/
    ├── css/
    │   ├── reset.css
    │   └── todos.css
    ├── icons/
    │   ├── check-mark.png
    │   └── logo.jpg
    └── javascript/
        ├── app.js
        └── todo.js

You can select a subsection of the tree via Funnel:

const funnel = require('broccoli-funnel');
const cssFiles = funnel('src/css');

/*
  cssFiles contains the following files:

  ├── reset.css
  └── todos.css
*/

// export the node for Broccoli to begin processing
module.exports = cssFiles;

Options

srcDir {String}

A string representing the portion of the input node to start the funneling from. This will be the base path for any include/exclude regexps.

Default: '.', the root path of the input node.

If your project has the following file structure:

.
├── Brocfile.js
└── src/
    ├── css/
    │   ├── reset.css
    │   └── todos.css
    ├── icons/
    │   ├── check-mark.png
    │   └── logo.jpg
    └── javascript/
        ├── app.js
        └── todo.js

You can select a subsection of the node via Funnel:

const funnel = require('broccoli-funnel');
const merge = require('broccoli-merge-trees');

// root of our source files
const projectFiles = 'src';

/* get a new node of only files in the 'src/css' directory
  cssFiles contains the following files:

  ├── reset.css
  └── todos.css
*/
const cssFiles = funnel(projectFiles, {
  srcDir: 'css'
});

/* get a new node of only files in the 'src/icons' directory
  imageFiles contains the following files:

  ├── check-mark.png
  └── logo.jpg
*/
const imageFiles = funnel(projectFiles, {
  srcDir: 'icons'
});


module.exports = merge([cssFiles, imageFiles]);

destDir {String}

A string representing the destination path that filtered files will be copied to.

Default: '.', the root path of input node.

If your project has the following file structure:

.
├── Brocfile.js
└── src/
    ├── css/
    │   ├── reset.css
    │   └── todos.css
    ├── icons/
    │   ├── check-mark.png
    │   └── logo.jpg
    └── javascript/
        ├── app.js
        └── todo.js

You can select a subsection of the directory structure via Funnel and copy it to a new location:

const funnel = require('broccoli-funnel');

const cssFiles = funnel('src/css', {
  destDir: 'build'
});

/*
  cssFiles contains the following files:

  build/
  ├── reset.css
  └── todos.css
*/

module.exports = cssFiles;

allowEmpty {Boolean}

When using srcDir/destDir options only (aka no filtering via include/exclude options), if the srcDir were missing an error would be thrown. Setting allowEmpty to true, will prevent that error by creating an empty directory at the destination path.


include {Array of GlobStrings|RegExps|Functions}

One or more matcher expression (regular expression, glob string, or function). Files within the node whose names match this expression will be copied (with the location inside their parent directories preserved) to the destDir.

Default: [].

If your project has the following file structure

.
├── Brocfile.js
└── src/
    ├── css/
    │   ├── reset.css
    │   └── todos.css
    ├── icons/
    │   ├── check-mark.png
    │   └── logo.jpg
    └── javascript/
        ├── app.js
        └── todo.js

You can select files that match a glob expression and copy those subdirectories to a new location, preserving their location within parent directories:

const funnel = require('broccoli-funnel');

// finds all files that match /todo/ and moves them
// the destDir
const todoRelatedFiles = funnel('src', {
  include: ['**/todo*']
});

/*
  todoRelatedFiles contains the following files:
  .
  ├── css
  │   └── todos.css
  └── javascript
      └── todo.js
*/

module.exports = todoRelatedFiles;

exclude {Array of Glob Strings|Function}

One or more matcher expression (regular expression, glob string, or function). Files within the node whose names match this expression will not be copied to the destDir if they otherwise would have been.

Note, in the case when a file matches both an include and exclude pattern, the exclude pattern wins

Default: [].

If your project has the following file structure:

.
├── Brocfile.js
└── src/
    ├── css/
    │   ├── reset.css
    │   └── todos.css
    ├── icons/
    │   ├── check-mark.png
    │   └── logo.jpg
    └── javascript/
        ├── app.js
        └── todo.js

You can select files that match a glob expression and exclude them from copying:

const funnel = require('broccoli-funnel');

// finds all files in 'src' EXCEPT `todo.js` in any directory
// or sub-directory and adds them to a node.
const nobodyLikesTodosAnyway = funnel('src', {
  exclude: ['**/todo.js']
});

/*
  nobodyLikesTodosAnyway contains the following files:
  .
  ├── css
  │   └── reset.css
  ├── icons
  │   ├── check-mark.png
  │   └── logo.jpg
  └── javascript
      └── app.js
*/

module.exports = nobodyLikesTodosAnyway;

files {Array of Strings}

One or more relative file paths. Files within the node whose relative paths match will be copied (with the location inside their parent directories preserved) to the destDir.

Default: [].

If your project has the following file structure

.
├── Brocfile.js
└── src/
    ├── css/
    │   ├── reset.css
    │   └── todos.css
    ├── icons/
    │   ├── check-mark.png
    │   └── logo.jpg
    └── javascript/
        ├── app.js
        └── todo.js

You can select a specific list of files copy those subdirectories to a new location, preserving their location within parent directories:

const funnel = require('broccoli-funnel');

// finds these specific files and moves them to the destDir
const someFiles = funnel('src', {
  files: ['css/reset.css', 'icons/check-mark.png']
});

/*
  someFiles contains the following files:
  .
  ├── css
  │   └── reset.css
  └── icons
      └── check-mark.png
*/

module.exports = someFiles;

getDestinationPath {Function}

This method will get called for each file, receiving the currently processing relativePath as its first argument. The value returned from getDestinationPath will be used as the destination for the new node. This is a very simple way to rename files or move them from one path to another (replacing the need for broccoli-file-mover for example).

The return value of this method is cached for each input file. This means that getDestinationPath will only be called once per relativePath.

In the following example, getDestinationPath is used to move main.js to ember-metal.js:

const node = funnel('packages/ember-metal/lib', {
  destDir: 'ember-metal',

  getDestinationPath: function(relativePath) {
    if (relativePath === 'lib/main.js') {
      return 'ember-metal.js';
    }

    return relativePath;
  }
});

Extending Funnel

If you desire to extend funnel follow the below snippet

const { Funnel } = require('broccoli-funnel');
class CustomFunnel extends Funnel {
  // cutstom implementation
}

ZOMG!!! TESTS?!?!!?

I know, right?

Running the tests:

npm install
npm test

License

This project is distributed under the MIT license.

changelog

v3.0.8 (2021-06-17)

:bug: Bug Fix

  • #147 Ensure adding additional trees does not affect include/exclude results. (@rwjblue)

Committers: 1

v3.0.7 (2021-06-10)

:rocket: Enhancement

  • #146 Allow subclasses to add additional input nodes. (@rwjblue)

Committers: 1

v3.0.6 (2021-05-26)

:bug: Bug Fix

  • #141 Stop gathering instantiation stack manually. (@rwjblue)

:house: Internal

Committers: 1

v3.0.5 (2021-05-03)

:bug: Bug Fix

  • #136 Ensure current working directory files do not affect broccoli-funnel operation (@ef4)

:house: Internal

Committers: 2

v3.0.4 (2021-04-13)

:bug: Bug Fix

Committers: 1

  • Edward Faulkner (@ef4)

v3.0.3 (2020-05-09)

:bug: Bug Fix

:house: Internal

  • #131 Add release automation. (@rwjblue)
  • #130 Fix test to actually check generated files with destDir (@mmun)

Committers: 4

master

3.0.1

  • Update all deps
  • Removing all the fs operation. Use input/output
  • Migrating to input/output facade

3.0.0

  • [Breaking] Upgrading to latest broccoli-plugin (Breaking only because of node version drops)
  • [Breaking] Modernize code: Class syntax, Async await etc.
  • [Breaking] Drop Node 8, as that is EOL end of the month
  • [Breaking] Drop Unsupported Node's (4, 6, 7) and add newly supported nodes (10, 12)

2.0.2

  • Fix usage of allowEmpty when also using include / exclude

2.0.1

  • Fixes issue with double dots in file names (#8) by using fs.existSync directly

2.0.0

1.2.0

  • Improve excludes performance by using node-walk-sync for excludes when possible (#93)

1.1.0

  • Opt out of cache directory creation.

1.0.7

  • [PERF] when linking roots, only remove symlinks when absolutely needed.

1.0.6

  • bump heimdall

1.0.5

  • include only needed files

1.0.4

  • switch to heimdalljs-logger, allowing logs to show context within the broccoli graph

1.0.3

  • update fs-tree-diff, fixes "rename only file in directory bug", possible performance improvements.

1.0.2

  • update minimatch

1.0.1

  • fix annotations

1.0.0

  • improve performance by improving output stability
  • Do not use CoreObject
  • Derive from broccoli-plugin base class

0.2.3

  • Make new operator optional
  • Use new .rebuild API
  • Avoid mutating array arguments

0.2.2

No changelog beyond this point. Here be dragons.