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:
- Add js loader to process index.js
- Parse the envLoader function
- 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 used
return
orThis 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 back
this.async
To 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.env
NODE_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 loaderoptions
Options.
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! ~