if (document.implementation) {
document.implementation.hasFeature('HTML', '1.0');
// => DOM HTML
}
<table>
<tr>
<td align="center" colspan="2">element node</td>
</tr>
<tr>
<td>text node</td>
<td>attribute node</td>
</tr>
</table>
document.createElement('nodeName');
document.createTextNode('String');
cloneNode();
node.remove();
parentElement.appendChild(childElement);
parentElement.insertBefore(newElement, targetElement);
parentElement.removeChild();
parentElement.replaceChild();
parentElement.replaceChildren();
parentElement.hasChildNode();
setAttribute();
getAttribute();
document.getElementById();
document.getElementsByTagName();
document.querySelector();
document.querySelectorAll();
const showAlert = (type, message, duration = 3) => {
const div = document.createElement('div');
div.className = type;
div.appendChild(document.createTextNode(message));
container.insertBefore(div, form);
setTimeout(() => div.remove(), duration * 1000);
};
| Method | Node | HTML | Text | IE | Event Listeners | Secure | | ------------------ | ---- | ---- | ---- | --- | --------------- | ------- | | append | Yes | No | Yes | No | Preserves | Yes | | appendChild | Yes | No | No | Yes | Preserves | Yes | | innerHTML | No | Yes | Yes | Yes | Loses | Careful | | insertAdjacentHTML | No | Yes | Yes | Yes | Preserves | Careful |
const testDiv = document.getElementById('testDiv');
const para = document.createElement('p');
testDiv.appendChild(para);
const txt = document.createTextNode('Hello World');
para.appendChild(txt);
// 4 positions
//
// <!-- beforebegin -->
// <p>
// <!-- afterbegin -->
// foo
// <!-- beforeend -->
// </p>
// <!-- afterend -->
const p = document.querySelector('p');
p.insertAdjacentHTML('beforebegin', '<a></a>');
p.insertAdjacentText('afterbegin', 'foo');
// simply be moved element, not copied element
p.insertAdjacentElement('beforebegin', link);
function insertAfter(newElement, targetElement) {
const parent = targetElement.parentNode;
if (parent.lastChild === targetElement) {
parent.appendChild(newElement);
} else {
parent.insertBefore(newElement, targetElement.nextSibling);
}
}
node.replaceChild(document.createTextNode(text), node.firstChild);
node.replaceChildren(...nodeList);
Node 除包括元素结点 (tag) 外, 包括许多其它结点 (甚至空格符视作一个结点), 需借助 nodeType 找出目标结点.
| nodeType | representation | | :------- | :------------- | | 1 | 元素结点 | | 2 | 属性结点 | | 3 | 文本结点 |
const type = node.nodeType;
const name = node.nodeName;
const value = node.nodeValue;
const parent = node.parentNode;
const children = node.childNodes;
const first = node.firstChild;
const last = node.lastChild;
const previous = node.previousSibling;
const next = node.nextSibling;
const textContent = node.textContent;
// Returns closest ancestor of current element matching selectors
node.closest(selectors);
Element-only navigation:
Navigation properties listed above refer to all nodes.
For instance, in childNodes
we can see both
text nodes, element nodes, and even comment nodes if there exist.
const parent = node.parentElement;
const children = node.children;
const first = node.firstElementChild;
const last = node.lastElementChild;
const previous = node.previousElementSibling;
const next = node.nextElementSibling;
减少 DOM 操作次数, 减少页面渲染次数:
const frag = document.createDocumentFragment();
let p;
let t;
p = document.createElement('p');
t = document.createTextNode('first paragraph');
p.appendChild(t);
frag.appendChild(p);
p = document.createElement('p');
t = document.createTextNode('second paragraph');
p.appendChild(t);
frag.appendChild(p);
// 只渲染一次HTML页面
document.body.appendChild(frag);
const oldNode = document.getElementById('result');
const clone = oldNode.cloneNode(true);
// work with the clone
// when you're done:
oldNode.parentNode.replaceChild(clone, oldNode);
const innerHTML = element.innerHTML;
const textContent = element.textContent;
innerHTML
: non-concrete, including all types of childNodes.
div.innerHTML = <p>Test<em>test</em>Test.</p>
<div>
<p>Test<em>test</em>Test.</p>
</div>
const body = document.body;
const images = documents.images;
const links = documents.links;
const forms = documents.forms;
const formElements = documents.forms[0].elements; // 第一个表单内的所有字段
element.alt = string;
element.classname = value;
document.querySelector('cssSelector');
document.querySelectorAll('cssSelector');
The CSS Object Model is a set of APIs allowing the manipulation of CSS from JavaScript. It is much like the DOM, but for the CSS rather than the HTML. It allows users to read and modify CSS style dynamically.
const style = element.style.XX;
const font = element.style.fontFamily;
const mt = element.style.marginTopWidth;
getPropertyPriority
: return ''
or important
const box = document.querySelector('.box');
box.style.setProperty('color', 'orange');
box.style.setProperty('font-family', 'Georgia, serif');
op.innerHTML = box.style.getPropertyValue('color');
op2.innerHTML = `${box.style.item(0)}, ${box.style.item(1)}`;
box.style.setProperty('font-size', '1.5em');
box.style.item(0); // "font-size"
document.body.style.removeProperty('font-size');
document.body.style.item(0); // ""
getPropertyValue
can get css variables tooconst background = window.getComputedStyle(document.body).background;
// dot notation, same as above
const backgroundColor = window.getComputedStyle(el).backgroundColor;
// square bracket notation
const backgroundColor = window.getComputedStyle(el)['background-color'];
// using getPropertyValue()
// can get css variables property too
window.getComputedStyle(el).getPropertyValue('background-color');
element.classList.add('class');
element.classList.remove('class');
element.classList.toggle('class');
Tip: bind class
function addClass(element, value) {
if (!element.className) {
element.className = value;
} else {
newClassName = element.className;
newClassName += ' ';
newClassName += value;
element.className = newClassName;
}
}
cssRules
:
STYLE_RULE (1), IMPORT_RULE (3), MEDIA_RULE (4), KEYFRAMES_RULE (7)selectorText
property of rulesstyle
property of rulesconst myRules = document.styleSheets[0].cssRules;
const p = document.querySelector('p');
for (i of myRules) {
if (i.type === 1) {
p.innerHTML += `<code>${i.selectorText}</code><br>`;
}
if (i.selectorText === 'a:hover') {
i.selectorText = 'a:hover, a:active';
}
const myStyle = i.style;
// Set the bg color on the body
myStyle.setProperty('background-color', 'peachPuff');
// Get the font size of the body
myStyle.getPropertyValue('font-size');
// Get the 5th item in the body's style rule
myStyle.item(5);
// Log the current length of the body style rule (8)
console.log(myStyle.length);
// Remove the line height
myStyle.removeProperty('line-height');
// log the length again (7)
console.log(myStyle.length);
// Check priority of font-family (empty string)
myStyle.getPropertyPriority('font-family');
}
conditionText
property of media rulecssRules
const myRules = document.styleSheets[0].cssRules;
const p = document.querySelector('.output');
for (i of myRules) {
if (i.type === 4) {
p.innerHTML += `<code>${i.conditionText}</code><br>`;
for (j of i.cssRules) {
p.innerHTML += `<code>${j.selectorText}</code><br>`;
}
}
}
name
property of keyframe rulecssRules
:keyText
property of rulesconst myRules = document.styleSheets[0].cssRules;
const p = document.querySelector('.output');
for (i of myRules) {
if (i.type === 7) {
p.innerHTML += `<code>${i.name}</code><br>`;
for (j of i.cssRules) {
p.innerHTML += `<code>${j.keyText}</code><br>`;
}
}
}
const myStylesheet = document.styleSheets[0];
console.log(myStylesheet.cssRules.length); // 8
document.styleSheets[0].insertRule(
'article { line-height: 1.5; font-size: 1.5em; }',
myStylesheet.cssRules.length
);
console.log(document.styleSheets[0].cssRules.length); // 9
const myStylesheet = document.styleSheets[0];
console.log(myStylesheet.cssRules.length); // 8
myStylesheet.deleteRule(3);
console.log(myStylesheet.cssRules.length); // 7
event.preventDefault()
.event.stopPropagation()
.element.dispatchEvent(event)
to trigger events.bubble
mode, can change to capture
mode.function handleEvent(event) {
node.matches(event.target); // return false or true
node.contains(event.target); // return false or true
}
DOMContentLoaded:
defer
脚本执行完成之后, DOMContentLoaded 事件触发async
脚本影响,
不需要等待 async 脚本执行与样式表加载,
HTML 解析完毕后, DOMContentLoaded 立即触发document.addEventListener('DOMContentLoaded', event => {
console.log('DOM fully loaded and parsed.');
});
window.addEventListener('visibilitychange', () => {
switch (document.visibilityState) {
case 'hidden':
console.log('Tab隐藏');
break;
case 'visible':
console.log('Tab被聚焦');
break;
default:
throw new Error('Unsupported visibility!');
}
});
const videoElement = document.getElementById('videoElement');
// AutoPlay the video if application is visible
if (document.visibilityState === 'visible') {
videoElement.play();
}
// Handle page visibility change events
function handleVisibilityChange() {
if (document.visibilityState === 'hidden') {
videoElement.pause();
} else {
videoElement.play();
}
}
document.addEventListener('visibilitychange', handleVisibilityChange, false);
// <form className='validated-form' noValidate onSubmit={onSubmit}>
const onSubmit = event => {
event.preventDefault();
const form = event.target;
const isValid = form.checkValidity(); // returns true or false
const formData = new FormData(form);
const validationMessages = Array.from(formData.keys()).reduce((acc, key) => {
acc[key] = form.elements[key].validationMessage;
return acc;
}, {});
setErrors(validationMessages);
console.log({
validationMessages,
data,
isValid,
});
if (isValid) {
// here you do what you need to do if is valid
const data = Array.from(formData.keys()).reduce((acc, key) => {
acc[key] = formData.get(key);
return acc;
}, {});
}
};
function validateForm(e) {
const form = e.target;
if (form.checkValidity()) {
// form is valid - make further checks
} else {
// form is invalid - cancel submit
e.preventDefault();
// apply invalid class
Array.from(form.elements).forEach(i => {
if (i.checkValidity()) {
// field is valid - remove class
i.parentElement.classList.remove('invalid');
} else {
// field is invalid - add class
i.parentElement.classList.add('invalid');
}
});
}
}
const input = document.querySelector('input');
input.addEventListener('select', event => {
const log = document.getElementById('log');
const selection = event.target.value.substring(
event.target.selectionStart,
event.target.selectionEnd
);
log.textContent = `You selected: ${selection}`;
});
onclick
.ondbclick
.onmousedown/move/enter/out/leave/over
.For click event, no need for X/Y to judge internal/outside state.
Use DOM API element.contains
to check is a better way.
window.addEventListener('click', event => {
if (document.getElementById('main').contains(event.target)) {
process();
}
});
event.preventDefault()
in drop zone.event.preventDefault()
in drop zone.Key point for implementing DnD widget is DataTransfer:
DataTransfer.dropEffect
and DataTransfer.effectAllowed
to define DnD UI type.DataTransfer.getData
and DataTransfer.setData
to transfer data.DataTransfer.files
and DataTransfer.items
to transfer data.const noContext = document.getElementById('noContextMenu');
noContext.addEventListener('contextmenu', e => {
e.preventDefault();
});
onkeypress/up/down
document.onkeydown = function (event) {
// eslint-disable-next-line no-caller
const e = event || window.event || arguments.callee.caller.arguments[0];
if (e && e.keyCode === 13) {
// enter 键
// coding
}
};
'Alt';
'CapsLock';
'Control';
'Fn';
'Numlock';
'Shift';
'Enter';
'Tab';
' '; // space bar
'ArrowDown';
'ArrowLeft';
'ArrowRight';
'ArrowUp';
'Home';
'End';
'PageDOwn';
'PageUp';
'Backspace';
'Delete';
'Redo';
'Undo';
const source = document.querySelector('div.source');
source.addEventListener('copy', event => {
const selection = document.getSelection();
event.clipboardData.setData(
'text/plain',
selection.toString().concat('copyright information')
);
event.preventDefault();
});
function myHandler(e) {
// get event and source element
const e = e || window.event;
const src = e.target || e.srcElement;
// 事件授权
if (src.nodeName.toLowerCase() !== 'button') {
return;
}
// actual work: update label
parts = src.innerHTML.split(': ');
parts[1] = parseInt(parts[1], 10) + 1;
src.innerHTML = `${parts[0]}: ${parts[1]}`;
// no bubble
if (typeof e.stopPropagation === 'function') {
e.stopPropagation();
}
if (typeof e.cancelBubble !== 'undefined') {
e.cancelBubble = true;
}
// prevent default action
if (typeof e.preventDefault === 'function') {
e.preventDefault();
}
if (typeof e.returnValue !== 'undefined') {
e.returnValue = false;
}
}
document.write();
const URI = document.URI;
const title = document.title;
window.location(string);
window.innerWidth(number);
window.closed(boolean);
Tip: 实现 jQuery 中 $(document).ready(function(){})
.
// initialize.
window.onload = readyFunction;
function readyFunction() {}
// add more ready function
function addLoadEvent(func) {
const oldOnLoad = window.onload;
if (typeof window.onload != 'function') {
window.onload = func;
} else {
window.onload = function () {
oldOnLoad();
func();
};
}
}
| 属性 | 描述 | | :------- | :------------------------------------------ | | hash | 设置或返回从井号 (#) 开始的 URL(锚) | | host | 设置或返回主机名和当前 URL 的端口号 | | hostname | 设置或返回当前 URL 的主机名 | | href | 设置或返回完整的 URL | | pathname | 设置或返回当前 URL 的路径部分 | | port | 设置或返回当前 URL 的端口号 | | protocol | 设置或返回当前 URL 的协议 | | search | 设置或返回从问号 (?) 开始的 URL(查询部分) |
window.addEventListener(
'hashchange',
event => {
// event.oldURL
// event.nweURL
if (window.location.hash === '#someCoolFeature') {
someCoolFeature();
}
},
false
);
getBoundingClientRect
:
const height =
window.innerHeight ||
document.documentElement.clientHeight ||
document.body.clientHeight;
:::tip Rect API In case of transforms, the offsetWidth and offsetHeight returns the layout width and height (all the same), while getBoundingClientRect() returns the rendering width and height. :::
const supportPageOffset = window.pageXOffset !== undefined;
const isCSS1Compat = (document.compatMode || '') === 'CSS1Compat';
const x = supportPageOffset
? window.pageXOffset
: isCSS1Compat
? document.documentElement.scrollLeft
: document.body.scrollLeft;
const y = supportPageOffset
? window.pageYOffset
: isCSS1Compat
? document.documentElement.scrollTop
: document.body.scrollTop;
if (window.innerHeight + window.pageYOffset === document.body.scrollHeight) {
console.log('Scrolled to Bottom!');
}
const isElementInViewport = el => {
const { top, height, left, width } = el.getBoundingClientRect();
const w = window.innerWidth || document.documentElement.clientWidth;
const h = window.innerHeight || document.documentElement.clientHeight;
return top <= h && top + height >= 0 && left <= w && left + width >= 0;
};
const re = /pattern/gim;
function codePointLength(text) {
const result = text.match(/[\s\S]/gu);
return result ? result.length : 0;
}
const s = '𠮷𠮷';
const length = s.length; // 4
codePointLength(s); // 2
| Characters | Meaning |
| :--------- | :-------------------- |
| .
| [^\n\r\u2020\u2029]
|
| \d
| [0-9]
|
| \D
| [^0-9]
|
| \w
| [0-9a-zA-Z_]
|
| \W
| [^0-9a-zA-Z_]
|
| \s
| [\r\n\f\t\v]
|
| \S
| [^\r\n\f\t\v]
|
| \b
| start/end of word |
| \B
| not start/end of word |
| ^
| start of string |
| $
| end of string |
| Quantifiers | Repeat Times |
| :---------- | :----------- |
| *
| 0+ |
| +
| 1+ |
| ?
| 0 ~ 1 |
| {n}
| n |
| {n,}
| n+ |
| {n,m}
| n ~ m |
| Lazy Quantifiers | Repeat Times (As Less As Possible) |
| :--------------- | :------------------------------------- |
| *?
| 0+ |
| +?
| 1+ |
| ??
| 0 ~ 1 |
| {n,}?
| n+ |
| {n,m}?
| n ~ m |
位置编号 - 左括号的顺序:
\1 \2 \3
: 第 n 个子表达式匹配的结果字符.$1 $2 $3
: 第 n 个子表达式匹配的结果字符.const regExp = /((<\/?\w+>.*)\2)/g;
const text = 'ooo111ooo222ooo333ooo123';
const regExp = /(\d)\1\1/g;
const result = text.match(regExp);
console.log(result); // [111, 222, 333]
:::danger RegExp Static Property
Most RegExp.XXX
/RegExp.$X
static property aren't standard.
Avoid use them in production:
RegExp.input ($_)
.RegExp.lastMatch ($&)
.RegExp.lastParen ($+)
.RegExp.leftContext
.RegExp.rightContext ($')
.RegExp.$1-$9
.:::
| 分类 | 代码/语法 | 说明 |
| :------- | :------------- | :---------------------------------------------- |
| 捕获 | (exp)
| 匹配 exp,并捕获文本到自动命名的组里 |
| | (?<name>exp)
| 匹配 exp,并捕获文本到名称为 name 的组里 |
| | (?:exp)
| 匹配 exp,不捕获匹配的文本, 也不给此分组分配组号 |
| 零宽断言 | (?<=exp)
| 匹配左侧是 exp 的位置 |
| | (?<!exp)
| 匹配左侧不是 exp 的位置 |
| | (?=exp)
| 匹配右侧是 exp 的位置 |
| | (?!exp)
| 匹配右侧不是 exp 的位置 |
| 注释 | (?#comment)
| 用于提供注释让人阅读 |
(?<=\d)th
-> 9th
.(?<!\d)th
-> health
.six(?=\d)
-> six6
.hi(?!\d)
-> high
.const string = 'Favorite GitHub Repos: tc39/ecma262 v8/v8.dev';
const regex = /\b(?<owner>[a-z0-9]+)\/(?<repo>[a-z0-9\.]+)\b/g;
for (const match of string.matchAll(regex)) {
console.log(`${match[0]} at ${match.index} with '${match.input}'`);
console.log(`owner: ${match.groups.owner}`);
console.log(`repo: ${match.groups.repo}`);
}
^/$ x \u363A [a-z] \b
, 避免以分组表达式开始:
e.g \s\s*
优于 \s{1,}
.(?:...)
优于 (...)
.split
.match
.search
.replace
.test
.exec
./[a-z|A-Z|0-9]/gim.test(str);
const ignoreList = [
// # All
'^npm-debug\\.log$', // Error log for npm
'^\\..*\\.swp$', // Swap file for vim state
// # macOS
'^\\.DS_Store$', // Stores custom folder attributes
'^\\.AppleDouble$', // Stores additional file resources
'^\\.LSOverride$', // Contains the absolute path to the app to be used
'^Icon\\r$', // Custom Finder icon: http://superuser.com/questions/298785/icon-file-on-os-x-desktop
'^\\._.*', // Thumbnail
'^\\.Spotlight-V100(?:$|\\/)', // Directory that might appear on external disk
'\\.Trashes', // File that might appear on external disk
'^__MACOSX$', // Resource fork
// # Linux
'~$', // Backup file
// # Windows
'^Thumbs\\.db$', // Image file cache
'^ehthumbs\\.db$', // Folder config file
'^Desktop\\.ini$', // Stores custom folder attributes
'@eaDir$', // "hidden" folder where the server stores thumbnails
];
export const junkRegex = new RegExp(ignoreList.join('|'));
export function isJunk(filename) {
return junkRegex.test(filename);
}
replace(regExp, str / func);
第二个参数若为函数式参数,replace 方法会向它传递一系列参数:
if (!String.prototype.trim) {
// eslint-disable-next-line no-extend-native
String.prototype.trim = function () {
return this.replace(/^\s+/, '').replace(/\s+$/, '');
};
}
if (!String.prototype.trim) {
// eslint-disable-next-line no-extend-native
String.prototype.trim = function () {
const str = this.replace(/^\s+/, '');
let end = str.length - 1;
const ws = /\s/;
while (ws.test(str.charAt(end))) {
end--;
}
return str.slice(0, end + 1);
};
}
/^[\u4e00-\u9fa5a-zA-Z]+$/i
/^[1-9]*$/i
/[(^\s+)(\s+$)]/g
undefined
)window
)this
object(refer to window
)undefined
)/function, store them into memorythis
objectundefined
)/function, store them into memory如果 JavaScript 引擎在函数执行上下文中找不到变量, 它会在最近的父级执行上下文中查找该变量. 这个查找链将会一直持续, 直到引擎查找到全局执行上下文. 这种情况下, 如果全局执行上下文也没有该变量, 那么将会抛出引用错误 (Reference Error). 子函数“包含”它父级函数的变量环境,把这个概念称为闭包(Closure), 即使父级函数执行环境已经从执行栈弹出了, 子函数还是可以访问父级函数变量 x (通过作用域链).
The job of the event loop is to look into the call stack and determine if the call stack is empty or not. If the call stack is empty, it looks into the ES6 job queue and message queue to see if there’s any pending call back waiting to be executed:
Promises
(higher priority)setTimeout
, DOM events
process.nextTick
.Promises.then
(Promise 构造函数是同步函数).Object.observer
, MutationObserver
.catch finally
.scripts
: 整体脚本视作一个宏任务.MessageChannel
, postMessage
.setImmediate
, I/O
.setTimeout
, setInterval
.XHR
callback function.requestAnimationFrame
.events
callback function.Event Loop
与 Call Stack
有且仅有一个, Task/Job/Message Queue
可以有多个.:::tip Event Loop 宏任务队列取宏任务 -> 执行 1 个宏任务 -> 检查微任务队列并执行所有微任务 -> requestAnimationFrame -> 浏览器重排/重绘 -> requestIdleCallback -> 宏任务队列取宏任务 :::
Event Loop
simple model:
for (macroTask of macroTaskQueue) {
// 1. Handle current MacroTask.
handleMacroTask(macroTask);
// 2. Handle all MicroTasks.
for (microTask of microTaskQueue) {
handleMicroTask(microTask);
}
}
Using setTimeout
with 0
seconds timer
helps to defer execution of Promise
and bar
until the stack is empty.
const bar = () => {
console.log('bar');
};
const baz = () => {
console.log('baz');
};
const foo = () => {
console.log('foo');
setTimeout(bar, 0);
new Promise((resolve, reject) => {
resolve('Promise resolved');
})
.then(res => console.log(res))
.catch(err => console.log(err));
baz();
};
foo();
// foo
// baz
// Promised resolved
// bar
process.nextTick()
run before Promise.then()
:
console.log('1');
setTimeout(() => {
console.log(2);
Promise.resolve().then(() => {
console.log(3);
process.nextTick(function foo() {
console.log(4);
});
});
});
Promise.resolve().then(() => {
console.log(5);
setTimeout(() => {
console.log(6);
});
Promise.resolve().then(() => {
console.log(7);
});
});
process.nextTick(function foo() {
console.log(8);
process.nextTick(function foo() {
console.log(9);
});
});
console.log('10');
// 1 10 8 9 5 7 2 3 4 6
Promise 构造函数本身是同步函数:
console.log('script start');
const promise1 = new Promise(function (resolve) {
console.log('promise1');
resolve();
console.log('promise1 end');
}).then(function () {
console.log('promise2');
});
setTimeout(function () {
console.log('setTimeout');
});
console.log('script end');
// 输出顺序:
// script start
// promise1
// promise1 end
// script end
// promise2
// setTimeout.
await a(); b()
等价于 Promise(a()).then(b())
: a 是同步执行, b 是 MicroTask:
async function async1() {
console.log('async1 start');
await async2();
console.log('async1 end');
}
async function async2() {
console.log('async2');
}
console.log('script start');
setTimeout(function () {
console.log('setTimeout');
}, 0);
async1();
new Promise(function (resolve) {
console.log('promise1');
resolve();
}).then(function () {
console.log('promise2');
});
console.log('script end');
// script start
// async1 start
// async2
// promise1
// script end
// async1 end
// promise2
// setTimeout
当调用栈没有同步函数时, 清空 MicroTask 任务队列里的函数, 再从 MacroTask 任务队列里取出一个函数执行 (第二次 Event Loop):
function test() {
console.log('start');
setTimeout(() => {
console.log('children2');
Promise.resolve().then(() => {
console.log('children2-1');
});
}, 0);
setTimeout(() => {
console.log('children3');
Promise.resolve().then(() => {
console.log('children3-1');
});
}, 0);
Promise.resolve().then(() => {
console.log('children1');
});
console.log('end');
}
test();
// start
// end
// children1
// children2
// children2-1
// children3
// children3-1
Node.js can run I/O operations in a non-blocking way, meaning other code (and even other I/O operations) can be executed while an I/O operation is in progress.
Instead of having to ‘wait’ for an I/O operation to complete (and essentially waste CPU cycles sitting idle), Node.js can use the time to execute other tasks.
When the I/O operation completes, event loop give back control to the piece of code that is waiting for the result of that I/O operation.
The Node.js execution model was designed to cater to the needs of most web servers, which tend to be I/O-intensive (due to non-blocking I/O).
console.log('glob1');
setTimeout(function () {
console.log('timeout1');
process.nextTick(function () {
console.log('timeout1_nextTick');
});
new Promise(function (resolve) {
console.log('timeout1_promise');
resolve();
}).then(function () {
console.log('timeout1_then');
});
});
setImmediate(function () {
console.log('immediate1');
process.nextTick(function () {
console.log('immediate1_nextTick');
});
new Promise(function (resolve) {
console.log('immediate1_promise');
resolve();
}).then(function () {
console.log('immediate1_then');
});
});
process.nextTick(function () {
console.log('glob1_nextTick');
});
new Promise(function (resolve) {
console.log('glob1_promise');
resolve();
}).then(function () {
console.log('glob1_then');
});
setTimeout(function () {
console.log('timeout2');
process.nextTick(function () {
console.log('timeout2_nextTick');
});
new Promise(function (resolve) {
console.log('timeout2_promise');
resolve();
}).then(function () {
console.log('timeout2_then');
});
});
process.nextTick(function () {
console.log('glob2_nextTick');
});
new Promise(function (resolve) {
console.log('glob2_promise');
resolve();
}).then(function () {
console.log('glob2_then');
});
setImmediate(function () {
console.log('immediate2');
process.nextTick(function () {
console.log('immediate2_nextTick');
});
new Promise(function (resolve) {
console.log('immediate2_promise');
resolve();
}).then(function () {
console.log('immediate2_then');
});
});
console.log('glob2');
// glob1
// glob1_promise
// glob2_promise
// glob2
// glob1_nextTick
// glob2_nextTick
// glob1_then
// glob2_then
// timeout1
// timeout1_promise
// timeout1_nextTick
// timeout1_then
// timeout2
// timeout2_promise
// timeout2_nextTick
// timeout2_then
// immediate1
// immediate1_promise
// immediate1_nextTick
// immediate1_then
// immediate2
// immediate2_promise
// immediate2_nextTick
// immediate2_then
// o1 and o2 have the same shape
// JSObject(1, 2) => Shape('x', 'y')
// JSObject(3, 4) => Shape('x', 'y')
// 'x' => 0 Offset, Writable, Enumerable, Configurable
// 'y' => 1 Offset, Writable, Enumerable, Configurable
const o1 = { x: 1, y: 2 };
const o2 = { x: 3, y: 4 };
Shape Transform
// Shape chain: Shape(empty) => Shape(x) => Shape(x, y)
const o = {};
o.x = 1;
o.y = 2;
// Shape chain: Shape(empty) => Shape(y) => Shape(y, x)
const o = {};
o.y = 2;
o.x = 1;
// Shape chain: Shape(x)
const o = { x: 1 };
array shape: Shape('length'), 'length' => 0 Offset, Writable
V8 use ICs to memorize information (same shape) where to find properties on objects:
V8 分代垃圾回收算法, 将堆分为两个空间:
Scavenge
回收算法, 副垃圾回收器.Mark-Sweep-Compact
回收算法, 主垃圾回收器.垃圾回收优先于代码执行, 会先停止代码的执行, 等到垃圾回收完毕, 再执行 JS 代码, 成为全停顿.
Orinoco 优化 (优化全停顿现象):
JS + Mark + JS + Mark ...
.RenderNG pipeline (Main Thread + Compositor Thread + Viz Process):
DTD is context-sensitive grammar. Use State Machine pattern to implement a tokenizer:
:::tip Tokenizer Data -> Tag Open -> Tag Name -> Tag Close -> Data. :::
tokenizer send tokens to constructor, constructing DOM tree:
:::tip DOM Tree Constructor initial -> before HTML -> before head -> in head -> after head -> in body -> after body -> after after body -> EOF token. :::
HTML parser performance:
<= 1500
DOM nodes.<= 60
children nodes.<= 32
levels.CSS is context-free grammar. Webkit use flex/bison (bottom-to-up), Gecko use up-to-bottom.
ruleSet
: selector [ ',' S* selector ]*
'{' S* declaration [ ';' S* declaration ]* '}' S*
;
selector
: simple_selector [ combinator selector | S+ [ combinator? selector ]? ]?
;
simple_selector
: element_name [ HASH | class | attrib | pseudo ]*
| [ HASH | class | attrib | pseudo ]+
;
class
: '.' IDENT
;
element_name
: IDENT | '*'
;
attrib
: '[' S* IDENT S* [ [ '=' | INCLUDES | DASHMATCH ] S*
[ IDENT | STRING ] S* ] ']'
;
pseudo
: ':' [ IDENT | FUNCTION S* [IDENT S*] ')' ]
;
Render blocking resources are files that 'press pause' on the critical rendering path. They interrupt one or more of the steps:
defer
, async
, or module
attribute on scripts.为避免对所有细小更改都进行整体布局,浏览器采用了一种“dirty 位”系统。 如果某个呈现器发生了更改,或者将自身及其子代标注为“dirty”,则需要进行布局:
Paint Order:
setTimeout
/setInterval
:
clear with clearTimeout
/clearInterval
.removeEventListener
.unsubscribe(id)
.babel
/tsc
.Set
/Map
:
WeakSet
/WeakMap
don't bother GC.delete
操作符并不会释放内存,
而且会使得附加到对象上的 hidden class
(V8
为了优化属性访问时间而创建的隐藏类)失效,
让对象变成 slow object
.eval()
.with () {}
.new Function()
.const DOM = tazimi.util.Dom;
DOM.method.call(/* 关注 this 指针 */);
由于作用域链的关系,标识符解析时,寻找局部变量速度远快于寻找全局变量速度. 故应将全局变量作为参数传入函数进行调用,不但效率高,而且易于维护与测试. 即利用局部变量引用全局变量,加快标识符解析.
倒序循环可提升性能:
for (let i = item.length; i--; ) {
process(items[i]);
}
let j = items.length;
while (j--) {
process(items[i]);
}
let k = items.length;
do {
process(items[k]);
} while (k--);
Duff's Device:
let i = items.length % 8;
while (i) {
process(items[i--]);
}
i = Math.floor(items.length / 8);
while (i) {
process(items[i--]);
process(items[i--]);
process(items[i--]);
process(items[i--]);
process(items[i--]);
process(items[i--]);
process(items[i--]);
process(items[i--]);
}
function MyError(...args) {
Error.call(this, args);
this.message = args[0];
}
MyError.prototype = new Error('Error');
MyError.prototype.constructor = MyError;
const err = {
name: 'XXError',
message: 'something wrong',
extra: 'This was rather embarrassing',
remedy: genericErrorHandler, // 处理错误的函数名.
};
try {
throwError();
} catch (e) {
console.log(e.message);
e.remedy(); // genericErrorHandler.
}
调用栈尺寸限制异常,应立即定位在代码中的递归实例上
try {
recursion();
} catch (ex) {
console.error('error info');
}
try catch
.Promise.catch
.window.addEventListener('error', handler, true)
.window.addEventListener('unhandledrejection', handler, true)
.process.on('uncaughtException', handleError)
.process.on('SIGHUP', handleExit)
.process.on('SIGINT', handleExit)
.process.on('SIGQUIT', handleExit)
.process.on('SIGTERM', handleExit)
.const object = ['foo', 'bar'];
try {
for (let i = 0; i < object.length; i++) {
// do something that throws an exception
}
} catch (e) {
// handle exception
}
window.onload = function () {
const oUl = document.getElementById('ul');
const aLi = oUl.getElementsByTagName('li');
oUl.onmouseover = function (e) {
const e = e || window.event;
const target = e.target || e.srcElement;
// alert(target.innerHTML);
if (target.nodeName.toLowerCase() === 'li') {
target.style.background = 'red';
}
// 阻止默认行为并取消冒泡
if (typeof e.preventDefault === 'function') {
e.preventDefault();
e.stopPropagation();
} else {
e.returnValue = false;
e.cancelBubble = true;
}
};
oUl.onmouseout = function (e) {
const e = e || window.event;
const target = e.target || e.srcElement;
// alert(target.innerHTML);
if (target.nodeName.toLowerCase() === 'li') {
target.style.background = '';
}
// 阻止默认行为并取消冒泡
if (typeof e.preventDefault === 'function') {
e.preventDefault();
e.stopPropagation();
} else {
e.returnValue = false;
e.cancelBubble = true;
}
};
};
缓存对象属性与 DOM 对象
合并脚本后再进行高级加载技术
... The full body of the page ...
<script>
window.onload = function () {
const script = document.createElement("script");
script.src = "all_lazy_20100426.js";
document.documentElement.firstChild.appendChild(script);
};
</script>
</body>
</html>
function requireScript(file, callback) {
const script = document.getElementsByTagName('script')[0];
const newJS = document.createElement('script');
// IE
newJS.onreadystatechange = function () {
if (newJS.readyState === 'loaded' || newJS.readyState === 'complete') {
newJS.onreadystatechange = null;
callback();
}
};
// others
newJS.onload = function () {
callback();
};
// 添加至html页面
newJS.src = file;
script.parentNode.insertBefore(newJS, script);
}
requireScript('the_rest.js', function () {
Application.init();
});
const btn = document.getElementById('btn');
function toArray(coll) {
for (let i = 0, a = [], len = coll.length; i < len; i++) {
a[i] = coll[i];
}
return a;
}
childNodes[next]
获取或改变布局的操作会导致渲染树变化队列刷新, 执行渲染队列中的"待处理变化", 重排 DOM 元素
display="none"
, 修改完成后, display=""
.document.createDocumentFragment()
.const fragment = document.createDocumentFragment();
appendDataToElement(fragment, data);
document.getElementById('myList').appendChild(fragment);
const old = document.getElementById('myList');
const clone = old.cloneNode(true);
appendDataToElement(clone, data);
old.parentNode.replaceChild(clone, old);
run scripts as early as possible:
requestAnimationFrame()
runs after the CPU work is done (UI events and JS scripts),
and just before the frame is rendered (layout, paint, composite etc.).
在 js 中(除定位属性) 外,不直接操作 element.style.attr/element.cssText:
element.classList.add('className');
element.className += ' className';
:::tip Pipeline Script -> Style ->Layout -> Paint -> Composite. :::
Make script
stage become: read then write.
Interleaved read and write will trigger multiple times
of re-layout/repaint/re-composite.
:::danger Forced Synchronous Layout read css -> write css (re-layout/paint/composite) -> read css -> write css (re-layout/paint/composite) -> read css -> write css (re-layout/paint/composite). :::
:::tip High Performance read css -> write css (only re-layout/paint/composite once). :::
JavaScript 代码与 UI 共享线程.
setTimeout
/setInterval
:
nextTick
API.const button = document.getElementById('myButton');
button.onclick = function () {
oneMethod();
setTimeout(function () {
document.getElementById('notice').style.color = 'red';
}, 250);
};
/*
* usage: start -> stop -> getTime
*/
const Timer = {
_data: {},
start(key) {
Timer._data[key] = new Date();
},
stop(key) {
const time = Timer._data[key];
if (time) {
Timer._data[key] = new Date() - time;
}
},
getTime(key) {
return Timer._data[key];
},
};
function saveDocument(id) {
// 利用闭包封装待执行任务
const tasks = [openDocument, writeText, closeDocument, updateUI];
setTimeout(function sliceTask() {
// 执行下一个任务
const task = tasks.shift();
task(id);
// 检查是否还有其他任务
if (tasks.length > 0) {
// 递归调用(每次参数不同)
setTimeout(sliceTask, 25);
}
}, 25);
}
function processArray(items, process, callback) {
// 克隆原数组
const todo = items.concat();
setTimeout(function sliceTask() {
process(todo.shift());
if (todo.length > 0) {
setTimeout(sliceTask, 25);
} else {
callback(items);
}
}, 25);
}
function timedProcessArray(items, process, callback) {
// 克隆原始数组
const todo = items.concat();
setTimeout(function sliceTask() {
const start = +new Date();
// 一次批处理任务持续 0.05s
do {
process(todo.shift());
} while (todo.length < 0 && +new Date() - start < 50);
if (todo.length > 0) {
setTimeout(sliceTask, 25);
} else {
callback(items);
}
}, 25);
}
防抖动和节流本质是不一样的:
// 这个是用来获取当前时间戳的
function now() {
return +new Date();
}
/**
* 防抖函数,返回函数连续调用时,空闲时间必须大于或等于 wait,func 才会执行
*
* @param {function} func 回调函数
* @param {number} wait 表示时间窗口的间隔
* @param {boolean} immediate 设置为 true 时,是否立即调用函数
* @return {function} 返回客户调用函数
*/
function debounce(func, wait = 50, immediate = true) {
let timer, context, args;
// 延迟执行函数
const later = () =>
setTimeout(() => {
// 延迟函数执行完毕,清空缓存的定时器序号
timer = null;
// 延迟执行的情况下,函数会在延迟函数中执行
// 使用到之前缓存的参数和上下文
if (!immediate) {
func.apply(context, args);
context = args = null;
}
}, wait);
// 这里返回的函数是每次实际调用的函数
return function (...params) {
// 如果没有创建延迟执行函数(later),就创建一个
if (!timer) {
timer = later();
// 如果是立即执行,调用函数
// 否则缓存参数和调用上下文
if (immediate) {
func.apply(this, params);
} else {
// eslint-disable-next-line @typescript-eslint/no-this-alias
context = this;
args = params;
}
} else {
// 如果已有延迟执行函数(later),调用的时候清除原来的并重新设定一个
// 这样做延迟函数会重新计时
clearTimeout(timer);
timer = later();
}
};
}
// simple throttle
function throttle(action) {
let isRunning = false;
return function () {
if (isRunning) return;
isRunning = true;
window.requestAnimationFrame(() => {
action();
isRunning = false;
});
};
}
/**
* underscore 节流函数,返回函数连续调用时,func 执行频率限定为 次 / wait
*
* @param {function} func 回调函数
* @param {number} wait 表示时间窗口的间隔
* @param {object} options 如果想忽略开始函数的的调用,传入{leading: false}。
* 如果想忽略结尾函数的调用,传入{trailing: false}
* 两者不能共存,否则函数不能执行
* @return {function} 返回客户调用函数
*/
_.throttle = function (func, wait, options) {
let context, args, result;
let timeout = null;
// 之前的时间戳
let previous = 0;
// 如果 options 没传则设为空对象
if (!options) options = {};
// 定时器回调函数
const later = function () {
// 如果设置了 leading,就将 previous 设为 0
// 用于下面函数的第一个 if 判断
previous = options.leading === false ? 0 : _.now();
// 置空一是为了防止内存泄漏,二是为了下面的定时器判断
timeout = null;
result = func.apply(context, args);
if (!timeout) context = args = null;
};
return function (...original_args) {
// 获得当前时间戳
const now = _.now();
// 首次进入前者肯定为 true
// 如果需要第一次不执行函数
// 就将上次时间戳设为当前的
// 这样在接下来计算 remaining 的值时会大于0
if (!previous && options.leading === false) previous = now;
// 计算剩余时间
const remaining = wait - (now - previous);
// eslint-disable-next-line @typescript-eslint/no-this-alias
context = this;
args = original_args;
// 如果当前调用已经大于上次调用时间 + wait
// 或者用户手动调了时间
// 如果设置了 trailing,只会进入这个条件
// 如果没有设置 leading,那么第一次会进入这个条件
// 还有一点,你可能会觉得开启了定时器那么应该不会进入这个 if 条件了
// 其实还是会进入的,因为定时器的延时
// 并不是准确的时间,很可能你设置了2秒
// 但是他需要2.2秒才触发,这时候就会进入这个条件
if (remaining <= 0 || remaining > wait) {
// 如果存在定时器就清理掉否则会调用二次回调
if (timeout) {
clearTimeout(timeout);
timeout = null;
}
previous = now;
result = func.apply(context, args);
if (!timeout) context = args = null;
} else if (!timeout && options.trailing !== false) {
// 判断是否设置了定时器和 trailing
// 没有的话就开启一个定时器
// 并且不能不能同时设置 leading 和 trailing
timeout = setTimeout(later, remaining);
}
return result;
};
};
function useAnimation() {
let frameId = 0;
let ticking = false;
const handleResize = event => {
if (ticking) return;
ticking = true;
frameId = requestAnimationFrame(() => handleUpdate(event));
};
const handleUpdate = event => {
console.log('resize update');
ticking = false;
};
useEffect(() => {
window.addEventListener('resize', handleResize);
handleUpdate();
return () => {
window.removeEventListener('resize', handleResize);
cancelAnimationFrame(frameId);
};
});
}
i%2
=> i&0x1
.const OPTION_A = 1;
const OPTION_B = 2;
const OPTION_C = 4;
const OPTION_D = 8;
const OPTION_E = 16;
const options = OPTION_A | OPTION_C | OPTION_D;
从缓存位置上来说分为四种, 并且各自有优先级, 当依次查找缓存且都没有命中的时候, 才会去请求网络:
// eslint-disable-next-line no-restricted-globals
self.addEventListener('install', event => {
async function buildCache() {
const cache = await caches.open(cacheName);
return cache.addAll(['/main.css', '/main.mjs', '/offline.html']);
}
event.waitUntil(buildCache());
});
// eslint-disable-next-line no-restricted-globals
self.addEventListener('fetch', event => {
async function cachedFetch(event) {
const cache = await caches.open(cacheName);
let response = await cache.match(event.request);
if (response) return response;
response = await fetch(event.request);
cache.put(event.request, response.clone());
return response;
}
event.respondWith(cachedFetch(event));
});
浏览器缓存,也称 HTTP 缓存,
分为强缓存和协商缓存.
优先级较高的是强缓存,
在命中强缓存失败的情况下或者Cache-Control: no-cache
时,
才会走协商缓存.
强缓存是利用 HTTP 头中的 Expires 和 Cache-Control 两个字段来控制的.
强缓存中, 当请求再次发出时, 浏览器会根据其中的 expires 和 cache-control 判断目标资源是否 命中
强缓存,
若命中则直接从缓存中获取资源, 不会再与服务端发生通信.
Cache-Control 相对于 expires 更加准确,它的优先级也更高.
当 Cache-Control 与 expires 同时出现时,以 Cache-Control 为准.
expires: Wed, 12 Sep 2019 06:12:18 GMT
cache-control: max-age=31536000
协商缓存机制下,
浏览器需要向服务器去询问缓存的相关信息,
进而判断是重新发起请求、下载完整的响应,
还是从本地获取缓存的资源.
如果服务端提示缓存资源未改动 (Not Modified),
资源会被重定向到浏览器缓存,
这种情况下网络请求对应的状态码是 304
.
Last-Modified 是一个时间戳, 如果启用了协商缓存, 它会在首次请求时随着 Response Headers 返回:
Last-Modified: Fri, 27 Oct 2017 06:35:57 GMT
随后每次请求时, 会带上一个叫 If-Modified-Since 的时间戳字段, 它的值正是上一次 response 返回给它的 last-modified 值:
If-Modified-Since: Fri, 27 Oct 2017 06:35:57 GMT
服务器可能无法正确感知文件的变化 (未实际改动或改动过快), 为了解决这样的问题, Etag 作为 Last-Modified 的补充出现了. Etag 是由服务器为每个资源生成的唯一的标识字符串, 这个标识字符串可以是基于文件内容编码的, 因此 Etag 能够精准地感知文件的变化.
download -> compile -> store into on-disk cache
fetch from browser cache -> compile -> store metadata
fetch scripts and metadata from browser cache -> skip compile
< 1KB
) and inline scripts前端性能监控分为两种方式, 一种叫做合成监控 (Synthetic Monitoring, SYN), 另一种是真实用户监控 (Real User Monitoring, RUM).
在一个模拟场景里, 去提交一个需要做性能审计的页面, 通过一系列的工具、规则去运行你的页面, 提取一些性能指标, 得出一个审计报告.
常见的工具有 Google 的 Lighthouse,WebPageTest,PageSpeed 等
| 优点 | 缺点 | | :------------------------------------- | :--------------------------: | | 实现简单 | 无法还原全部真实场景 | | 能采集到丰富的数据,如硬件指标或瀑布图 | 登录等场景需要额外解决 | | 不影响真实用户的访问性能 | 单次数据不够稳定 | | 可以提供页面加载幻灯片等可视化分析途径 | 数据量较小,无法发挥更大价值 |
用户在页面访问之后就会产生各种各样的性能指标, 之后会将这些性能指标上传的我们的日志服务器上, 进行数据的提起清洗加工, 最后在监控平台上进行展示和分析的一个过程.
| 优点 | 缺点 | | :------------------------------------- | :------------------------------- | | 无需配置模拟条件,完全还原真实场景 | 影响真实用户的访问性能及流量消耗 | | 不存在登录等需要额外解决的场景 | 无法采集硬件相关指标 | | 数据样本足够庞大,可以减少统计误差 | 无法采集完整的资源加载瀑布图 | | 新年数据可与其它数据关联,产生更大价值 | 无法可视化展示加载过程 |
| 对比项 | 合成监控 | 真实用户监控 | | :------------- | :--------------------- | :------------------------- | | 实现难度及成本 | 较低 | 较高 | | 采集数据丰富度 | 丰富 | 基础 | | 数据样本量 | 较小 | 大(视业务体量) | | 适合场景 | 定性分析, 小数据量分析 | 定量分析, 业务数据深度挖掘 |
在真实用户性能数据采集时, 要关注四个方面的东西:
采集性能数据时先抹平 Navigation Timing spec 差异 优先使用 PerformanceTimeline API (在复杂场景,亦可考虑优先使用 PerformanceObserver):
First Meaningful Paint: 首次有效渲染时长, 它的一个核心的想法是渲染并不一定代表着用户看到了主要内容, Load 也不一定代表用户看到主要内容. 假设当一个网页的 DOM 结构发生剧烈的变化的时候, 就是这个网页主要内容出现的时候, 那么在这样的一个时间点上, 就是用户看到主要内容的一个时间点.
它的优点是相对校准的估算出内容渲染时间, 贴近用户感知. 但缺点是无原生 API 支持, 算法推导时 DOM 节点含权重.
Google Core Web Vitals:
不同的页面操作/页面打开方式/浏览器环境都会对我们页面加载的性能会有影响, 需要上报这些维度的数据, 以便深入性能分析:
解决上报对性能的影响问题有以下方案:
onload
事件后, 并合并多个上报请求.post
上报.visibilitychange
/pagehide
event.
unload
/beforeunload
event not precise for mobile users:
e.g switch to another app not trigger unload
event.mp4 smaller than gif (ffmpeg
)
<!-- ffmpeg -i dog.gif dog.mp4 -->
<video autoplay loop muted playsinline>
<source src="dog.mp4" type="video/mp4" />
</video>
WebP 25-35% smaller than jpg/png
<picture>
<source type="image/webp" srcset="flower.webp" />
<source type="image/jpeg" srcset="flower.jpg" />
<img src="flower.jpg" />
</picture>
responsive images: provide 3~5 different sizes reduce image transfer sizes by average of ~20%
<img
srcset="flower-small.jpg 480w, flower-large.jpg 1080w"
sizes="50vw"
src="flower-large.jpg"
/>
<link rel="preload" as="script" href="critical.js" />
<link rel="modulepreload" href="critical-module.mjs" />
<link rel="preload" as="image" href="..." />
<link rel="preload" as="font" href="..." crossorigin />
<link rel="preload" as="fetch" href="..." crossorigin />
Lazy Loading Polyfill:
<img data-src="flower.jpg" class="lazyload" />
window.addEventListener('scroll', function (event) {
Array.from(document.querySelectorAll('.lazyload')).forEach(image => {
if (image.slideIntoView(event.getBoundingClientRect())) {
image.setAttribute('src', image.dataset.src);
}
});
});
Observer Lazy Loading:
const observer = new IntersectionObserver(nodes => {
nodes.forEach(v => {
if (v.isIntersecting) {
v.target.src = v.target.dataset.src;
observer.unobserve(v.target);
}
});
});
const images = document.querySelectorAll('img.lazyload');
images.forEach(v => observer.observe(v));
Native Lazy Loading:
<img src="flower.jpg" lazyload="auto" />
<img src="flower.jpg" lazyload="on" />
<img src="flower.jpg" lazyload="off" />
async
: downloads the script during parsing the document,
but will pause the parser to execute the script.defer
: downloads the script while the document is still parsing,
but waits until the document has finished parsing before executing it
(in order).async
.defer
.<head>
,
in such script can't access DOM directly
(DOM haven't get parsed).Best practice: lazy loading scripts not execute immediately. (Chrome Coverage Devtools)
<script src="myScript.js"></script>
<script src="myScript.js" async></script>
<script src="myScript.js" defer></script>
const DetailsComponent = lazy(() => import('./details'));
const PageComponent = () => {
<Suspense fallback={<div>Loading...</div>}>
<DetailsComponent />
</Suspense>;
};
<link rel="preload" /> <link rel="prefetch" />
Why not to PreFetch and PreRender:
modules
: always false
, keep esm
for bundler (e.g webpack) tree shaking.useBuiltIns
:
entry
: 将 core-js import
替换为特性列表.usage
: 按使用引入用到的特性列表.{
"presets": [
[
"@babel/preset-env",
{
"targets": {
"esmodules": true,
"node": ">= 8",
"browsers": "> 0.25%"
},
"modules": false,
"useBuiltIns": "usage"
}
]
]
}
<script type="module" src="main.mjs"></script>
<script nomodule src="legacy.js"></script>
chrome://inspect/#devices
to start inspectingCSS Performance
):Effective JavaScript
) (DOM/React/Concurrency).splitChunks
.performance.mark('mainThread-start');
expensiveCalculation();
performance.mark('mainThread-stop');
performance.measure('mainThread', 'mainThread-start', 'mainThread-stop');
const entryHandler = list => {
for (const entry of list.getEntries()) {
if (entry.name === 'first-paint') {
observer.disconnect();
}
console.log(entry);
}
};
const observer = new PerformanceObserver(entryHandler);
observer.observe({ type: 'paint', buffered: true });
// {
// duration: 0,
// entryType: "paint",
// name: "first-paint",
// startTime: 359,
// }
document.addEventListener('DOMContentLoaded', function () {
console.log('DOM 挂载时间: ', Date.now() - timerStart);
// 性能日志上报...
});
window.addEventListener('load', function () {
console.log('所有资源加载完成时间: ', Date.now() - timerStart);
// 性能日志上报...
});
// 计算加载时间.
function getPerformanceTiming() {
const performance = window.performance;
if (!performance) {
// 当前浏览器不支持.
console.log('你的浏览器不支持 performance 接口');
return;
}
const t = performance.timing;
const times = {};
// 【重要】页面加载完成的时间.
// 【原因】几乎代表了用户等待页面可用的时间.
times.loadPage = t.loadEventEnd - t.navigationStart;
// 【重要】解析 DOM 树结构的时间.
// 【原因】DOM 树嵌套过多.
times.domReady = t.domComplete - t.responseEnd;
// 【重要】重定向的时间.
// 【原因】拒绝重定向. e.g http://example.com/ 不应写成 http://example.com.
times.redirect = t.redirectEnd - t.redirectStart;
// 【重要】DNS 查询时间.
// 【原因】DNS 预加载做了么? 页面内是不是使用了太多不同的域名导致域名查询的时间太长?
// 可使用 HTML5 Prefetch 预查询 DNS, 见: [HTML5 prefetch](http://segmentfault.com/a/1190000000633364).
times.lookupDomain = t.domainLookupEnd - t.domainLookupStart;
// 【重要】读取页面第一个字节的时间.
// 【原因】这可以理解为用户拿到你的资源占用的时间, 加异地机房了么, 加CDN 处理了么? 加带宽了么? 加 CPU 运算速度了么?
// TTFB 即 Time To First Byte 的意思.
// 维基百科: https://en.wikipedia.org/wiki/Time_To_First_Byte.
times.ttfb = t.responseStart - t.navigationStart;
// 【重要】内容加载完成的时间.
// 【原因】页面内容经过 gzip 压缩了么, 静态资源 `CSS`/`JS` 等压缩了么?
times.request = t.responseEnd - t.requestStart;
// 【重要】执行 onload 回调函数的时间.
// 【原因】是否太多不必要的操作都放到 onload 回调函数里执行了, 考虑过延迟加载/按需加载的策略么?
times.loadEvent = t.loadEventEnd - t.loadEventStart;
// DNS 缓存时间.
times.appCache = t.domainLookupStart - t.fetchStart;
// 卸载页面的时间.
times.unloadEvent = t.unloadEventEnd - t.unloadEventStart;
// TCP 建立连接完成握手的时间.
times.connect = t.connectEnd - t.connectStart;
return times;
}
const [pageNav] = performance.getEntriesByType('navigation');
// Measuring DNS lookup time.
const totalLookupTime = pageNav.domainLookupEnd - pageNav.domainLookupStart;
// Quantifying total connection time.
const connectionTime = pageNav.connectEnd - pageNav.connectStart;
let tlsTime = 0; // <-- Assume 0 to start with
// Was there TLS negotiation?
if (pageNav.secureConnectionStart > 0) {
// Awesome! Calculate it!
tlsTime = pageNav.connectEnd - pageNav.secureConnectionStart;
}
// Cache seek plus response time of the current document.
const fetchTime = pageNav.responseEnd - pageNav.fetchStart;
// Service worker time plus response time.
let workerTime = 0;
if (pageNav.workerStart > 0) {
workerTime = pageNav.responseEnd - pageNav.workerStart;
}
// Request time only (excluding redirects, DNS, and connection/TLS time).
const requestTime = pageNav.responseStart - pageNav.requestStart;
// Response time only (download).
const responseTime = pageNav.responseEnd - pageNav.responseStart;
// Request + response time.
const requestResponseTime = pageNav.responseEnd - pageNav.requestStart;
First Contentful Paint:
defer
or async
attributes to <script>
tags.const entryHandler = list => {
for (const entry of list.getEntries()) {
if (entry.name === 'first-contentful-paint') {
observer.disconnect();
}
console.log(entry);
}
};
const observer = new PerformanceObserver(entryHandler);
observer.observe({ type: 'paint', buffered: true });
// {
// duration: 0,
// entryType: "paint",
// name: "first-contentful-paint",
// startTime: 459,
// }
Largest Contentful Paint:
srcset
on <img>
or <picture>
.const entryHandler = list => {
if (observer) {
observer.disconnect();
}
for (const entry of list.getEntries()) {
console.log(entry);
}
};
const observer = new PerformanceObserver(entryHandler);
observer.observe({ type: 'largest-contentful-paint', buffered: true });
// {
// duration: 0,
// element: p,
// entryType: 'largest-contentful-paint',
// id: '',
// loadTime: 0,
// name: '',
// renderTime: 1021.299,
// size: 37932,
// startTime: 1021.299,
// url: '',
// }
Cumulative Layout Shift:
height
and width
attributes of image or video,
so that it won’t move content around it once it’s loaded.popups
or overlays
unless they appear when the user interacts with the page.transform
animations.let sessionValue = 0;
let sessionEntries = [];
const cls = {
subType: 'layout-shift',
name: 'layout-shift',
type: 'performance',
pageURL: getPageURL(),
value: 0,
};
const entryHandler = list => {
for (const entry of list.getEntries()) {
// Only count layout shifts without recent user input.
if (!entry.hadRecentInput) {
const firstSessionEntry = sessionEntries[0];
const lastSessionEntry = sessionEntries[sessionEntries.length - 1];
// If the entry occurred less than 1 second after the previous entry and
// less than 5 seconds after the first entry in the session, include the
// entry in the current session. Otherwise, start a new session.
if (
sessionValue &&
entry.startTime - lastSessionEntry.startTime < 1000 &&
entry.startTime - firstSessionEntry.startTime < 5000
) {
sessionValue += entry.value;
sessionEntries.push(formatCLSEntry(entry));
} else {
sessionValue = entry.value;
sessionEntries = [formatCLSEntry(entry)];
}
// If the current session value is larger than the current CLS value,
// update CLS and the entries contributing to it.
if (sessionValue > cls.value) {
cls.value = sessionValue;
cls.entries = sessionEntries;
cls.startTime = performance.now();
lazyReportCache(deepCopy(cls));
}
}
}
};
const observer = new PerformanceObserver(entryHandler);
observer.observe({ type: 'layout-shift', buffered: true });
// {
// duration: 0,
// entryType: "layout-shift",
// hadRecentInput: false,
// lastInputTime: 0,
// name: "",
// sources: (2) [LayoutShiftAttribution, LayoutShiftAttribution],
// startTime: 1176.199999999255,
// value: 0.000005752046026677329,
// }
web.dev
performance complete guide.performance.now()
is more precise (100 us)performance.now()
is strictly monotonic (unaffected by changes of machine time)let lastVisibilityChange = 0;
window.addEventListener('visibilitychange', () => {
lastVisibilityChange = performance.now();
});
// don’t log any metrics started before the last visibility change
// don't log any metrics if the page is hidden
// discard perf data from when the machine was not running app at full speed
function metrics() {
if (metric.start < lastVisibilityChange || document.hidden) {
return;
}
process();
}
requestAnimationFrame(() => {
requestAnimationFrame(timestamp => {
metric.finish(timestamp);
});
});
const puppeteer = require('puppeteer');
(async () => {
const browser = await puppeteer.launch();
const page = await browser.newPage();
await page.goto('https://example.com');
await page.screenshot({ path: 'example.png' });
await browser.close();
})();
// Create a new incognito browser context
const context = await browser.createIncognitoBrowserContext();
// Create a new page inside context.
const page = await context.newPage();
// ... do stuff with page ...
await page.goto('https://example.com');
// Dispose context once it's no longer needed.
await context.close();
page.$(selector)
same to querySelector
// wait for selector
await page.waitFor('.foo');
// wait for 1 second
await page.waitFor(1000);
// wait for predicate
await page.waitFor(() => !!document.querySelector('.foo'));
const puppeteer = require('puppeteer');
puppeteer.launch().then(async browser => {
const page = await browser.newPage();
const watchDog = page.waitForFunction('window.innerWidth < 100');
await page.setViewport({ width: 50, height: 50 });
await watchDog;
await browser.close();
});
const [response] = await Promise.all([
page.waitForNavigation(), // The promise resolves after navigation has finished
page.click('a.my-link'), // Clicking the link will indirectly cause a navigation
]);
const firstRequest = await page.waitForRequest('http://example.com/resource');
const finalRequest = await page.waitForRequest(
request =>
request.url() === 'http://example.com' && request.method() === 'GET'
);
return firstRequest.url();
const firstResponse = await page.waitForResponse(
'https://example.com/resource'
);
const finalResponse = await page.waitForResponse(
response =>
response.url() === 'https://example.com' && response.status() === 200
);
return finalResponse.ok();
await page.evaluate(() => window.open('https://www.example.com/'));
const newWindowTarget = await browserContext.waitForTarget(
target => target.url() === 'https://www.example.com/'
);
const [response] = await Promise.all([
page.waitForNavigation(waitOptions),
page.click(selector, clickOptions),
]);
// Using ‘page.mouse’ to trace a 100x100 square.
await page.mouse.move(0, 0);
await page.mouse.down();
await page.mouse.move(0, 100);
await page.mouse.move(100, 100);
await page.mouse.move(100, 0);
await page.mouse.move(0, 0);
await page.mouse.up();
await page.keyboard.type('Hello World!');
await page.keyboard.press('ArrowLeft');
await page.keyboard.down('Shift');
for (let i = 0; i < ' World'.length; i++)
await page.keyboard.press('ArrowLeft');
await page.keyboard.up('Shift');
await page.keyboard.press('Backspace');
// Result text will end up saying 'Hello!'
await page.tracing.start({ path: 'trace.json' });
await page.goto('https://www.google.com');
await page.tracing.stop();
page.setOfflineMode
page.setGeolocation
page.metrics
page.accessibility
page.coverage
*
扇出) ^ 2.V(G) = e - n + 2
: **<10**
.
函数复杂度 = (扇入 *
扇出) ^ 2.
引用:
**<7**
.被引用:
5 级耦合度:
O.property = 'tazimi';
O.method = function () {};
O.prototype.method = function () {};
4 级耦合度, 共享全局变量:
let Global = 'global';
function A() {
Global = 'A';
}
function B() {
Global = 'B';
}
3 级耦合度:
const absFactory = new AbstractFactory({ env: 'TEST' });
2 级耦合度:
O.prototype.makeBread = function (args) {
return new Bread(args.type, args.size);
};
O.makeBread({ type: wheat, size: 99, name: 'foo' });
1 级耦合度.
0 级耦合度.
const mockery = require('mockery');
mockery.enable();
describe('Sum suite File', function () {
beforeEach(function () {
mockery.registerAllowable('./mySumFS', true);
});
afterEach(function () {
mockery.deregisterAllowable('./mySumFS');
});
it('Adds Integers!', function () {
const filename = 'numbers';
const fsMock = {
readFileSync(path, encoding) {
expect(path).toEqual(filename);
expect(encoding).toEqual('utf8');
return JSON.stringify({ a: 9, b: 3 });
},
};
mockery.registerMock('fs', fsMock);
const mySum = require('./mySumFS');
expect(mySum.sum(filename)).toEqual(12);
mockery.deregisterMock('fs');
});
});
Inject trace function (log, monitor, report service)
to window pushState
and replaceState
.
const _wr = function (type) {
const orig = window.history[type];
return function (...args) {
const rv = orig.apply(this, args);
const e = new Event(type.toLowerCase());
e.arguments = args;
window.dispatchEvent(e);
return rv;
};
};
window.history.pushState = _wr('pushState');
window.history.replaceState = _wr('replaceState');
window.addEventListener('pushstate', function (event) {
console.trace('pushState');
});
window.addEventListener('replacestate', function (event) {
console.trace('replaceState');
});
const originalStopPropagation = MouseEvent.prototype.stopPropagation;
MouseEvent.prototype.stopPropagation = function (...args) {
console.trace('stopPropagation');
originalStopPropagation.call(this, ...args);
};
let originalScrollTop = element.scrollTop;
Object.defineProperty(element, 'scrollTop', {
get() {
return originalScrollTop;
},
set(newVal) {
console.trace('scrollTop');
originalScrollTop = newVal;
},
});
console.XXX
.copy
: copy complex object to clipboard.monitor
: monitor object.const devtools = /./;
devtools.toString = function () {
this.opened = true;
};
console.log('%c', devtools);
// devtools.opened will become true if/when the console is opened
// Basic console functions
console.assert();
console.clear();
console.log();
console.debug();
console.info();
console.warn();
console.error();
// Different output styles
console.dir();
console.dirxml();
console.table();
console.group();
console.groupCollapsed();
console.groupEnd();
// Trace console functions
console.trace();
console.count();
console.countReset();
console.time();
console.timeEnd();
console.timeLog();
// Non-standard console functions
console.profile();
console.profileEnd();
console.timeStamp();
console.log
// `sprinf` style log
console.log('%d %o %s', integer, object, string);
console.log('%c ...', 'css style');
console.table
// display array of object (tabular data)
const transactions = [
{
id: '7cb1-e041b126-f3b8',
seller: 'WAL0412',
buyer: 'WAL3023',
price: 203450,
time: 1539688433,
},
{
id: '1d4c-31f8f14b-1571',
seller: 'WAL0452',
buyer: 'WAL3023',
price: 348299,
time: 1539688433,
},
{
id: 'b12c-b3adf58f-809f',
seller: 'WAL0012',
buyer: 'WAL2025',
price: 59240,
time: 1539688433,
},
];
console.table(data, ['id', 'price']);
debugger
:
// debugger;
copy(obj); // to clipboard
window.onerror = function (errorMessage, scriptURI, lineNo, columnNo, error) {
console.log(`errorMessage: ${errorMessage}`); // 异常信息
console.log(`scriptURI: ${scriptURI}`); // 异常文件路径
console.log(`lineNo: ${lineNo}`); // 异常行号
console.log(`columnNo: ${columnNo}`); // 异常列号
console.log(`error: ${error}`); // 异常堆栈信息
// ...
// 异常上报
};
window.addEventListener('error', function () {
console.log(error);
// ...
// 异常上报
});
const traceProperty = (object, property) => {
let value = object[property];
Object.defineProperty(object, property, {
get() {
console.trace(`${property} requested`);
return value;
},
set(newValue) {
console.trace(`setting ${property} to `, newValue);
value = newValue;
},
});
};
node --inspect
ndb index.js
不使用特性/浏览器推断,往往容易推断错误(且会随着浏览器更新产生新的错误)
// 特性检测
if (document.getElementById) {
element = document.getElementById(id);
}
npm i -D jest ts-jest @types/jest react-test-renderer
jest.config.js
:
const { pathsToModuleNameMapper } = require('ts-jest/utils');
const { compilerOptions } = require('./tsconfig.json');
const paths = pathsToModuleNameMapper(compilerOptions.paths, {
prefix: '<rootDir>/',
});
/** @type {import('ts-jest/dist/types').InitialOptionsTsJest} */
module.exports = {
roots: ['<rootDir>/src'],
collectCoverage: true,
coverageDirectory: 'coverage',
transform: {
'^.+\\.jsx?$': '<rootDir>/jest.transformer.js',
'^.+\\.tsx?$': 'ts-jest',
},
transformIgnorePatterns: ['node_modules/(?!(gatsby)/)'],
moduleFileExtensions: ['ts', 'tsx', 'js', 'jsx', 'json', 'node'],
moduleNameMapper: {
'.+\\.(css|styl|less|sass|scss)$': 'identity-obj-proxy',
'.+\\.(jpg|jpeg|png|gif|eot|otf|webp|svg|ttf|woff|woff2|mp4|webm|wav|mp3|m4a|aac|oga)$':
'<rootDir>/__mocks__/jest.mock.js',
...paths,
'^@components/(.*)$': '<rootDir>/src/components/$1',
'^@hooks/(.*)$': '<rootDir>/src/hooks/$1',
'^@layouts/(.*)$': '<rootDir>/src/layouts/$1',
'^@types/(.*)$': '<rootDir>/src/types/$1',
},
testPathIgnorePatterns: ['node_modules', '\\.cache', '<rootDir>.*/build'],
testRegex: '(/__tests__/.*|(\\.|/)(test|spec))\\.(jsx?|tsx?)$',
globals: {
window: {},
'ts-jest': {
tsConfig: './tsconfig.json',
},
},
testURL: 'http://localhost',
testEnvironment: 'jsdom',
setupFiles: ['<rootDir>/jest.env.setup.js'],
setupFilesAfterEnv: ['<rootDir>/jest.setup.js'],
setupTestFrameworkScriptFile: '<rootDir>/src/setupEnzyme.ts',
};
jest.env.setup.js
:
import path from 'path';
import dotenv from 'dotenv';
console.log(`============ env-setup Loaded ===========`);
dotenv.config({
path: path.resolve(process.cwd(), 'tests', 'settings', '.test.env'),
});
jest.setup.js
:
import '@testing-library/jest-dom/extend-expect';
// Global/Window object Stubs for Jest
window.matchMedia =
window.matchMedia ||
function () {
return {
matches: false,
addListener() {},
removeListener() {},
};
};
Object.defineProperty(window, 'matchMedia', {
writable: true,
value: jest.fn().mockImplementation(query => ({
matches: false,
media: query,
onchange: null,
addListener: jest.fn(), // deprecated
removeListener: jest.fn(), // deprecated
addEventListener: jest.fn(),
removeEventListener: jest.fn(),
dispatchEvent: jest.fn(),
})),
});
window.requestAnimationFrame = function (callback) {
setTimeout(callback);
};
window.cancelAnimationFrame = window.clearTimeout;
window.localStorage = {
getItem() {},
setItem() {},
};
Object.values = () => [];
setupEnzyme.ts
:
import { configure } from 'enzyme';
import * as EnzymeAdapter from 'enzyme-adapter-react-16';
configure({ adapter: new EnzymeAdapter() });
describe
block.test
statement.it
statement.test.todo
:
import * as React from 'react';
import { shallow } from 'enzyme';
import { Checkbox } from './Checkbox';
describe('Checkbox should', () => {
test('changes the text after click', () => {
const checkbox = shallow(<Checkbox labelOn="On" labelOff="Off" />);
expect(checkbox.text()).toEqual('Off');
checkbox.find('input').simulate('change');
expect(checkbox.text()).toEqual('On');
});
});
jest -u
to overwrite existing snapshot.// Link.react.test.js
import React from 'react';
import renderer from 'react-test-renderer';
import Link from '@components/Link';
describe('Link should', () => {
test('changes the class when hovered', () => {
const component = renderer.create(
<Link page="http://www.facebook.com">Facebook</Link>
);
let tree = component.toJSON();
expect(tree).toMatchSnapshot();
// manually trigger the callback
tree.props.onMouseEnter();
// re-rendering
tree = component.toJSON();
expect(tree).toMatchSnapshot();
// manually trigger the callback
tree.props.onMouseLeave();
// re-rendering
tree = component.toJSON();
expect(tree).toMatchSnapshot();
});
});
Jest async guide:
await expect(asyncCall()).resolves.toEqual('Expected');
await expect(asyncCall()).rejects.toThrowError();
__mocks__
:
jest.createMockFromModule('moduleName')
.jest.requireActual('moduleName')
.// react-dom.js
import React from 'react';
const reactDom = jest.requireActual('react-dom');
function mockCreatePortal(element, target) {
return (
<div>
<div id="content">{element}</div>
<div id="target" data-target-tag-name={target.tagName}></div>
</div>
);
}
reactDom.createPortal = mockCreatePortal;
module.exports = reactDom;
// gatsby.js
import React from 'react';
const gatsby = jest.requireActual('gatsby');
module.exports = {
...gatsby,
graphql: jest.fn(),
Link: jest
.fn()
.mockImplementation(
({
activeClassName,
activeStyle,
getProps,
innerRef,
partiallyActive,
ref,
replace,
to,
...rest
}) =>
React.createElement('a', {
...rest,
href: to,
})
),
StaticQuery: jest.fn(),
useStaticQuery: jest.fn(),
};
A simple test runner implementation:
import { promises as fs } from 'fs';
import { basename, dirname, join } from 'path';
import { pathToFileURL } from 'url';
async function* walk(dir: string): AsyncGenerator<string> {
for await (const d of await fs.opendir(dir)) {
const entry = join(dir, d.name);
if (d.isDirectory()) {
yield* walk(entry);
} else if (d.isFile()) {
yield entry;
}
}
}
async function runTestFile(file: string): Promise<void> {
for (const value of Object.values(
await import(pathToFileURL(file).toString())
)) {
if (typeof value === 'function') {
try {
await value();
} catch (e) {
console.error(e instanceof Error ? e.stack : e);
process.exit(1);
}
}
}
}
async function run(arg = '.') {
if ((await fs.lstat(arg)).isFile()) {
return runTestFile(arg);
}
for await (const file of walk(arg)) {
if (
!dirname(file).includes('node_modules') &&
(basename(file) === 'test.js' || file.endsWith('.test.js'))
) {
console.log(file);
await runTestFile(file);
}
}
}
run(process.argv[2]);
When it comes to test heavy visual features, (e.g fixed navigation based on window scroll event), E2E testing helps a lot.
mkdir e2e
cd e2e
npm init -y
npm install cypress webpack @cypress/webpack-preprocessor typescript ts-loader
npx cypress open
cypress open
will initialize the cypress folder structure for us.
e2e/plugins/index.js
: setup TypeScript to transpile tests:
const wp = require('@cypress/webpack-preprocessor');
module.exports = on => {
const options = {
webpackOptions: {
resolve: {
extensions: ['.ts', '.tsx', '.js', '.jsx'],
},
module: {
rules: [
{
test: /\.tsx?$/,
loader: 'ts-loader',
options: { transpileOnly: true },
},
],
},
},
};
on('file:preprocessor', wp(options));
};
e2e/tsconfig.json
:
{
"compilerOptions": {
"strict": true,
"sourceMap": true,
"module": "commonjs",
"target": "es5",
"lib": ["DOM", "ES6"],
"jsx": "react",
"experimentalDecorators": true
},
"compileOnSave": false
}
e2e/package.json
:
{
"scripts": {
"cypress:open": "cypress open",
"cypress:run": "cypress run"
}
}
support/commands.ts
:
import '@testing-library/cypress/add-commands';
integration/component.spec.ts
:
/// <reference types="cypress"/>
describe('component', () => {
it('should work', () => {
cy.visit('http://localhost:8000');
cy.get('#onOff')
.should('have.text', 'off')
.click()
.should('have.text', 'on');
});
});
integration/payment.spec.ts
:
import { v4 as uuid } from 'uuid';
describe('payment', () => {
it('user can make payment', () => {
// Login.
cy.visit('/');
cy.findByRole('textbox', { name: /username/i }).type('sabertaz');
cy.findByLabelText(/password/i).type('secret');
cy.findByRole('checkbox', { name: /remember me/i }).check();
cy.findByRole('button', { name: /sign in/i }).click();
// Check account balance.
let oldBalance;
cy.get('[data-test=nav-user-balance]').then(
$balance => (oldBalance = $balance.text())
);
// Click on new button.
cy.findByRole('button', { name: /new/i }).click();
// Search for user.
cy.findByRole('textbox').type('devon becker');
cy.findByText(/devon becker/i).click();
// Add amount and note and click pay.
const paymentAmount = '5.00';
cy.findByPlaceholderText(/amount/i).type(paymentAmount);
const note = uuid();
cy.findByPlaceholderText(/add a note/i).type(note);
cy.findByRole('button', { name: /pay/i }).click();
// Return to transactions.
cy.findByRole('button', { name: /return to transactions/i }).click();
// Go to personal payments.
cy.findByRole('tab', { name: /mine/i }).click();
// Click on payment.
cy.findByText(note).click({ force: true });
// Verify if payment was made.
cy.findByText(`-$${paymentAmount}`).should('be.visible');
cy.findByText(note).should('be.visible');
// Verify if payment amount was deducted.
cy.get('[data-test=nav-user-balance]').then($balance => {
const convertedOldBalance = parseFloat(oldBalance.replace(/\$|,/g, ''));
const convertedNewBalance = parseFloat(
$balance.text().replace(/\$|,/g, '')
);
expect(convertedOldBalance - convertedNewBalance).to.equal(
parseFloat(paymentAmount)
);
});
});
});
const x = document.createElement('div');
Object.defineProperty(x, 'id', {
get() {
// devtool opened.
return 'id';
},
});
console.log(x);
// eslint-disable-next-line prefer-regex-literals
const c = new RegExp('1');
c.toString = function () {
// devtool opened
};
console.log(c);
Anti Method: hook
console
object, disable all outputs.
(function () {}.constructor('debugger')());
(() => {
function block() {
if (
window.outerHeight - window.innerHeight > 200 ||
window.outerWidth - window.innerWidth > 200
) {
document.body.innerHTML = 'Debug detected, please reload page!';
}
setInterval(() => {
(function () {
return false;
}
.constructor('debugger')
.call());
}, 50);
}
try {
block();
} catch (err) {}
})();
const startTime = new Date();
// debugger;
const endTime = new Date();
const isDev = endTime - startTime > 100;
while (true) {
// debugger;
}
Anti Method: use chrome protocol to block all
debugger
request. Anti Method: hookFunction.prototype.constructor
and replacedebugger
string.
Elements
panel: search DOM nodelong click reload: multiple reload options e.g clean cache
$0
: the reference to the currently selected element in the Elements panel.
const listener = getEventListeners($0).click[0].listener;
$0.removeEventListener('click', listener);
$0.addEventListener('click', e => {
// do something
// ...
// then
listener(e);
});
Blackbox script
item.Same thing in VSCode
debug panel (log points, break points etc).
C+S+P
: performance monitor.C+S+P
: FPS.script -> style -> layout -> paint -> composite
.3G
(slow network)sensor
(geolocation)audit
coverage
Tool for composite stage analysis:
{
"env": {
"browser": true,
"es2021": true,
"node": true,
"jest": true
},
"extends": [
"eslint:recommended",
"plugin:import/recommended",
"plugin:jsx-a11y/recommended",
"plugin:react/recommended",
"plugin:react-hooks/recommended",
"plugin:unicorn/recommended",
"plugin:promise/recommended",
"plugin:@typescript-eslint/recommended",
"plugin:prettier/recommended"
],
"parser": "@typescript-eslint/parser",
"parserOptions": {
"ecmaFeatures": {
"jsx": true
},
"ecmaVersion": 12,
"sourceType": "module"
},
"plugins": [
"import",
"jsx-a11y",
"react",
"react-hooks",
"@typescript-eslint"
],
"settings": {
"react": {
"version": "detect"
},
"import/parsers": {
"@typescript-eslint/parser": [".ts", ".tsx"]
},
"import/resolver": {
"typescript": {
"alwaysTryTypes": true,
"project": "./"
}
}
},
"rules": {
"react/prop-types": 0,
"react/jsx-props-no-spreading": 0
}
}
_method
: 表示私有化方法.jQuery
对象的变量使用 $
作为前缀.let/const
let/const
// bad
let i;
let len;
let dragonBall;
let items = getItems();
let goSportsTeam = true;
// bad
let i;
const items = getItems();
let dragonBall;
const goSportsTeam = true;
let len;
// good
const goSportsTeam = true;
const items = getItems();
let dragonBall;
let i;
let length;
// bad
(function example() {
// JavaScript 把它解释为
// let a = ( b = ( c = 1 ) );
// let 关键词只适用于变量 a ;变量 b 和变量 c 则变成了全局变量。
const a = (b = c = 1);
})();
console.log(a); // throws ReferenceError
console.log(b); // 1
console.log(c); // 1
// good
(function example() {
const a = 1;
const b = a;
const c = a;
})();
console.log(a); // throws ReferenceError
console.log(b); // throws ReferenceError
console.log(c); // throws ReferenceError
()
wrap multiple line// bad
const foo = superLongLongLongLongLongLongLongLongFunctionName();
// bad
const foo = 'superLongLongLongLongLongLongLongLongString';
// good
const foo = superLongLongLongLongLongLongLongLongFunctionName();
// good
const foo = 'superLongLongLongLongLongLongLongLongString';
// bad
// eslint-disable-next-line no-new-object
const item = new Object();
// good
const item = {};
// bad
const atom = {
lukeSkyWalker,
addValue(value) {
return atom.value + value;
},
};
// good
const atom = {
lukeSkyWalker,
addValue(value) {
return atom.value + value;
},
};
Object.prototype.XX
not object.xx
// bad
// 在模块范围内的缓存中查找一次
// eslint-disable-next-line no-prototype-builtins
console.log(object.hasOwnProperty(key));
// good
console.log(Object.prototype.hasOwnProperty.call(object, key));
// best
const has = Object.prototype.hasOwnProperty; // https://www.npmjs.com/package/has
console.log(has.call(object, key));
// very bad
const original = { a: 1, b: 2 };
const copy = Object.assign(original, { c: 3 }); // 变异的 `original` ಠ_ಠ
delete copy.a; // 这....
// bad
const original = { a: 1, b: 2 };
const copy = Object.assign({}, original, { c: 3 });
// good
const original = { a: 1, b: 2 };
const copy = { ...original, c: 3 }; // copy => { a: 1, b: 2, c: 3 }
const { a, ...noA } = copy; // noA => { b: 2, c: 3 }
.
for static name, use []
for variable name// good
const isJedi = luke.jedi;
function getProp(prop) {
return luke[prop];
}
push
not []
Array.from
(good)const foo = document.querySelectorAll('.foo');
// good
const nodes = Array.from(foo);
// best
const nodes = [...foo];
对于多个返回值使用对象解构,而不是数组解构
// bad
function processInputBad(input) {
// 处理代码...
return [left, right, top, bottom];
}
// 调用者需要考虑返回数据的顺序。
const [left, __, top] = processInputBad(input);
// good
function processInput(input) {
// 处理代码 ...
process();
return { left, right, top, bottom };
}
// 调用者只选择他们需要的数据。
const { left, top } = processInput(input);
'
not "
.${}
` not 'str1' + 'str2'
.// bad
function foo() {
// ...
}
// bad
const foo = function () {
// ...
};
// good
// 从变量引用调用中区分的词汇名称
const short = function longUniqueMoreDescriptiveLexicalFoo() {
// ...
};
...args
not arguments
// bad
function concatenateAll() {
// eslint-disable-next-line prefer-rest-params
const args = Array.prototype.slice.call(arguments);
return args.join('');
}
// good
function concatenateAll(...args) {
return args.join('');
}
// bad
function handleThings(opts) {
this.opts = opts || {};
}
// good
function handleThings(opts = {}) {
this.opts = opts;
}
()
and {}
should pair// bad
arr.map(x => x + 1);
arr.map((x, index) => x + index);
[1, 2, 3].map(x => {
const y = x + 1;
return x * y;
});
// good
arr.map(x => x + 1);
arr.map((x, index) => {
return x + index;
});
[1, 2, 3].map(x => {
const y = x + 1;
return x * y;
});
()
wrap multiple line return value// bad
['get', 'post', 'put'].map(httpMethod =>
Object.prototype.hasOwnProperty.call(
httpMagicObjectWithAVeryLongName,
httpMethod
)
);
// good
['get', 'post', 'put'].map(httpMethod =>
Object.prototype.hasOwnProperty.call(
httpMagicObjectWithAVeryLongName,
httpMethod
)
);
export from
.// bad
// filename es6.js
export { es6 as default } from './AirbnbStyleGuide';
// good
// filename es6.js
import { es6 } from './AirbnbStyleGuide';
export default es6;
// bad
// eslint-disable-next-line import/no-duplicates
import foo from 'foo';
// … 其他导入 … //
// eslint-disable-next-line import/no-duplicates
import { named1, named2 } from 'foo';
// good
import foo, { named1, named2 } from 'foo';
let
.// bad
// let foo = 3;
// export { foo };
// good
const foo = 3;
export { foo };
map/reduce/filter/any/every/some/find/findIndex/ ...
遍历数组Object.keys() / Object.values() / Object.entries()
迭代对象生成数组const numbers = [1, 2, 3, 4, 5];
// bad
let sum = 0;
for (const num of numbers) {
// eslint-disable-next-line no-const-assign
sum += num;
}
console.log(sum === 15);
// good
let sum = 0;
numbers.forEach(num => {
// eslint-disable-next-line no-const-assign
sum += num;
});
console.log(sum === 15);
// best (use the functional force)
const sum = numbers.reduce((total, num) => total + num, 0);
console.log(sum === 15);
// bad
const increasedByOne = [];
for (let i = 0; i < numbers.length; i++) {
increasedByOne.push(numbers[i] + 1);
}
// good
const increasedByOne = [];
numbers.forEach(num => {
increasedByOne.push(num + 1);
});
// best (keeping it functional)
const increasedByOne = numbers.map(num => num + 1);
function*
for generator// bad
function* fooBad() {
// ...
}
// bad
const bar = function* () {
// ...
};
// good
function* foo() {
// ...
}
// good
const foo = function* () {
// ...
};
if 语句使用 ToBoolean 的抽象方法来计算表达式的结果:
对于布尔值使用简写,但是对于字符串和数字进行显式比较
// bad
if (isValid === true) {
// ...
}
// good
if (isValid) {
// ...
}
// bad
if (someName) {
// ...
}
// good
if (someName !== '') {
// ...
}
// bad
if (collection.length) {
// ...
}
// good
if (collection.length > 0) {
// ...
}
Use {}
warp case
when exists const
/let
/function
/class
declaration:
// good
switch (foo) {
case 1: {
const x = 1;
break;
}
case 2: {
const y = 2;
break;
}
case 3: {
function f() {
// ...
}
break;
}
case 4:
bar();
break;
default: {
class C {}
}
}
// bad
const arr = [
[0, 1],
[2, 3],
[4, 5],
];
const objectInArray = [
{
id: 1,
},
{
id: 2,
},
];
const numberInArray = [1, 2];
// good
const arr = [
[0, 1],
[2, 3],
[4, 5],
];
const objectInArray = [
{
id: 1,
},
{
id: 2,
},
];
const numberInArray = [1, 2];
()
wrap multiple line assignment or arguments// good
const foo = superLongLongLongLongLongLongLongLongFunctionName();
['get', 'post', 'put'].map(httpMethod =>
Object.prototype.hasOwnProperty.call(
httpMagicObjectWithAVeryLongName,
httpMethod
)
);
Good places to use a white space include:
,
/;
后.+
,-
,*
,/
,<
,>
,=
前后.function () {}
.function foo() {}
.} if/for/while () {}
.} else {}
.()
[]
.{}
.let d = 0;
let a = b + 1;
if (a && b && c) {
d = a % c;
a += d;
}
// anti pattern
// missing or inconsistent spaces
// make the code confusing
let d = 0;
let a = b + 1;
if (a && b && c) {
d = a % c;
a += d;
}
/*
* comments
* comments
*/
/*
* @module app
* @namespace APP
*/
/*
* @class mathStuff
*/
/*
* @property propertyName
* @type Number/String
*/
/*
* @constructor
* @method sum
* @param {Number}/{String} instructions
* @return {Number}/{String} instructions
*/
Progressive Web Apps:
HTTPS
ServiceWorker
(web cache for offline and performance)// 20~100 ms for desktop
// 100 ms for mobile
const entry = performance.getEntriesByName(url)[0];
const swStartupTime = entry.requestStart - entry.workerStart;
NO SW
),const entry = performance.getEntriesByName(url)[0];
// no remote request means this was handled by the cache
if (entry.transferSize === 0) {
const cacheTime = entry.responseStart - entry.requestStart;
}
async function handleRequest(event) {
const cacheStart = performance.now();
const response = await caches.match(event.request);
const cacheEnd = performance.now();
}
5 caching strategy in workbox.
Stale-While-Revalidate:
// eslint-disable-next-line no-restricted-globals
self.addEventListener('fetch', function (event) {
event.respondWith(
caches.open(cacheName).then(function (cache) {
cache.match(event.request).then(function (cacheResponse) {
fetch(event.request).then(function (networkResponse) {
cache.put(event.request, networkResponse);
});
return cacheResponse || networkResponse;
});
})
);
});
Cache first, then Network:
// eslint-disable-next-line no-restricted-globals
self.addEventListener('fetch', function (event) {
event.respondWith(
caches.open(cacheName).then(function (cache) {
cache.match(event.request).then(function (cacheResponse) {
if (cacheResponse) return cacheResponse;
return fetch(event.request).then(function (networkResponse) {
cache.put(event.request, networkResponse.clone());
return networkResponse;
});
});
})
);
});
Network first, then Cache:
// eslint-disable-next-line no-restricted-globals
self.addEventListener('fetch', function (event) {
event.respondWith(
fetch(event.request).catch(function () {
return caches.match(event.request);
})
);
});
Cache only:
// eslint-disable-next-line no-restricted-globals
self.addEventListener('fetch', function (event) {
event.respondWith(
caches.open(cacheName).then(function (cache) {
cache.match(event.request).then(function (cacheResponse) {
return cacheResponse;
});
})
);
});
Network only:
// eslint-disable-next-line no-restricted-globals
self.addEventListener('fetch', function (event) {
event.respondWith(
fetch(event.request).then(function (networkResponse) {
return networkResponse;
})
);
});
// Check that service workers are registered
if ('serviceWorker' in navigator) {
// Use the window load event to keep the page load performance
window.addEventListener('load', () => {
navigator.serviceWorker.register('/sw.js');
});
}
Broken images use case:
function isImage(fetchRequest) {
return fetchRequest.method === 'GET' && fetchRequest.destination === 'image';
}
// eslint-disable-next-line no-restricted-globals
self.addEventListener('fetch', e => {
e.respondWith(
fetch(e.request)
.then(response => {
if (response.ok) return response;
// User is online, but response was not ok
if (isImage(e.request)) {
// Get broken image placeholder from cache
return caches.match('/broken.png');
}
})
.catch(err => {
// User is probably offline
if (isImage(e.request)) {
// Get broken image placeholder from cache
return caches.match('/broken.png');
}
process(err);
})
);
});
// eslint-disable-next-line no-restricted-globals
self.addEventListener('install', e => {
// eslint-disable-next-line no-restricted-globals
self.skipWaiting();
e.waitUntil(
caches.open('precache').then(cache => {
// Add /broken.png to "precache"
cache.add('/broken.png');
})
);
});
worker.terminate()
或在 worker 内部调用 self.close()
<button onclick="startComputation()">Start computation</button>
<script>
const worker = new Worker('worker.js');
worker.addEventListener(
'message',
function (e) {
console.log(e.data);
},
false
);
function startComputation() {
worker.postMessage({ cmd: 'average', data: [1, 2, 3, 4] });
}
</script>
// worker.js
// eslint-disable-next-line no-restricted-globals
self.addEventListener(
'message',
function (e) {
const data = e.data;
switch (data.cmd) {
case 'average': {
const result = calculateAverage(data);
// eslint-disable-next-line no-restricted-globals
self.postMessage(result);
break;
}
default:
// eslint-disable-next-line no-restricted-globals
self.postMessage('Unknown command');
}
},
false
);
// 文件名为index.js
function work() {
onmessage = ({ data: { jobId, message } }) => {
console.log(`i am worker, receive:-----${message}`);
postMessage({ jobId, result: 'message from worker' });
};
}
const makeWorker = f => {
const pendingJobs = {};
const worker = new Worker(
URL.createObjectURL(new Blob([`(${f.toString()})()`]))
);
worker.onmessage = ({ data: { result, jobId } }) => {
// 调用 resolve, 改变 Promise 状态
pendingJobs[jobId](result);
delete pendingJobs[jobId];
};
return (...message) =>
new Promise(resolve => {
const jobId = String(Math.random());
pendingJobs[jobId] = resolve;
worker.postMessage({ jobId, message });
});
};
const testWorker = makeWorker(work);
testWorker('message from main thread').then(message => {
console.log(`i am main thread, i receive:-----${message}`);
});
/*
* JSONParser.js
*/
// eslint-disable-next-line no-restricted-globals
self.onmessage = function (event) {
const jsonText = event.data;
const jsonData = JSON.parse(jsonText);
// eslint-disable-next-line no-restricted-globals
self.postMessage(jsonData);
};
/*
* main.js
*/
const worker = new Worker('JSONParser.js');
worker.onmessage = function (event) {
const jsonData = event.data;
evaluateData(jsonData);
};
worker.postMessage(jsonText);
// <img class="lzy_img" src="lazy_img.jpg" data-src="real_img.jpg" />
document.addEventListener('DOMContentLoaded', () => {
const imageObserver = new IntersectionObserver((entries, imgObserver) => {
entries.forEach(entry => {
if (entry.isIntersecting) {
const lazyImage = entry.target;
console.log('Lazy loading ', lazyImage);
lazyImage.src = lazyImage.dataset.src;
// only load image once
lazyImage.classList.remove('lzy');
imgObserver.unobserve(lazyImage);
}
});
});
const lazyImages = document.querySelectorAll('img.lzy_img');
lazyImages.forEach(lazyImage => imageObserver.observe(lazyImage));
});
如果文档中连续插入 1000 个 <li>
元素, 就会连续触发 1000 个插入事件,
执行每个事件的回调函数, 这很可能造成浏览器的卡顿;
而 Mutation Observer 完全不同, 只在 1000 个段落都插入结束后才会触发, 而且只触发一次.
Mutation Observer 有以下特点:
const mutationObserver = new MutationObserver(mutations => {
mutations.forEach(mutation => {
console.log(mutation);
});
});
// 开始侦听页面的根 HTML 元素中的更改。
mutationObserver.observe(document.documentElement, {
attributes: true,
characterData: true,
childList: true,
subtree: true,
attributeOldValue: true,
characterDataOldValue: true,
});
const target = document.querySelector('#container');
const callback = (mutations, observer) => {
mutations.forEach(mutation => {
switch (mutation.type) {
case 'attributes':
// the name of the changed attribute is in
// mutation.attributeName
// and its old value is in mutation.oldValue
// the current value can be retrieved with
// target.getAttribute(mutation.attributeName)
break;
case 'childList':
// any added nodes are in mutation.addedNodes
// any removed nodes are in mutation.removedNodes
break;
default:
throw new Error('Unsupported mutation!');
}
});
};
const observer = new MutationObserver(callback);
observer.observe(target, {
attributes: true,
attributeFilter: ['foo'], // only observe attribute 'foo'
attributeOldValue: true,
childList: true,
});
KeyframeEffect
.Animation
.const rabbitDownKeyframes = new KeyframeEffect(
whiteRabbit, // element to animate
[
{ transform: 'translateY(0%)' }, // keyframe
{ transform: 'translateY(100%)' }, // keyframe
],
{ duration: 3000, fill: 'forwards' } // keyframe options
);
const rabbitDownAnimation = new Animation(
rabbitDownKeyFrames,
document.timeline
);
whiteRabbit.addEventListener('click', downHandler);
function downHandler() {
rabbitDownAnimation.play();
whiteRabbit.removeEventListener('click', downHandler);
}
element.animate
.const animationKeyframes = [
{
transform: 'rotate(0)',
color: '#000',
},
{
color: '#431236',
offset: 0.3,
},
{
transform: 'rotate(360deg)',
color: '#000',
},
];
const animationTiming = {
duration: 3000,
iterations: Infinity,
};
const animation = document
.querySelector('alice')
.animate(animationKeyframes, animationTiming);
animation.currentTime
.animation.playState
.animation.effect
.animation.pause()/play()/reverse()/finish()/cancel()
.animation.pause();
animation.currentTime = animation.effect.getComputedTiming().duration / 2;
function currentTime(time = 0) {
animations.forEach(function (animation) {
if (typeof animation.currentTime === 'function') {
animation.currentTime(time);
} else {
animation.currentTime = time;
}
});
}
function createPlayer(animations) {
return Object.freeze({
play() {
animations.forEach(animation => animation.play());
},
pause() {
animations.forEach(animation => animation.pause());
},
currentTime(time = 0) {
animations.forEach(animation => (animation.currentTime = time));
},
});
}
const context = canvas.getContext('2d');
// 根据参数画线
function drawLine(fromX, fromY, toX, toY) {
context.moveTo(fromX, fromY);
context.lineTo(toX, toY);
context.stroke();
}
// 根据参数画圆
function drawCircle(x, y, radius, color) {
context.fillStyle = color;
context.beginPath();
context.arc(x, y, radius, 0, Math.PI * 2, true);
context.closePath();
context.fill();
context.stroke();
}
// 改变 canvas 中图形颜色
function changeColor(color) {
context.fillStyle = color;
context.fill();
}
for all objects:
position{x, y}
, speed{x, y}
, size{x, y}
const canvas = document.getElementById('gameScreen');
const ctx = canvas.getContext('2d');
const GAME_WIDTH = 800;
const GAME_HEIGHT = 600;
const game = new Game(GAME_WIDTH, GAME_HEIGHT);
let lastTime = 0;
function gameLoop(timestamp) {
const deltaTime = timestamp - lastTime;
lastTime = timestamp;
ctx.clearRect(0, 0, GAME_WIDTH, GAME_HEIGHT);
game.update(deltaTime);
game.draw(ctx);
requestAnimationFrame(gameLoop);
}
requestAnimationFrame(gameLoop);
Canvas buffer:
frontCanvasContext.drawImage(bufferCanvas, 0, 0);
const ctx = canvas.getContext('2d', { alpha: false });
Offscreen canvas:
// index.js
const offscreenCanvas = document.querySelector('#frame2');
const offscreen = offscreenCanvas.transferControlToOffscreen();
const worker = new Worker('./worker.js');
worker.postMessage({ canvas: offscreen }, [offscreen]);
// worker.js
onmessage = function (event) {
canvas = event.data.canvas;
context = canvas.getContext('2d');
};
-3 -1 1 4 6 9 11
-4 -2 0 2 3 5 7 8 10 12
.___________________________________________________________________________.
: | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | :
: | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | :
: | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | :
<-: |_| | |_| |_| | |_| |_| |_| | |_| |_| | |_| |_| |_| | |_| |_| :->
: | | | | | | | | | | | | | | | | | | :
: A | B | C | D | E | F | G | A | B | C | D | E | F | G | A | B | C | D | E :
:___|___|___|___|___|___|___|___|___|___|___|___|___|___|___|___|___|___|___:
^ ^ ^ ^ ^
220 Hz 440 Hz 523.25 Hz 880 Hz 1174.65 Hz
(-1 Octave) (middle A) (+1 Octave)
const audioContext = new AudioContext();
const baseFrequency = 440;
const getNoteFreq = (base, pitch) => base * Math.pow(2, pitch / 12);
// oscillator.frequency.value = getNoteFreq(440, 7);
const getNoteDetune = pitch => pitch * 100;
// oscillator.detune.value = getNoteDetune(7);
const play = (type, delay, pitch, duration) => {
const oscillator = audioContext.createOscillator();
oscillator.connect(audioContext.destination);
oscillator.type = type;
oscillator.detune.value = getNoteDetune(pitch);
const startTime = audioContext.currentTime + delay;
const stopTime = startTime + duration;
oscillator.start(startTime);
oscillator.stop(stopTime);
};
const sampleSize = 1024; // number of samples to collect before analyzing data
const audioUrl = 'viper.mp3';
let audioData = null;
let audioPlaying = false;
const audioContext = new AudioContext();
const sourceNode = audioContext.createBufferSource();
const analyserNode = audioContext.createAnalyser();
const javascriptNode = audioContext.createScriptProcessor(sampleSize, 1, 1);
// Create the array for the data values
const amplitudeArray = new Uint8Array(analyserNode.frequencyBinCount);
// Now connect the nodes together
sourceNode.connect(audioContext.destination);
sourceNode.connect(analyserNode);
analyserNode.connect(javascriptNode);
javascriptNode.connect(audioContext.destination);
// setup the event handler that is triggered
// every time enough samples have been collected
// trigger the audio analysis and draw the results
javascriptNode.onaudioprocess = function () {
// get the Time Domain data for this sample
analyserNode.getByteTimeDomainData(amplitudeArray);
// draw the display if the audio is playing
// if (audioPlaying == true) {
// requestAnimFrame(drawTimeDomain);
// }
};
// Load the audio from the URL via Ajax and store it in global variable audioData
// Note that the audio load is asynchronous
function loadSound(url) {
fetch(url)
.then(response => {
audioContext.decodeAudioData(response, buffer => {
audioData = buffer;
playSound(audioData);
});
})
.catch(error => {
console.error(error);
});
}
// Play the audio and loop until stopped
function playSound(buffer) {
sourceNode.buffer = buffer;
sourceNode.start(0); // Play the sound now
sourceNode.loop = true;
audioPlaying = true;
}
function stopSound() {
sourceNode.stop(0);
audioPlaying = false;
}
const WIDTH = this.canvas.clientWidth;
const HEIGHT = this.canvas.clientHeight;
this.analyserNode.fftSize = 256;
const bufferLengthAlt = this.analyserNode.frequencyBinCount;
const dataArrayAlt = new Uint8Array(bufferLengthAlt);
this.ctx.clearRect(0, 0, WIDTH, HEIGHT);
const draw = () => {
const drawVisual = requestAnimationFrame(draw);
this.analyserNode.getByteFrequencyData(dataArrayAlt);
this.ctx.fillStyle = 'rgb(255, 255, 255)';
this.ctx.fillRect(0, 0, WIDTH, HEIGHT);
const barWidth = (WIDTH / bufferLengthAlt) * 2.5;
let barHeight;
let x = 0;
for (let i = 0; i < bufferLengthAlt; i++) {
barHeight = dataArrayAlt[i];
this.ctx.fillStyle = `rgb(${barHeight + 100},15,156)`;
this.ctx.fillRect(x, HEIGHT - barHeight / 2, barWidth, barHeight / 2);
x += barWidth + 1;
}
};
draw();
JSON.parse
与 JSON.stringify
if (!localStorage.getItem('bgColor')) {
populateStorage();
} else {
setStyles();
}
function populateStorage() {
localStorage.setItem('bgColor', document.getElementById('bgColor').value);
localStorage.setItem('font', document.getElementById('font').value);
localStorage.setItem('image', document.getElementById('image').value);
setStyles();
}
function setStyles() {
const currentColor = localStorage.getItem('bgColor');
const currentFont = localStorage.getItem('font');
const currentImage = localStorage.getItem('image');
document.getElementById('bgColor').value = currentColor;
document.getElementById('font').value = currentFont;
document.getElementById('image').value = currentImage;
htmlElem.style.backgroundColor = `#${currentColor}`;
pElem.style.fontFamily = currentFont;
imgElem.setAttribute('src', currentImage);
}
export class IndexedDB {
constructor(dbName, dbVersion, dbUpgrade) {
return new Promise((resolve, reject) => {
this.db = null;
if (!('indexedDB' in window)) {
reject(new Error('not supported'));
}
const dbOpen = indexedDB.open(dbName, dbVersion);
if (dbUpgrade) {
dbOpen.onupgradeneeded = e => {
dbUpgrade(dbOpen.result, e.oldVersion, e.newVersion);
};
}
dbOpen.onsuccess = () => {
this.db = dbOpen.result;
resolve(this);
};
dbOpen.onerror = e => {
reject(new Error(`IndexedDB error: ${e.target.errorCode}`));
};
});
}
get(storeName, name) {
return new Promise((resolve, reject) => {
const transaction = this.db.transaction(storeName, 'readonly');
const store = transaction.objectStore(storeName);
const request = store.get(name);
request.onsuccess = () => {
resolve(request.result);
};
request.onerror = () => {
reject(request.error);
};
});
}
set(storeName, name, value) {
return new Promise((resolve, reject) => {
const transaction = this.db.transaction(storeName, 'readwrite');
const store = transaction.objectStore(storeName);
store.put(value, name);
transaction.oncomplete = () => {
resolve(true);
};
transaction.onerror = () => {
reject(transaction.error);
};
});
}
}
// IndexedDB usage
import { IndexedDB } from './indexedDB.js';
export class State {
static dbName = 'stateDB';
static dbVersion = 1;
static storeName = 'state';
static DB = null;
static target = new EventTarget();
constructor(observed, updateCallback) {
this.updateCallback = updateCallback;
this.observed = new Set(observed);
// subscribe `set` event with `updateCallback`
State.target.addEventListener('set', e => {
if (this.updateCallback && this.observed.has(e.detail.name)) {
this.updateCallback(e.detail.name, e.detail.value);
}
});
}
async dbConnect() {
State.DB =
State.DB ||
(await new IndexedDB(
State.dbName,
State.dbVersion,
(db, oldVersion, newVersion) => {
// upgrade database
switch (oldVersion) {
case 0: {
db.createObjectStore(State.storeName);
break;
}
default:
throw new Error('Unsupported version!');
}
}
));
return State.DB;
}
async get(name) {
this.observedSet.add(name);
const db = await this.dbConnect();
return await db.get(State.storeName, name);
}
async set(name, value) {
this.observed.add(name);
const db = await this.dbConnect();
await db.set(State.storeName, name, value);
// publish event to subscriber
const event = new CustomEvent('set', { detail: { name, value } });
State.target.dispatchEvent(event);
}
}
const networkType = navigator.connection.effectiveType; // 2G - 5G
Request Header:
GET /chat HTTP/1.1
Host: example.com
Upgrade: websocket
Connection: Upgrade
Sec-WebSocket-Key: 16-byte, base64 encoded
Sec-WebSocket-Version: 13
Sec-Websocket-Protocol: protocol [,protocol]*
Sec-Websocket-Extension: extension [,extension]*
Response Header:
HTTP/1.1 101 "Switching Protocols" or other description
Upgrade: websocket
Connection: Upgrade
Sec-WebSocket-Accept: 20-byte, MD5 hash in base64
Sec-Websocket-Protocol: protocol [,protocol]*
Sec-Websocket-Extension: extension [,extension]*
通信功能
function WebSocketTest() {
if ('WebSocket' in window) {
alert('WebSocket is supported by your Browser!');
// Let us open a web socket
const ws = new WebSocket('ws://localhost:9998/echo');
ws.onopen = function () {
// Web Socket is connected, send data using send()
ws.send('Message to send');
alert('Message is sent...');
};
ws.onmessage = function (evt) {
const received_msg = evt.data;
alert('Message is received...');
};
ws.onclose = function () {
// websocket is closed.
alert('Connection is closed...');
};
} else {
// The browser doesn't support WebSocket
alert('WebSocket NOT supported by your Browser!');
}
}
连接终止时,Web Socket 不会自动恢复, 需要自己实现, 通常为了保持连接状态,需要增加心跳机制.
每隔一段时间会向服务器发送一个数据包, 告诉服务器自己 Alive, 服务器端如果 Alive, 就会回传一个数据包给客户端. 主要在一些长时间连接的应用场景需要考虑心跳机制及重连机制, 以保证长时间的连接及数据交互.
const gamepads = {};
function gamepadHandler(event, connecting) {
// gamepad === navigator.getGamepads()[gamepad.index]
const { gamepad } = event;
if (connecting) {
gamepads[gamepad.index] = gamepad;
} else {
delete gamepads[gamepad.index];
}
}
window.addEventListener('gamepadconnected', e => {
gamepadHandler(e, true);
});
window.addEventListener('gamepaddisconnected', e => {
gamepadHandler(e, false);
});
if (window.navigator.geolocation) {
// getCurrentPosition第三个参数为可选参数
navigator.geolocation.getCurrentPosition(locationSuccess, locationError, {
// 指示浏览器获取高精度的位置,默认为false
enableHighAccuracy: true,
// 指定获取地理位置的超时时间,默认不限时,单位为毫秒
timeout: 5000,
// 最长有效期,在重复获取地理位置时,此参数指定多久再次获取位置。
maximumAge: 3000,
});
} else {
alert('Your browser does not support Geolocation!');
}
locationError 为获取位置信息失败的回调函数,可以根据错误类型提示信息:
function locationError(error) {
switch (error.code) {
case error.TIMEOUT:
showError('A timeout occurred! Please try again!');
break;
case error.POSITION_UNAVAILABLE:
showError("We can't detect your location. Sorry!");
break;
case error.PERMISSION_DENIED:
showError('Please allow geolocation access for this to work.');
break;
case error.UNKNOWN_ERROR:
showError('An unknown error occurred!');
break;
default:
throw new Error('Unsupported error!');
}
}
locationSuccess 为获取位置信息成功的回调函数,返回的数据中包含经纬度等信息,结合 Google Map API 即可在地图中显示当前用户的位置信息,如下:
function locationSuccess(position) {
const coords = position.coords;
const latlng = new google.maps.LatLng(
// 维度
coords.latitude,
// 精度
coords.longitude
);
const myOptions = {
// 地图放大倍数
zoom: 12,
// 地图中心设为指定坐标点
center: latlng,
// 地图类型
mapTypeId: google.maps.MapTypeId.ROADMAP,
};
// 创建地图并输出到页面
const myMap = new google.maps.Map(document.getElementById('map'), myOptions);
// 创建标记
const marker = new google.maps.Marker({
// 标注指定的经纬度坐标点
position: latlng,
// 指定用于标注的地图
map: myMap,
});
// 创建标注窗口
const infoWindow = new google.maps.InfoWindow({
content: `您在这里<br/>纬度: ${coords.latitude}<br/>经度:${coords.longitude}`,
});
// 打开标注窗口
infoWindow.open(myMap, marker);
}
navigator.geolocation.watchPosition(
locationSuccess,
locationError,
positionOption
);
自动更新地理位置
| Format | Size (bytes) | Download (ms) | Parse (ms) | | :------------------------------- | -----------: | ------------: | ---------: | | Verbose XML | 582,960 | 999.4 | 343.1 | | Verbose JSON-P | 487,913 | 598.2 | 0.0 | | Simple XML | 437,960 | 475.1 | 83.1 | | Verbose JSON | 487,895 | 527.7 | 26.7 | | Simple JSON | 392,895 | 498.7 | 29.0 | | Simple JSON-P | 392,913 | 454.0 | 3.1 | | Array JSON | 292,895 | 305.4 | 18.6 | | Array JSON-P | 292,912 | 316.0 | 3.4 | | Custom Format (script insertion) | 222,912 | 66.3 | 11.7 | | Custom Format (XHR) | 222,892 | 63.1 | 14.5 |
const localCache = {};
function xhrRequest(url, callback) {
// Check the local cache for this URL.
if (localCache[url]) {
callback.success(localCache[url]);
return;
}
// If this URL wasn't found in the cache, make the request.
const req = createXhrObject();
req.onerror = function () {
callback.error();
};
req.onreadystatechange = function () {
if (req.readyState === 4) {
if (req.responseText === '' || req.status === '404') {
callback.error();
return;
}
// Store the response on the local cache.
localCache[url] = req.responseText;
callback.success(req.responseText);
}
};
}
req.open('GET', url, true);
// req.set();
req.send(null);
client.request(config)
.client.get(url[, config])
.client.delete(url[, config])
.client.head(url[, config])
.client.options(url[, config])
.client.post(url[, data[, config]])
.client.put(url[, data[, config]])
.client.patch(url[, data[, config]])
.client.getUri([config])
.const client = axios.create({
baseURL: 'https://some-domain.com/api/',
timeout: 1000,
headers: { 'X-Custom-Header': 'foobar' },
});
// Add a request interceptor
client.interceptors.request.use(
config => {
// Do something before request is sent.
return config;
},
error => {
// Do something with request error.
return Promise.reject(error);
}
);
client.interceptors.response.use(
response => {
// Any status code that lie within the range of 2xx trigger this function.
// Do something with response data.
return response;
},
error => {
// Any status codes that falls outside the range of 2xx trigger this function.
// Do something with response error.
return Promise.reject(error);
}
);
const XHR = (function () {
const standard = {
createXHR() {
return new XMLHttpRequest();
},
};
const newActionXObject = {
createXHR() {
return new ActionXObject('Msxml12.XMLHTTP');
},
};
const oldActionXObject = {
createXHR() {
return new ActionXObject('Microsoft.XMLHTTP');
},
};
// 根据兼容性返回对应的工厂对象
// 此立即函数运行一次即可完成兼容性检查, 防止重复检查
if (standard.createXHR()) {
return standard;
} else {
try {
newActionXObject.createXHR();
return newActionXObject;
} catch (o) {
oldActionXObject.createXHR();
return oldActionXObject;
}
}
})();
const request = XHR.createXHR();
// 3rd argument : async mode
request.open('GET', 'example.txt', true);
request.onreadystatechange = function () {
// do something
/*
switch(request.readyState) {
case 0: initialize
case 1: loading
case 2: loaded
case 3: transaction
case 4: complete
}
*/
if (request.readyState === 4) {
const para = document.createElement('p');
const txt = document.createTextNode(request.responseText);
para.appendChild(txt);
document.getElementById('new').appendChild(para);
}
};
request.send(null);
ajax({
url: './TestXHR.aspx', // 请求地址
type: 'POST', // 请求方式
data: { name: 'super', age: 20 }, // 请求参数
dataType: 'json',
success(response, xml) {
// 此处放成功后执行的代码
},
fail(status) {
// 此处放失败后执行的代码
},
});
function ajax(options) {
options = options || {};
options.type = (options.type || 'GET').toUpperCase();
options.dataType = options.dataType || 'json';
const params = formatParams(options.data);
let xhr;
// 创建 - 非IE6 - 第一步
if (window.XMLHttpRequest) {
xhr = new XMLHttpRequest();
} else {
// IE6及其以下版本浏览器
xhr = new ActiveXObject('Microsoft.XMLHTTP');
}
// 接收 - 第三步
xhr.onreadystatechange = function () {
if (xhr.readyState === 4) {
const status = xhr.status;
if (status >= 200 && status < 300) {
options.success && options.success(xhr.responseText, xhr.responseXML);
} else {
options.fail && options.fail(status);
}
}
};
// 连接 和 发送 - 第二步
if (options.type === 'GET') {
xhr.open('GET', `${options.url}?${params}`, true);
xhr.send(null);
} else if (options.type === 'POST') {
xhr.open('POST', options.url, true);
// 设置表单提交时的内容类型
xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
xhr.send(params);
}
}
// 格式化参数
function formatParams(data) {
const arr = [];
for (const name in data) {
arr.push(`${encodeURIComponent(name)}=${encodeURIComponent(data[name])}`);
}
arr.push(`v=${Math.random()}`.replace('.', ''));
return arr.join('&');
}
<!-- HTML -->
<meta http-equiv="Access-Control-Allow-Origin" content="*" />
Response.Headers.Add('Access-Control-Allow-Origin', '*');
$.ajax({
url: 'http://map.oicqzone.com/gpsApi.php?lat=22.502412986242&lng=113.93832783228',
type: 'GET',
dataType: 'JSONP', // 处理Ajax 跨域问题.
success(data) {
$('body').append(`Name: ${data}`);
},
});
Get and Post:
const response = await fetch('/api/names', {
headers: {
Accept: 'application/json',
},
});
const response = await fetch('/api/names', {
method: 'POST',
body: JSON.stringify(object),
headers: {
'Content-Type': 'application/json',
},
});
Request object:
const request = new Request('/api/names', {
method: 'POST',
body: JSON.stringify(object),
headers: {
'Content-Type': 'application/json',
},
});
const response = await fetch(request);
JamStack 指的是一套用于构建现代网站的技术栈:
CSR (Client Side Rendering): SPA -> SSR (Server Side Rendering): SPA with SEO -> SSG (Static Site Generation): SPA with pre-rendering -> ISR (Incremental Static Regeneration) = SSG + SSR.
import { TimeSection } from '@components';
export default function CSRPage() {
const [dateTime, setDateTime] = React.useState<string>();
React.useEffect(() => {
axios
.get('https://worldtimeapi.org/api/ip')
.then(res => {
setDateTime(res.data.datetime);
})
.catch(error => console.error(error));
}, []);
return (
<main>
<TimeSection dateTime={dateTime} />
</main>
);
}
if (isBotAgent) {
// return pre-rendering static html to search engine crawler
// like Gatsby
} else {
// server side rendering at runtime for real interactive users
// ReactDOMServer.renderToString()
}
import { TimeSection } from '@components';
export default function SSRPage({ dateTime }: SSRPageProps) {
return (
<main>
<TimeSection dateTime={dateTime} />
</main>
);
}
export const getServerSideProps: GetServerSideProps = async () => {
const res = await axios.get('https://worldtimeapi.org/api/ip');
return {
props: { dateTime: res.data.datetime },
};
};
npm run build
.import { TimeSection } from '@components';
export default function SSGPage({ dateTime }: SSGPageProps) {
return (
<main>
<TimeSection dateTime={dateTime} />
</main>
);
}
export const getStaticProps: GetStaticProps = async () => {
const res = await axios.get('https://worldtimeapi.org/api/ip');
return {
props: { dateTime: res.data.datetime },
};
};
import { TimeSection } from '@components';
export default function ISR20Page({ dateTime }: ISR20PageProps) {
return (
<main>
<TimeSection dateTime={dateTime} />
</main>
);
}
export const getStaticProps: GetStaticProps = async () => {
const res = await axios.get('https://worldtimeapi.org/api/ip');
return {
props: { dateTime: res.data.datetime },
revalidate: 20,
};
};
<title>
and <meta>
in <head>
(with tool like react-helmet
).robots.txt
file.import { Helmet } from 'react-helmet';
function App() {
const seo = {
title: 'About',
description:
'This is an awesome site that you definitely should check out.',
url: 'https://www.mydomain.com/about',
image: 'https://mydomain.com/images/home/logo.png',
};
return (
<Helmet
title={`${seo.title} | Code Mochi`}
meta={[
{
name: 'description',
property: 'og:description',
content: seo.description,
},
{ property: 'og:title', content: `${seo.title} | Code Mochi` },
{ property: 'og:url', content: seo.url },
{ property: 'og:image', content: seo.image },
{ property: 'og:image:type', content: 'image/jpeg' },
{ property: 'twitter:image:src', content: seo.image },
{ property: 'twitter:title', content: `${seo.title} | Code Mochi` },
{ property: 'twitter:description', content: seo.description },
]}
/>
);
}
const obj = JSON.parse(json);
const json = JSON.stringify(obj);
JSON.stringify
:
Symbol
/function
/NaN
/Infinity
/undefined
: null
/ignored.BitInt
: throw TypeError
.TypeError
.toJSON
method:const obj = {
name: 'zc',
toJSON() {
return 'return toJSON';
},
};
// return toJSON
console.log(JSON.stringify(obj));
// "2022-03-06T08:24:56.138Z"
JSON.stringify(new Date());
HyperText Transfer Protocol (HTTP) + Transport Layer Security (TLS):
证书获取及验证过程 (CA 认证体系):
加密密钥传输 (B 端加密 - 传输 - S 端解密):
加密报文传输 (S 端加密 - 传输 - B 端解密):
在 HTTP/1.x 中,每次请求都会建立一次 HTTP 连接:
HTTP/2 的多路复用就是为了解决上述的两个性能问题. 在 HTTP/2 中, 有两个非常重要的概念, 分别是帧(frame)和流(stream). 帧代表着最小的数据单位, 每个帧会标识出该帧属于哪个流, 流也就是多个帧组成的数据流. 多路复用, 就是在一个 TCP 连接中可以存在多条流, 避免队头阻塞问题和连接数过多问题.
HTTP/2 = HTTP
+ HPack / Stream
+ TLS 1.2+
+ TCP
:
HTTP/2 虽然通过多路复用解决了 HTTP 层的队头阻塞, 但仍然存在 TCP 层的队头阻塞.
HTTP/3 = HTTP
+ QPack / Stream
+ QUIC / TLS 1.3+
+ UDP
:
Use reasonable HTTP status codes:
Cross Origin Resource Sharing:
protocol + host + port
.Cache-Control
, Content-Language
, Content-Length
, Content-Type
,
Expires
, Last-Modified
, Pragma
.Access-Control-Allow-Origin: *
Access-Control-Expose-Headers: X-Custom-Header, Content-Encoding
Access-Control-Expose-Headers: *
Access-Control-Allow-Credentials: true
Access-Control-Allow-Origin: https://example.com
Vary: Cookie, Origin
Access-Control-Max-Age: 600
Access-Control-Allow-Methods: Custom-Method, CUSTOM-METHOD
Access-Control-Allow-Headers: X-Custom-Header
HTTP basic authentication is 401 authentication:
Get /index.html HTTP/1.0
Host:www.google.com
401
WWW-Authenticate: Basic realm=”google.com”
HTTP/1.0 401 Unauthorized
Server: SokEvo/1.0
WWW-Authenticate: Basic realm=”google.com”
Content-Type: text/html
Content-Length: xxx
Get /index.html HTTP/1.0
Host:www.google.com
Authorization: Basic d2FuZzp3YW5n
HTTP 协议是一个无状态的协议,
服务器不会知道到底是哪一台浏览器访问了它,
因此需要一个标识用来让服务器区分不同的浏览器.
Cookie 就是这个管理服务器与客户端之间状态的标识.
Response header with Set-Cookie
, Request header with Cookie
.
浏览器第一次访问服务端, 服务端就会创建一次 Session, 在会话中保存标识该浏览器的信息. Session 缓存在服务端, Cookie 缓存在客户端, 他们都由服务端生成, 实现 HTTP 协议的状态.
Set-Cookie: <Session ID>
.Cookie: <Session ID>
.<Session ID>
进行用户验证,
利用 Session Cookie 机制可以简单地实现用户登录状态验证,
保护需要登录权限才能访问的路由服务.Max-Age
priority higher than Expires
.
When both to null
, cookie become session cookie.Set-Cookie: username=tazimi; domain=tazimi.dev; Expires=Wed, 21 Oct 2022 08:00:00
Set-Cookie: username=tazimi; domain=tazimi.dev; path=/blog
Set-Cookie: username=tazimi; domain=tazimi.dev; path=/blog; Secure; HttpOnly
Set-Cookie: username=tazimi; domain=github.com
Set-Cookie: height=100; domain=me.github.com
Set-Cookie: weight=100; domain=me.github.com
Authorization: <Token>
.{"alg": "HS256","typ": "JWT"}
..
分隔: HeaderBase64.PayloadBase64.Signature
.access token
.
越是权限敏感的业务, access token
有效期足够短, 以避免被盗用.access token
的 token, 称为 refresh token
.
refresh token
用来获取 access token
, 有效期更长,
通过独立服务和严格的请求方式增加安全性.fetch
/axios
request headers for rest requests
Redux
/Vuex
global state.localStorage
/sessionStorage
.OAuth (Open Authorization) 是一个开放标准, 作用于第三方授权和第三方访问. 用户数据的所有者告诉系统, 同意授权第三方应用进入系统, 获取这些数据. 系统从而产生一个短期进入令牌 (Token), 用来代替密码供第三方应用使用.
第三方应用申请令牌之前, 都必须先到系统备案, 说明自己的身份, 然后会拿到两个身份识别码: Client ID 和 Client Secret. 这是为了防止令牌被滥用, 没有备案过的第三方应用拿不到令牌 (Token).
OAuth Token 特征:
https://github.com/login/oauth/authorize/?client_id=${clientID}
.access_token
令牌
https://github.com/login/oauth/access_token?client_id=${clientID}&client_secret=${clientSecret}&code=${code}
(3rd-Party Server vs Authorization Server)https://api.github.com/user?access_token=${accessToken}
或者 Request Header Authorization: token ${accessToken}
.
可以构建第三方网站自己的 Token, 做进一步相关鉴权操作 (如 Session Cookie).
(3rd-Party Server vs Resource Server)OAuth 2.0 允许自动更新令牌. 资源所有者颁发令牌时一次性颁发两个令牌, 一个用于获取数据 (Access Token), 另一个用于获取新的令牌 (Refresh Token). 令牌到期前, 第三方网站使用 Refresh Token 发请求更新令牌:
https://github.com/login/oauth/access_token
?client_id=CLIENT_ID
&client_secret=CLIENT_SECRET
&grant_type=refresh_token
&refresh_token=REFRESH_TOKEN
SSO
:
单点登录要求不同域下的系统一次登录,全线通用,
通常由独立的 SSO
系统记录登录状态, 下发 ticket
,
各业务系统配合存储和认证 ticket
.
CSP help prevent from XSS
{
"header": {
"Content-Security-Policy":
script-src 'nonce-random123' 'strict-dynamic' 'unsafe-eval';
object-src 'none';
base-uri 'none'
}
}
<script>
alert('xss');
</script>
// XSS injected by attacker - blocked by CSP
<script nonce="random123">
alert('this is fine!)
</script>
<script nonce="random123" src="https://cdnjs.com/lib.js"></script>
nonce only CSP block 3rd scripts and dynamic scripts generate by trusted users, 'strict-dynamic' can tackle it.
<!-- Content-Security-Policy: script-src 'nonce-random123' 'strict-dynamic' -->
<script nonce="random123">
const s = document.createElement('script)
s.src = '/path/to/script.js';
document.head.appendChild(s); // can execute correctly
</script>
<!-- Given this CSP header -->
Content-Security-Policy: script-src https://example.com/
<!-- The following third-party script will not be loaded or executed -->
<script src="https://not-example.com/js/library.js"></script>
// fallback policy
TrustedTypes.createPolicy(
'default',
{
createHTML(s) {
console.error('Please fix! Insecure string assignment detected:', s);
return s;
},
},
true
);
// Content-Security-Policy-Report-Only: trusted-types myPolicy; report-uri /cspReport
const SanitizingPolicy = TrustedTypes.createPolicy(
'myPolicy',
{
createHTML: (s: string) => myCustomSanitizer(s),
},
false
);
const trustedHTML = SanitizingPolicy.createHTML(foo);
element.innerHTML = trustedHTML;
GET request
没有副作用.request
正常渠道发起 (Hidden token check in form).# Reject cross-origin requests to protect from
# CSRF, XSSI & other bugs
def allow_request(req):
# Allow requests from browsers which don't send Fetch Metadata
if not req['sec-fetch-site']:
return True
# Allow same-site and browser-initiated requests
if req['sec-fetch-site'] in ('same-origin', 'same-site', 'none'):
return True
# Allow simple top-level navigation from anywhere
if req['sec-fetch-mode'] == 'navigate' and req.method == 'GET':
return True
return False
eval()
:
它能访问执行上下文中的局部变量, 也能访问所有全局变量, 是一个非常危险的函数.new Function()
:
在全局作用域中被创建, 不会创建闭包.
当运行函数时, 只能访问本地变量和全局变量,
不能访问 Function 构造器被调用生成的上下文的作用域.with () {}
:
它首先会在传入的对象中查找对应的变量,
如果找不到就会往更上层的全局作用域去查找,
导致全局环境污染.ProxySandbox:
function sandbox(code) {
code = `with (sandbox) {${code}}`;
// eslint-disable-next-line no-new-func
const fn = new Function('sandbox', code);
return function (sandbox) {
const sandboxProxy = new Proxy(sandbox, {
has(target, key) {
return true;
},
get(target, key) {
if (key === Symbol.unscopables) return undefined;
return target[key];
},
});
return fn(sandboxProxy);
};
}
// 简化伪代码示例
const frame = document.body.appendChild(
document.createElement('iframe', {
src: 'about:blank',
sandbox:
'allow-scripts allow-same-origin allow-popups allow-presentation allow-top-navigation',
style: 'display: none;',
})
);
const window = new Proxy(frame.contentWindow, {});
const document = new Proxy(document, {});
const location = new Proxy(window.location);
const history = new Proxy(window.history);
// eslint-disable-next-line no-new-func
const sandbox = new Function(`
return function ({ window, location, history, document }, code){
with(window) {
${code}
}
}`);
sandbox().call(window, { window, location, history, document }, code);
function getCanvasFingerprint() {
const canvas = document.createElement('canvas');
const context = canvas.getContext('2d');
context.font = '18pt Arial';
context.textBaseline = 'top';
context.fillText('Hello, user.', 2, 2);
return canvas.toDataURL('image/jpeg');
}
getCanvasFingerprint();
Fingerprinting is a technique that makes the name of a file, dependent on the contents of the file, not on the timestamp differ from servers. When the file contents change, the filename is also changed. For content that is static or infrequently changed, this provides an easy way to tell whether two versions of a file are identical, even across different servers or deployment dates.
When a filename is unique and based on its content, HTTP headers
can be set to encourage caches(code: 200
) everywhere
(whether at CDNs, at ISPs, in networking equipment, or in web browsers)
to keep their own copy of the content.
When the content is updated(),
the fingerprint will change.
This will cause the remote clients to request a new copy of the content.
This is generally known as cache busting.
两套系统, 一套稳定的绿色系统, 一套即将发布的蓝色系统. 不断切换并迭代发布到生产环境中.
多个集群实例的服务中, 在不影响服务的情况下, 停止一个或多个实例, 逐步进行版本更新.
Canary Release: 全量或增量部署新文件, 并逐步把流量切换至新 CDN URL. 根据灰度白名单, 将灰度测试用户的 CDN Assets 更换至不同 Version Number 或者 Fingerprint 的新版本前端页面文件.
通过灰度发布收集用户反馈 (转化率等指标), 决定后续是否全面将所有流量切至新版本, 或者完全放弃新版本, 亦或是通过 FLAGS 结合用户特征图像, (如用户级别, UA, Cookie Location, IP, Feature List 等) 只向部分流量投放新版本. 可以实现千人千页, 每个用户获得由不同的功能 (FLAGS 开启关闭) 组成的不同页面.
业界成熟的灰度方案:
# Canary Deployment
map $COOKIE_canary $group {
# canary account
~*devui$ server_canary;
default server_default;
}
# 流量均分, 注释掉其中某一边, 另一边为灰度流量访问边
upstream server_canary {
server 11.11.11.11:8000 weight=1 max_fails=1 fail_timeout=30s;
server 22.22.22.22 weight=1 max_fails=1 fail_timeout=30s;
}
# 流量均分, 注释掉其中某一边, 另一边为正常流量访问边
upstream server_default {
server 11.11.11.11:8000 weight=2 max_fails=1 fail_timeout=30s;
server 22.22.22.22 weight=2 max_fails=1 fail_timeout=30s;
}
# 配置 8000 端口的转发规则, 并且 expose port
server {
listen 8000;
server_name _;
root /var/canaryDemo;
# Load configuration files for the default server block.
include /etc/nginx/default.d/*.conf;
location / {
root /var/canaryDemo;
index index.html;
try_files $uri $uri/ /index.html;
}
}
server {
listen 80 default_server;
listen [::]:80 default_server;
server_name _;
# root /usr/share/nginx/html;
root /var/canaryDemo;
# Load configuration files for the default server block.
include /etc/nginx/default.d/*.conf;
location / {
proxy_pass http://$group;
# root /var/canaryDemo;
# index index.html;
}
error_page 404 /404.html;
location = /40x.html {
}
error_page 500 502 503 504 /50x.html;
location = /50x.h
}
前端优化: 每一个页面都需要去获取灰度规则,这个灰度请求将阻塞页面. 可以使用 localStorage 存储这个用户是否为灰度用户, 然后定期的更新 localStorage, 取代大量的请求造成的体验问题.
后端优化: 利用 MemCache 在内存中缓存灰度规则与灰度用户列表, 提升灰度发布性能.
从防御的角度来讲, 内部风险是外部风险的超集: 当攻击者攻陷任何一个内部人员 (合法用户或员工) 的设备后, 攻击者便成了内部人员. 零信任 从这个角度看就是假设任何一台主机都有可能被攻陷.
零信任并不是完全没有信任, 而是几个基本的最小化的信任根 (Root of Trust), 重构信任链 (Chain of Trust). 通过一系列的标准化流程 (Standard Process) 建立的一个完整的信任链 (信任树 Tree of Trust 或者信任网 Web of Trust).
几个典型的例子包括:
身份 2.0 是对于以上的信任链的标准化, 以便于在安全访问策略中使用这些在建立信任过程中收集到的信息.
在身份 2.0 中, 一切本体 (Entity) 都有身份. 用户有用户身份, 员工有员工身份, 机器有机器身份, 软件有软件身份.
在身份 2.0 中, 一切访问 (Access) 都带有访问背景 (Access Context):
持续访问控制会在软件开发和运行的各个环节持续地进行访问控制:
零信任的实施依赖于扎实的基础安全架构, 没有基础就没有上层建筑. 谷歌零信任依赖于以下基础设施提供的基本安全保障:
babel example.js -o compiled.js
babel src -d lib -s
A read-eval-print loop(REPL) can replace node REPL.
提供 babel 转码 API
npm install babel-core --save
const babel = require('babel-core');
// 字符串转码
babel.transform('code();', options);
// => { code, map, ast }
// 文件转码(异步)
babel.transformFile('filename.js', options, function (err, result) {
process(err);
return result; // => { code, map, ast }
});
// 文件转码(同步)
babel.transformFileSync('filename.js', options);
// => { code, map, ast }
// Babel AST转码
babel.transformFromAst(ast, code, options);
// => { code, map, ast }
Use Babel to refactor code:
babel-plugin-transform-xxx
.{
"main": "index.js"
}
// index.js
module.exports = babel => {
const t = babel.types;
let isJSXExisted = false;
let isMeactContextEnabled = false;
return {
visitor: {
Program: {
exit(path) {
if (isJSXExisted === true && isMeactContextEnabled === false) {
throw path.buildCodeFrameError(`Meact isn't in current context!`);
}
},
},
ImportDeclaration(path, state) {
if (path.node.specifiers[0].local.name === 'Meact') {
isMeactContextEnabled = true;
}
},
MemberExpression(path, state) {
if (
path.node.object.name === 'React' &&
path.node.property.name === 'createElement'
) {
isJSXExisted = true;
path.replaceWith(
t.MemberExpression(
t.identifier('Meact'),
t.identifier('createElement')
)
);
}
},
},
};
};
.babelrc.js
.babel-preset-xxx
.// package.json
{
"main": "index.js",
"dependencies": {
"babel-plugin-transform-meact-jsx": "^0.1.2",
"@babel/plugin-proposal-object-rest-spread": "^7.0.0",
"@babel/plugin-transform-react-jsx": "^7.0.0",
"@babel/plugin-transform-runtime": "^7.0.0",
"@babel/preset-env": "^7.0.0"
}
}
// index.js
const defaultTargets = {
android: 30,
chrome: 35,
edge: 14,
explorer: 9,
firefox: 52,
safari: 8,
ucandroid: 1,
};
const buildTargets = options => {
return Object.assign({}, defaultTargets, options.additionalTargets);
};
module.exports = function buildMeactPreset(context, options) {
const transpileTargets =
(options && options.targets) || buildTargets(options || {});
return {
presets: [
require('@babel/preset-env').default(null, {
targets: transpileTargets,
modules: false,
}),
],
plugins: [
require('@babel/plugin-proposal-object-rest-spread'),
require('@babel/plugin-transform-react-jsx'),
require('babel-plugin-transform-meact-jsx'),
require('@babel/plugin-transform-runtime'),
].filter(Boolean),
};
};
Enable webpack configuration types intellisense:
npm i -D webpack webpack-cli webpack-dev-server
Enable devServer
type intellisense:
# Add `devServer` type to `webpack.Configuration`
npm i -D @types/webpack-dev-server
/** @type {import('webpack').Configuration} */
module.exports = {
entry: {
main: './src/index.ts',
},
output: {
filename: devMode ? '[name].js' : '[name].[contenthash].js',
path: path.resolve(__dirname, 'build'),
},
mode: devMode ? 'development' : 'production',
devServer: {
hot: true,
open: true,
port: 2333,
},
};
HMR:
[hash].hot-update.json
) 资源文件, 确认增量变更范围.module.hot.accept
回调, 执行代码变更逻辑.module.hot.accept
有两种调用模式:
module.hot.accept()
: 当前文件修改后, 重头执行当前文件代码.module.hot.accept(path, callback)
: 常用模式, 监听模块变更, 执行代码变更逻辑.// 该模块修改后, `console.log('bar')` 会重新执行
console.log('bar');
module.hot.accept();
import component from './component';
let demoComponent = component();
document.body.appendChild(demoComponent);
if (module.hot) {
module.hot.accept('./component', () => {
const nextComponent = component();
document.body.replaceChild(nextComponent, demoComponent);
demoComponent = nextComponent;
});
}
react-refresh-webpack-plugin
/vue-loader
/style-loader
利用 module.hot.accept
实现了 HMR (forceUpdate), 无需开发者编写热模块更新逻辑.
echo fs.notify.max_user_watches=524288 | sudo tee -a /etc/sysctl.conf
sudo sysctl -p
const TsconfigPathsPlugin = require('tsconfig-paths-webpack-plugin');
module.exports = {
resolve: {
alias: {
'#': path.resolve(__dirname, '/'),
'~': path.resolve(__dirname, 'src'),
'@': path.resolve(__dirname, 'src'),
'~@': path.resolve(__dirname, 'src'),
vendor: path.resolve(__dirname, 'src/vendor'),
'~component': path.resolve(__dirname, 'src/components'),
'~config': path.resolve(__dirname, 'config'),
},
extensions: ['.tsx', '.ts', '.jsx', '.js'],
plugins: [new TsconfigPathsPlugin({ configFile: './tsconfig.json' })],
},
};
get baseUrl
and paths
from tsconfig.json
:
const TsconfigPathsPlugin = require('tsconfig-paths-webpack-plugin');
module.exports = {
resolve: {
plugins: [new TsconfigPathsPlugin({ configFile: './tsconfig.json' })],
},
};
jsconfig.json
for vscode resolve path:
{
"compilerOptions": {
// This must be specified if "paths" is set
"baseUrl": ".",
// Relative to "baseUrl"
"paths": {
"*": ["*", "src/*"]
}
}
}
{
"compilerOptions": {
"target": "es2017",
"allowSyntheticDefaultImports": false,
"baseUrl": "./",
"paths": {
"Config/*": ["src/config/*"],
"Components/*": ["src/components/*"],
"Ducks/*": ["src/ducks/*"],
"Shared/*": ["src/shared/*"],
"App/*": ["src/*"]
}
},
"exclude": ["node_modules", "dist"]
}
| devtool | build | rebuild | production | quality | | ---------------------------- | ------- | ------- | ---------- | ------------ | | (none) / false | fastest | fastest | yes | bundle | | eval | fast | fastest | no | generated | | eval-cheap-source-map | ok | fast | no | transformed | | eval-cheap-module-source-map | slow | fast | no | lines only | | eval-source-map | slowest | ok | no | lines + rows |
cache
is set to type: 'memory'
in development mode
and disabled in production mode.
cache: true
is an alias to cache: { type: 'memory' }
.
Accelerate second build time:
module.exports = {
cache: {
type: 'filesystem',
},
};
[contenthash]
and long-term browser cache to improve second access time.const path = require('path');
module.exports = {
entry: {
'bod-cli.min': path.join(__dirname, './src/index.js'),
'bod-cli': path.join(__dirname, './src/index.js'),
},
output: {
path: path.join(__dirname, './dist'),
filename: '[name].[contenthash].js',
library: 'bod',
libraryExport: 'default',
libraryTarget: 'esm',
globalObject: 'this',
},
};
const config = {
test: /\.(js|mjs|jsx|ts|tsx)$/,
include: path.resolve('src'),
use: [
'thread-loader',
{
loader: require.resolve('babel-loader'),
},
],
options: {
customize: require.resolve('babel-preset-react-app/webpack-overrides'),
plugins: [
[
require.resolve('babel-plugin-named-asset-import'),
{
loaderMap: {
svg: {
ReactComponent: '@svgr/webpack?-svgo,+titleProp,+ref![path]',
},
},
},
],
['lodash'],
],
cacheDirectory: true,
cacheCompression: false,
compact: isEnvProduction,
},
};
const MiniCssExtractPlugin = require('mini-css-extract-plugin');
const CssMinimizerPlugin = require('css-minimizer-webpack-plugin');
const devMode = process.env.NODE_ENV !== 'production';
module.exports = {
module: {
rules: [
{
test: /.s?css$/,
exclude: /node_modules$/,
use: [
devMode ? 'style-loader' : MiniCssExtractPlugin.loader,
{
loader: 'css-loader',
options: {
modules: {
compileType: 'module',
localIdentName: '[local]__[hash:base64:5]',
},
},
},
'sass-loader',
{
loader: 'postcss-loader',
options: {
postcssOptions: {
plugins: [['autoprefixer']],
},
},
},
],
},
],
},
optimization: {
minimizer: [
// `...`,
new CssMinimizerPlugin(),
],
},
};
asset/resource
emits separate file and exports the URL
(file-loader
).asset/inline
exports data URI of the asset
(url-loader).asset/source
exports source code of the asset
(raw-loader).asset
automatically chooses between exporting data URI and separate file
(url-loader
with asset size limit, default 8kb
).const config = {
rules: [
{
test: /\.(png|jpg|gif|jpeg|webp|svg|eot|ttf|woff|woff2)$/,
type: 'asset',
parser: {
dataUrlCondition: {
maxSize: 4 * 1024, // 4kb
},
},
},
],
};
const path = require('path');
module.exports = {
entry: './src/index.js',
output: {
filename: 'main.js',
path: path.resolve(__dirname, 'dist'),
assetModuleFilename: 'images/[hash][ext][query]',
},
module: {
rules: [
{
test: /\.png/,
type: 'asset/resource',
},
{
test: /\.html/,
type: 'asset/resource',
generator: {
filename: 'static/[hash][ext][query]',
},
},
],
},
};
import mainImage from './images/main.png';
img.src = mainImage; // '/dist/151cfcfa1bd74779aadb.png'
const path = require('path');
const svgToMiniDataURI = require('mini-svg-data-uri');
module.exports = {
entry: './src/index.js',
output: {
filename: 'main.js',
path: path.resolve(__dirname, 'dist'),
},
module: {
rules: [
{
test: /\.svg/,
type: 'asset/inline',
generator: {
dataUrl: content => {
content = content.toString();
return svgToMiniDataURI(content);
},
},
},
],
},
};
import metroMap from './images/metro.svg';
block.style.background = `url(${metroMap})`;
// => url(...vc3ZnPgo=)
const path = require('path');
module.exports = {
entry: './src/index.js',
output: {
filename: 'main.js',
path: path.resolve(__dirname, 'dist'),
},
module: {
rules: [
{
test: /\.txt/,
type: 'asset/source',
},
],
},
};
import exampleText from './example.txt';
block.textContent = exampleText; // 'Hello world'
const config = {
rules: [
{
loader: 'thread-loader',
// loaders with equal options will share worker pools
options: {
// the number of spawned workers, defaults to (number of cpus - 1) or
// fallback to 1 when require('os').cpus() is undefined
workers: 2,
// number of jobs a worker processes in parallel
// defaults to 20
workerParallelJobs: 50,
// additional node.js arguments
workerNodeArgs: ['--max-old-space-size=1024'],
// Allow to respawn a dead worker pool
// respawning slows down the entire compilation
// and should be set to false for development
poolRespawn: false,
// timeout for killing the worker processes when idle
// defaults to 500 (ms)
// can be set to Infinity for watching builds to keep workers alive
poolTimeout: 2000,
// number of jobs the poll distributes to the workers
// defaults to 200
// decrease of less efficient but more fair distribution
poolParallelJobs: 50,
// name of the pool
// can be used to create different pools with elseWise identical options
name: 'my-pool',
},
},
// your expensive loader (e.g babel-loader)
],
};
const threadLoader = require('thread-loader');
threadLoader.warmup(
{
// pool options, like passed to loader options
// must match loader options to boot the correct pool
},
[
// modules to load
// can be any module, i. e.
'babel-loader',
'babel-preset-es2015',
'sass-loader',
]
);
npm i -D worker-loader
module.exports = {
module: {
rules: [
{
test: /\.worker\.js$/,
use: { loader: 'worker-loader' },
},
],
},
};
{
"externals": {
"moment": "window.moment",
"antd": "window.antd",
"lodash": "window._",
"react": "window.React",
"react-dom": "window.ReactDOM"
}
}
const config = new webpack.optimize.CommonsChunkPlugin({
name: string, // or
names: [string],
// The chunk name of the commons chunk.
// An existing chunk can be selected by passing a name of an existing chunk.
// If an array of strings is passed this is equal to
// invoking the plugin multiple times for each chunk name.
// If omitted and `options.async` or `options.children`
// is set all chunks are used, otherwise `options.filename`
// is used as chunk name.
// When using `options.async` to create common chunks
// from other async chunks you must specify an entry-point
// chunk name here instead of omitting the `option.name`.
filename: string,
// The filename template for the commons chunk.
// Can contain the same placeholders as `output.filename`.
// If omitted the original filename is not modified
// (usually `output.filename` or `output.chunkFilename`).
// This option is not permitted if you're using `options.async` as well,
// see below for more details.
minChunks: number | Infinity | fn,
// (module, count) => boolean,
// The minimum number of chunks which need to contain a module
// before it's moved into the commons chunk.
// The number must be greater than or equal 2
// and lower than or equal to the number of chunks.
// Passing `Infinity` creates the commons chunk, but moves no modules into it.
// By providing a `function` you can add custom logic.
// (Defaults to the number of chunks)
chunks: [string],
// Select the source chunks by chunk names.
// The chunk must be a child of the commons chunk.
// If omitted all entry chunks are selected.
children: boolean,
// If `true` all children of the commons chunk are selected
deepChildren: boolean,
// If `true` all descendants of the commons chunk are selected
async: boolean | string,
// If `true` a new async commons chunk is created
// as child of `options.name` and sibling of `options.chunks`.
// It is loaded in parallel with `options.chunks`.
// Instead of using `option.filename`,
// it is possible to change the name of the output file by providing
// the desired string here instead of `true`.
minSize: number,
// Minimum size of all common module before a commons chunk is created.
});
const TerserPlugin = require('terser-webpack-plugin');
const MiniCssExtractPlugin = require('mini-css-extract-plugin');
const CssMinimizerPlugin = require('css-minimizer-webpack-plugin');
const isEnvProduction = process.env.NODE_ENV === 'production';
const isEnvProductionProfile =
isEnvProduction && process.argv.includes('--profile');
const shouldUseSourceMap = process.env.GENERATE_SOURCEMAP !== 'false';
module.exports = {
module: {
rules: [
{
test: /\.(js|mjs|jsx|ts|tsx)$/,
include: path.resolve('src'),
use: [
'thread-loader',
{
loader: require.resolve('babel-loader'),
},
],
},
{
test: /.s?css$/,
use: [MiniCssExtractPlugin.loader, 'css-loader', 'sass-loader'],
},
],
},
optimization: {
minimize: true,
minimizer: [
new TerserPlugin({
parallel: true,
terserOptions: {
parse: {
ecma: 8,
},
compress: {
ecma: 5,
warnings: false,
drop_console: true,
comparisons: false,
inline: 2,
},
mangle: {
safari10: true,
},
keep_classnames: isEnvProductionProfile,
keep_fnames: isEnvProductionProfile,
output: {
ecma: 5,
comments: false,
ascii_only: true,
},
},
}),
new CssMinimizerPlugin(),
],
},
};
require.ensure([], () => {});
.import
.React.Suspense
and React.lazy
.vendor.[hash].chunk.js
:
splitting vendor and application code
is to enable long term caching techniques
Since vendor code tends to change less often than the actual application code,
browser will be able to cache them separately,
and won't re-download them each time the app code changes.Split chunks configuration:
true
/ false
.true
/ false
.module.exports = {
optimization: {
runtimeChunk: true,
splitChunks: {
chunks: 'async',
minSize: 30000,
maxSize: 200000,
minChunks: 1,
maxAsyncRequests: 6,
maxInitialRequests: 4,
automaticNameDelimiter: '-',
cacheGroups: {
vendors: {
name: 'chunk-vendors',
priority: -10,
chunks: 'initial',
test: /[\\/]node_modules[\\/]/,
},
common: {
name: 'chunk-common',
priority: -20,
chunks: 'initial',
minChunks: 2,
reuseExistingChunk: true,
},
element: {
name: 'element-ui',
priority: 0,
chunks: 'all',
test: /[\\/]element-ui[\\/]/,
},
api: {
name: 'api',
priority: 0,
test: /[\\/]api[\\/]/,
},
subApi: {
name: 'subApi',
priority: 10,
minChunks: 2,
test: /[\\/]api[\\/]subApi[\\/]/,
},
mixin: {
name: 'mixin',
priority: 0,
test: /[\\/]mixin[\\/]/,
},
},
},
},
};
Live code inclusion (AST analysis) + dead code elimination:
CommonJS
形式.
@babel/preset-env
: always { "modules": false }
.modules
类型的转换.esm
模块进行 tree shaking.rollup
(ES6 module export + code flow static analysis),
并且提供 ES6 module 的版本, 入口文件地址设置到 package.json
的 module 字段.export { foo, bar }
better than export default alls
.Tree Shaking
的包: lodash-es
or babel-plugin-lodash
.const config = new HardSourceWebpackPlugin({
// Either an absolute path or relative to webpack options.context.
cacheDirectory: 'node_modules/.cache/hard-source/[confighash]',
// Either a string of object hash function given a webpack config.
configHash: webpackConfig => {
// node-object-hash on npm can be used to build this.
return require('node-object-hash')({ sort: false }).hash(webpackConfig);
},
// Either false, a string, an object, or a project hashing function.
environmentHash: {
root: process.cwd(),
directories: [],
files: ['package-lock.json', 'yarn.lock'],
},
// An object.
info: {
// 'none' or 'test'.
mode: 'none',
// 'debug', 'log', 'info', 'warn', or 'error'.
level: 'debug',
},
// Clean up large, old caches automatically.
cachePrune: {
// Caches younger than `maxAge` are not considered for deletion. They must
// be at least this (default: 2 days) old in milliseconds.
maxAge: 2 * 24 * 60 * 60 * 1000,
// All caches together must be larger than `sizeThreshold` before any
// caches will be deleted. Together they must be at least this
// (default: 50 MB) big in bytes.
sizeThreshold: 50 * 1024 * 1024,
},
});
Webpack 5
const config = {
cache: {
type: 'memory',
},
};
const config = {
cache: {
type: 'filesystem',
buildDependencies: {
config: [__filename],
},
},
};
const SpeedMeasurePlugin = require('speed-measure-webpack-plugin');
const smp = new SpeedMeasurePlugin();
const webpackConfig = smp.wrap({
plugins: [new MyPlugin(), new MyOtherPlugin()],
});
npx webpack --mode production --profile --json > stats.json
{
"husky": {
"hooks": {
"commit-msg": "commitlint -e -V",
"pre-commit": "lint-staged"
}
},
"lint-staged": {
"src/**/*.{js,jsx, ts, tsx}": ["eslint --fix", "git add"],
"src/**/*.{css, scss}": ["stylelint --fix", "git add"]
}
}
Webpack 5 support out of box cache.
module.exports = {
plugins: [
function () {
this.hooks.done.tap('done', stats => {
if (
stats.compilation.errors &&
stats.compilation.errors.length &&
!process.argv.includes('--watch')
) {
// Process build errors.
process.exit(1);
}
});
},
],
};
const childProcess = require('child_process');
const HtmlWebpackPlugin = require('html-webpack-plugin');
const branch = childProcess
.execSync('git rev-parse --abbrev-ref HEAD')
.toString()
.replace(/\s+/, '');
const version = branch.split('/')[1];
const scripts = [
'https://cdn.bootcss.com/react-dom/16.9.0-rc.0/umd/react-dom.production.min.js',
'https://cdn.bootcss.com/react/16.9.0/umd/react.production.min.js',
];
class HotLoad {
apply(compiler) {
compiler.hooks.beforeRun.tap('UpdateVersion', compilation => {
compilation.options.output.publicPath = `./${version}/`;
});
compiler.hooks.compilation.tap('HotLoadPlugin', compilation => {
HtmlWebpackPlugin.getHooks(compilation).alterAssetTags.tapAsync(
'HotLoadPlugin',
(data, cb) => {
scripts.forEach(src => [
data.assetTags.scripts.unshift({
tagName: 'script',
voidTag: false,
attributes: { src },
}),
]);
cb(null, data);
}
);
});
}
}
module.exports = HotLoad;
Typed webpack plugin from laravel-mix/
:
const readline = require('readline');
const _ = require('lodash');
const chalk = require('chalk');
const Table = require('cli-table3');
const stripAnsi = require('strip-ansi');
const { formatSize } = require('webpack/lib/SizeFormatHelpers');
const { version } = require('../../package.json');
/**
* @typedef {object} BuildOutputOptions
* @property {boolean} clearConsole
* @property {boolean} showRelated
**/
/**
* @typedef {object} StatsAsset
* @property {string} name
* @property {number} size
* @property {StatsAsset[]|{}} related
**/
/**
* @typedef {object} StatsData
* @property {StatsAsset[]} assets
**/
class BuildOutputPlugin {
/**
*
* @param {BuildOutputOptions} options
*/
constructor(options) {
this.options = options;
this.patched = false;
}
/**
* Apply the plugin.
*
* @param {import('webpack').Compiler} compiler
*/
apply(compiler) {
if (process.env.NODE_ENV === 'test') {
return;
}
compiler.hooks.done.tap('BuildOutputPlugin', stats => {
if (stats.hasErrors()) {
return false;
}
if (this.options.clearConsole) {
this.clearConsole();
}
const data = stats.toJson({
assets: true,
builtAt: true,
hash: true,
performance: true,
relatedAssets: this.options.showRelated,
});
this.heading(`Laravel Mix v${version}`);
console.log(
chalk.green.bold(`✔ Compiled Successfully in ${data.time}ms`)
);
if (data.assets.length) {
console.log(this.statsTable(data));
}
});
}
/**
* Print a block section heading.
*
* @param {string} text
*/
heading(text) {
console.log();
console.log(chalk.bgBlue.white.bold(this.section(text)));
console.log();
}
/**
* Create a block section.
*
* @param {string} text
*/
section(text) {
const padLength = 3;
const padding = ' '.repeat(padLength);
text = `${padding}${text}${padding}`;
const line = ' '.repeat(text.length);
return `${line}\n${text}\n${line}`;
}
/**
* Generate the stats table.
*
* @param {StatsData} data
* @returns {string}
*/
statsTable(data) {
const assets = this.sortAssets(data);
const table = new Table({
head: [chalk.bold('File'), chalk.bold('Size')],
colWidths: [35],
colAligns: ['right'],
style: {
head: [],
compact: true,
},
});
for (const asset of assets) {
table.push([chalk.green(asset.name), formatSize(asset.size)]);
}
this.extendTableWidth(table);
this.monkeyPatchTruncate();
return table.toString();
}
/**
*
* @param {StatsData} data
*/
sortAssets(data) {
let assets = data.assets;
assets = _.flatMap(assets, asset => [
asset,
...(Array.isArray(asset.related) ? asset.related : []),
]);
assets = _.orderBy(assets, ['name', 'size'], ['asc', 'asc']);
return assets;
}
/**
* Clear the entire screen.
*/
clearConsole() {
const blank = '\n'.repeat(process.stdout.rows);
console.log(blank);
readline.cursorTo(process.stdout, 0, 0);
readline.clearScreenDown(process.stdout);
}
/**
* Extend the width of the table
*
* Currently only increases the file column size
*
* @param {import('cli-table3').Table} table
* @param {number|null} targetWidth
* @param {number} maxWidth
*/
extendTableWidth(table, targetWidth = null, maxWidth = Infinity) {
targetWidth = targetWidth === null ? process.stdout.columns : targetWidth;
if (!targetWidth) {
return;
}
const tableWidth = this.calculateTableWidth(table);
const fileColIncrease = Math.min(
targetWidth - tableWidth,
maxWidth - tableWidth
);
if (fileColIncrease <= 0) {
return;
}
// @ts-expect-error Should error
table.options.colWidths[0] += fileColIncrease;
}
monkeyPatchTruncate() {
if (this.patched) {
return;
}
this.patched = true;
// @ts-expect-error Should error
const utils = require('cli-table3/src/utils');
const oldTruncate = utils.truncate;
/**
*
* @param {string} str
* @param {number} desiredLength
* @param {string} truncateChar
*/
utils.truncate = (str, desiredLength, truncateChar) => {
if (stripAnsi(str).length > desiredLength) {
str = `…${str.substr(-desiredLength + 2)}`;
}
return oldTruncate(str, desiredLength, truncateChar);
};
}
/**
* Calculate the width of the CLI Table
*
* `table.width` does not report the correct width
* because it includes ANSI control characters
*
* @internal
* @param {import('cli-table3').Table} table
*/
calculateTableWidth(table) {
const firstRow = table.toString().split('\n')[0];
return stripAnsi(firstRow).length;
}
}
module.exports = BuildOutputPlugin;
Make sure there's no webpack deprecation warnings.
node --trace-deprecation node_modules/webpack/bin/webpack.js