Write in front:

In the development of the project, the online environment needs to introduce some statistics and print log JS files. But for a development environment, speeding up packaging to reduce page rendering time is critical. I decided to write a simple loader to load some resources on demand, depending on the development environment.

For example, in index.js, use the custom function envLoader to add resources

index.js

/ /... envLoader( '/vendor/log.js' ) //......Copy the code

In order to complete the load on demand function. Plan to use a custom loader. The implementation idea is as follows:

  1. Add js loader to process index.js
  2. Parse the envLoader function
  3. Take the parameters passed in and determine whether to load them based on the environment.

Learn the working principle of WebPack Loader based on the Loader API on the official website.

The following API will be used

  • loader-utils
  • schema-utils
  • this.async
  • this.cacheable
  • getOptions
  • validateOptions
  • urlToRequest

Start loading your own loader (^ – ^)V

Webpack Loader

Loader is used to convert the module source code. Loader allows you to preprocess files when importing or “loading” modules. As such, Loader is similar to “tasks” in other build tools and provides a powerful way to handle the front-end build steps. Loader can convert files from different languages (such as TypeScript) to JavaScript, or convert inline images to data urls.

First, basic usage

Loader is a Node module exported as a function. This function is called when the Loader transforms the resource. The given function calls the Loader API and is accessed through the This context.

Loader is a Node module, so its basic form is as follows

module.exports = function(source) {
  return source;
};
Copy the code
  • Loader can only pass in a string containing the contents of the resource file (source)
  • If the loader is synchronous, it can be usedreturnorThis callback (err, value...)Return the code
  • Asynchronous Loader: In an asynchronous module, the callback method provided by the Loader API needs to be called when sending backthis.asyncTo get the callback function:
module.exports = function(content, map, meta) {
  var callback = this.async();
  someAsyncOperation(content, function(err, result) {
    if (err) return callback(err);
    callback(null, result, map, meta);
  });
};
Copy the code

Add loader to WebPack

The official website describes how to allocate a single loader and multiple loaders.

The path.resolve method is used to add a path to the Loader. You can also use resolveloader. modules to configure paths for multiple Loaders.

Webpack searches for loaders in these directories. I create a new loaders local directory in the project and modify the file as follows:

webpack.config.js

module.exports = { //... ResolveLoader: {modules: ['node_modules', path.resolve(__dirname, 'SRC ', 'loaders')]}, module: { rules:[ { test: /\.js$/, use: [ { loader: 'env-loader', options: { env: process.env.NODE_ENV } }, { loader:'babel-loader', options: { presets: ['env','es2015','react'], } }, ] }] }, //... };Copy the code

Note: Loader executes from right to left in a chain manner, with the result of the previous loader passing to the next loader

Package. json defines commands to package according to the environment

  "scripts": {
    "webpack": "cross-env NODE_ENV=development webpack-dev-server --open --mode development",
    "test": "cross-env NODE_ENV=test webpack --mode development",
    "dev": "cross-env NODE_ENV=dev webpack --mode development",
    "prd": "cross-env NODE_ENV=prd webpack --mode development",
    "boot":"cross-env NODE_ENV=boot webpack --mode development"
  },
Copy the code

Set NODE_ENV to distinguish between dev and PRD environments.

Q1: How do I obtain the environment parameters set in the command
A1:process.envNODE_ENV is available on the webpack.config.js object. When env-loader is introduced in webpack.config.js, the parameters can be passed to the loaderoptionsOptions.

webpack.config.js

{
   loader: 'env-loader',
   options: {
       env: process.env.NODE_ENV
   }
},
Copy the code

3. Use the Loader tool library to parse the Loader parameters

  • Loader – utils package. It provides many useful tools, but one of the most common is to get the options passed to the Loader
  • The schema-utils package works with loader-utils to ensure that the loader option is verified with the JSON Schema structure

Loader getOptions {env:’dev’} The schema-utils package and loader-utils package are used to ensure that the Loader option is verified with the JSON Schema structure. Add these two packages to index.js:

env-loader/index.js

const loaderUtils = require('loader-utils')
const validate = require('schema-utils');

let json = {
    "type": "object",
    "properties": {
        "content": {
            "type": "string"
        }
    }
}

module.exports = function(source) {
    this.cacheable();
    let callback = this.async();
    let options = loaderUtils.getOptions(this) //{env:'dev'}
    validate(json, options, "env-loader");
}
Copy the code

Use esprima to parse THE JS node

Esprima Parser converts JS programs into a syntax tree (AST) that describes the syntax structure of the program. The resulting syntax tree is useful for a variety of purposes, from program transformation to static program analysis.

I wrote an article about AST before. Click the link to check it out. I won’t go into details here.

Usage:

esprima.parseScript(input, config, delegate)
esprima.parseModule(input, config, delegate)
Copy the code
  • Input is a string representing the program to be parsed
  • Config is an object for customizing the parsing behavior (optional)
  • Delegate is the callback function invoked for each node (optional)

With source as the input parameter, the program will be parsed into an AST.

Node returns Syntax for each node. Meta is the node’s location in the program.

esprima.parseModule(source, {}, async(node, meta)=> {
    console.log(node.meta)
    //....
})
Copy the code

The analytical results are as follows:

Analyze whether the Syntax of each node meets the criteria, which determines the type of node and the name===’envLoader’ and type===’Identifier’ of the function callee being executed, and process the nodes that meet the criteria.

function judgeType(node) { return (node.type === 'CallExpression') && (node.callee.name === 'envLoader') && (node.callee.type === 'Identifier') } if (judgeType(node)) { flag = true node.arguments.map(argument=>{ entries.push({ val: argument.value, start: meta.start.offset, end: meta.end.offset }); })}Copy the code

5. File path processing

In the node analysis, get the external resource address passed in the custom envLoader function, and then re-loader.

Loaders typically use require() or import methods. This is because Webpack calculates the hash before converting the module path to the module ID, so we must avoid absolute paths to ensure consistent hashing across compilations.

Do not insert absolute paths in module code, because when the project root path changes, so does the file absolute path.

Loaderutils.urltorequest can convert some resource urls into webpack module requests.
Let downloadPath = path.resolve(process.cwd(), 'SRC ') if(env ==' PRD '){// If the environment is PRD // Convert the request to module const saveUrl = with loaderUtils loaderUtils.urlToRequest(`${extName}`,downloadPath); Var replaceText = 'import' ${saveUrl} ' '}else{var replaceText = 'function ' EnvLoad (){}'} // replace source = source.replace(transText, replaceText);Copy the code

Six, test,

By completing the above steps, you have developed a simple Loader that can run locally. Let’s do a simple unit test to make sure the Loader works the way we expect.

We will use the Jest framework. Then we need to install babel-Jest and some preset environments (presets) that allow us to use import/export and async/await.

6.1 Installation Dependencies

npm install --save-dev jest babel-jest babel-preset-env
Copy the code

.babelrc

{
  "presets": [[
    "env",
    {
      "targets": {
        "node": "4"
      }
    }
  ]]
}
Copy the code

Our loader will process.js files and place any instances in

envLoader('xxx')
Copy the code

Replace this with function envLoad(){} in development and import ‘path /xxx.js’ in production.

Create example.js under the test folder

envLoader(
    '/vendor/lodash.min.js'
)
Copy the code

We will use the Node.js API and memory-fs to perform webpack.

npm install --save-dev webpack memory-fss
Copy the code

test/compiler.js

import path from 'path';
import webpack from 'webpack';
import memoryfs from 'memory-fs';

export default (fixture, options = {}) => {
    const compiler = webpack({
        context: __dirname,
        entry: `./${fixture}`,
        output: {
            path: path.resolve(__dirname),
            filename: 'bundle.js',
        },
        module: {
            rules: [{
                test: /\.js$/,
                use: {
                    loader: path.resolve(__dirname, '../src/loaders/env-loader'),
                    options: {
                        env: process.env.NODE_ENV
                    }
                }
            }]
        }
    });

    compiler.outputFileSystem = new memoryfs();

    return new Promise((resolve, reject) => {
        compiler.run((err, stats) => {
            if (err || stats.hasErrors()) reject(err);

            resolve(stats);
        });
    });
};
Copy the code

Finally, we write the test and add NPM script to run it.

import compiler from './compiler.js'; test('envLoader to import', async () => { const stats = await compiler('example.js'); const output = stats.toJson().modules[0].source; if(process.env.NODE_ENV == 'prd'){ expect(output).toBe('import "/Users/yuan/Documents/yuanyuan/Project/env-loader/src/vendor/lodash.min.js"'); }else{ expect(output).toBe('function envLoad(){}'); }});Copy the code

package.json

{
  "scripts": {
    "test-boot": "cross-env NODE_ENV=boot jest",
    "test-prd": "cross-env NODE_ENV=prd jest"
  }
}
Copy the code

Run the two scripts separately

The verification succeeds to the test passes

Env-loader address details implementation process point here!!

Recruitment is stuck

Bytedance is hiring!

Position Description: Front-end Development (senior) ToB Direction — Video Cloud (Base: Shanghai, Beijing)

1. Responsible for the productization of multimedia services such as voD/live broadcast/real-time communication and the construction of business cloud platform;

2. Responsible for the construction and system development of multimedia quality system, operation and maintenance system;

3, good at abstract design, engineering thinking, focus on interaction, to create the ultimate user experience.

Job requirements

1. Major in Computer, communication and electronic information science is preferred;

2, familiar with all kinds of front end technology, including HTML/CSS/JavaScript/Node. Js, etc.

3. In-depth knowledge of JavaScript language and use of React or vue.js and other mainstream development frameworks;

4. Familiar with Node.js, Express/KOA and other frameworks, experience in developing large server programs is preferred;

5. Have some understanding of user experience, interactive operation and user demand analysis, experience in product or interface design is preferred;

6, own technical products, open source works or active open source community contributors are preferred.

Position window

Relying on the audio and video technology accumulation and basic resources of douyin, Watermelon video and other products, the video Cloud team provides customers with the ultimate one-stop audio and video multimedia services, including audio and video on demand, live broadcasting, real-time communication, picture processing, etc. Internally as the video technology center, service internal business; External to create productized audio and video multimedia service solutions, serving enterprise users.

The team has standardized project iteration process and perfect project role configuration; Strong technical atmosphere, embrace open source community, regular sharing, so that everyone can grow with the rapid business, with technology to change the world!

The delivery way

You can send your resume directly to: yuanyuan.wallace@bytedance.com

You can also scan the push two-dimensional code online delivery, looking forward to your participation! ~