perfect working environment
in 5 days
Who am i?
const ME = Object.freeze({
name: 'Sergey Bolshov',
position: 'Senior Front-end Software Engineer',
workPlace: 'Allegro',
email 'bolszow@gmail.com',
githubUsername: '@bigsergey'
});
DAY 1
First day at work
function additional(a, b) {
return
{
result: a+b
};
}
console.log(additonal(1, 2));
Ups, typo
data:image/s3,"s3://crabby-images/16eb1/16eb12b1f4818690d72eb023e14a2cf99fcb1213" alt=""
function additional(a, b) {
return
{
result: a+b
};
}
-console.log(additonal(1, 2));
+console.log(additional(1, 2));
function additional(a, b) {
return
{
result: a+b
};
}
console.log(additional(1, 2));
data:image/s3,"s3://crabby-images/14a24/14a24f8ffad693320a760a188a6c130e2719c783" alt=""
Bad new line
function additional(a, b) {
- return
- {
+ return {
result: a+b
};
}
console.log(additional(1, 2));
function additional(a, b) {
return {
result: a+b
};
}
console.log(additional(1, 2));
data:image/s3,"s3://crabby-images/52aab/52aab14f38561c487b87975008cb9eb5a2261362" alt=""
data:image/s3,"s3://crabby-images/e1ed3/e1ed3965e93fdbe68c33136dca72b3396d84ba13" alt=""
Source: https://steamcommunity.com
data:image/s3,"s3://crabby-images/2133b/2133b6ae3a622fb73d2b3cc9f4ae51dbf61c1292" alt=""
Source: https://conservativememes.com
Eslint to rescue
data:image/s3,"s3://crabby-images/ab253/ab2538a50b1a8a8b8d3b411f0bc9626e70d04f31" alt=""
eslint profits
- Eliminate errors
- Consistent code style
- Good practice/'Clean code'
- Easy to use
- IDE/Editors Support
- Pretteer integration
data:image/s3,"s3://crabby-images/1c8e1/1c8e14af5202f432bf060486d3671a2a82b8ca5f" alt=""
bonus tips
- Enable ESLint in editor
- Keep config updated
- Check for new plugins/configs
- Write your own rules
- Use lint-staged
lint-staged
{
"name": "your-project",
"version": "1.0.0",
"devDependencies": {
"husky": "2.2.0",
"lint-staged": "8.1.6"
},
"lint-staged": {
"*.js": [
"eslint --fix",
"git add"
]
},
"husky": {
"hooks": {
"pre-commit": "lint-staged"
}
}
}
lint-staged
{
"name": "your-project",
"version": "1.0.0",
"devDependencies": {
"husky": "2.2.0",
"lint-staged": "8.1.6"
},
"lint-staged": {
"*.js": [
"eslint --fix",
"git add"
]
},
"husky": {
"hooks": {
"pre-commit": "lint-staged"
}
}
}
stylelint
A mighty, modern linter that helps you avoid errors and enforce conventions in your styles.
data:image/s3,"s3://crabby-images/9cfc3/9cfc35db26df5a07d3f94e9724ead86bf9d2b438" alt=""
data:image/s3,"s3://crabby-images/47101/47101db8adc3080eb7e699d5a4e0581e82927ede" alt=""
Source: https://stylelint.io/
lint-staged + stylelint
{
"name": "your-project",
"version": "1.0.0",
"devDependencies": {
"husky": "2.2.0",
"lint-staged": "8.1.6",
"stylelint": "10.0.1"
},
"lint-staged": {
"*.js": [
"eslint --fix",
"git add"
],
"*.{css,pcss}": [
"stylelint --fix",
"git add"
]
},
"husky": {
"hooks": {
"pre-commit": "lint-staged"
}
}
}
data:image/s3,"s3://crabby-images/354c5/354c57657ee26ea81e1bf8e87b7357046ee836fd" alt=""
data:image/s3,"s3://crabby-images/82700/82700d9318b95e73b4f26be21b0bb57729ea05f3" alt=""
lint-staged + ESLINT + stylelint
useful configs
- stylelint-config-standard
- stylelint-config-rational-order (like CSScomb)
- stylelint-declaration-use-variable
- stylelint-high-performance-animation
- eslint-config-airbnb-base
- eslint-plugin-jsx-a11y
- ...
at the end of the DAY 1
data:image/s3,"s3://crabby-images/99408/9940805c0126beaa84aa406ad298915a3dea8ca1" alt=""
data:image/s3,"s3://crabby-images/66410/66410b01a31a81f5c33ef5f226d6566131fdf527" alt=""
data:image/s3,"s3://crabby-images/9cfc3/9cfc35db26df5a07d3f94e9724ead86bf9d2b438" alt=""
+
+
=
data:image/s3,"s3://crabby-images/eea5f/eea5f6a27e8b3b8387d25255634a99124c751f02" alt=""
ESLint
Prettier
Stylelint
day 2
Just Describe it
function add(a, b) {
return {
result: a + b
};
}
data:image/s3,"s3://crabby-images/9b957/9b957a8584694f450864cb6d744eb3907f26fb0d" alt=""
$ npm install mocha chai --save-dev
{
"name": "your-project",
"version": "1.0.0",
"scripts": {
"test": "mocha"
}
}
import { expect } from 'chai';
import add from '../add';
describe('add function', () => {
it('should return object', () => {
expect(add()).to.be.an('object');
});
it('should have result property', () => {
expect(add()).to.have.property('result');
});
it('should add 2 positive numbers together', () => {
expect(add(2, 2)).to.deep.equal({ result: 4 });
});
});
data:image/s3,"s3://crabby-images/1a1e1/1a1e1d0536a9b880cadaa668022a53d9e2e4c91b" alt=""
WHich tool should i use?
js testing tools
- Test runners (Karma, Jasmine, Jest, TestCafe, Cypress, Protractor)
- Testing structure (Mocha, Jasmine, Jest, Cucumber, TestCafe, Cypress, Protractor)
- Assertion functions (Chai, Jasmine, Jest, Unexpected, TestCafe, Cypress, Protractor)
- Generate and display test progress and results (Mocha, Jasmine, Jest, Karma, TestCafe, Cypress)
- Mocks, spies, and stubs (Sinon, Jasmine, enzyme, Jest, testdouble)
- Generate code coverage (Istanbul, Jest, Blanket)
Source: State of Javascript 2019
Testing, testing… Is this thing on? Sorry, I Jest, but I would never want to make a Mocha-ry of this section. That would be Jasmine (“just mean”? get it?), and that's bad Karma.
data:image/s3,"s3://crabby-images/5ead6/5ead677491d3c2c4c0f9f4fd401c31e0fbc9683b" alt=""
Salary breakdown
Source: State of Javascript 2018
data:image/s3,"s3://crabby-images/9d8a6/9d8a63fcc8c20444d2573296d0ad06ad07510ce1" alt=""
Salary breakdown
Source: State of Javascript 2018
data:image/s3,"s3://crabby-images/5157f/5157f2ac5f5d01572585c011cf581ccd6e435d3c" alt=""
data:image/s3,"s3://crabby-images/2c84d/2c84db82cac1a6694a0a59fc4c66a29e9421baac" alt=""
data:image/s3,"s3://crabby-images/71c3c/71c3c52896ed6cd9b3e63e409a895faa4cba0196" alt=""
Source: Jasmine and Aladdin
WHICH TOOL SHOULD I USE?
a tool does not matter,
writing and maintaining tests is important
at the end of the DAY 2
data:image/s3,"s3://crabby-images/8adfd/8adfd7d07117e75d69a169ad2272c6e140fd67db" alt=""
data:image/s3,"s3://crabby-images/fa0fd/fa0fdcff09b1f29c28cd96cf9a92e4a1e84d38d5" alt=""
data:image/s3,"s3://crabby-images/ca552/ca5529f9e2e14270bc0682115127e3b61c8d3a48" alt=""
+
+
=
data:image/s3,"s3://crabby-images/60311/60311d486a83ce21d1e599221a1d69d94fb1b4fa" alt=""
Sinon
Day 3
first blood
New feature request
data:image/s3,"s3://crabby-images/f57bf/f57bfd5fb4cf5445e37eb3408dec3bee7f014a81" alt=""
data:image/s3,"s3://crabby-images/6a7ac/6a7ac9db05834169e8b45be12fdf78983624e90e" alt=""
data:image/s3,"s3://crabby-images/3f9d3/3f9d33ddb702fbad8cad6e37b1846b2f638c0f8b" alt=""
Source: https://giphy.com
Fire in production!
data:image/s3,"s3://crabby-images/47329/47329a1010250362015d90d848990eae7d0118d0" alt=""
data:image/s3,"s3://crabby-images/7cbe1/7cbe1fa44f570292431e2c88aeb184604bed07d2" alt=""
data:image/s3,"s3://crabby-images/60f72/60f72c403082e7652cf8e6402697e8b443d4ccb8" alt=""
Source: https://giphy.com
data:image/s3,"s3://crabby-images/1790a/1790aba32a72d4ac192574d4d65824fd25b38ea7" alt=""
e2e tests to rescue!
import { Selector } from 'testcafe';
const dashboard = Selector('#dashboard');
fixture`Authentication`
.page`http://example.com/login`;
test('should redirect to dashboard after login', async (t) => {
await t
.typeText('#login', 'TestUser')
.typeText('#password', 'testpass')
.click('#login')
.expect(dashboard.exists)
.ok();
});
// testcafe puppeteer 'path/to/test/file.js'
$ npm i testcafe testcafe-browser-provider-puppeteer -D
data:image/s3,"s3://crabby-images/217b1/217b1ec1b1c31e84c54748decf7d2d29223b4ae9" alt=""
e2e tests tools
- ... and more other tools
at the end of the DAY 3
+
=
data:image/s3,"s3://crabby-images/fffd3/fffd3049b311cc94b9b464e3acc86ded412dca55" alt=""
data:image/s3,"s3://crabby-images/c3ca5/c3ca5a4d52d9c1a3bf4f9ae865b68ba7e3c79d6f" alt=""
data:image/s3,"s3://crabby-images/56822/568224aa663b06ad9710a0360889481cf2e861dd" alt=""
DAY 4
in hell
what's easy to break, hard to fix?
data:image/s3,"s3://crabby-images/40d06/40d06bbfa43731179d53753928eca2c2fe90b58a" alt=""
data:image/s3,"s3://crabby-images/e5deb/e5deb96b1159667fe7b007735015fc6cb6fbfa25" alt=""
data:image/s3,"s3://crabby-images/d1e67/d1e67a3318e62b3fda4f3bab4298e375462ee282" alt=""
data:image/s3,"s3://crabby-images/2044b/2044b48c0aabff10a362690edfdbd10d9ec08554" alt=""
data:image/s3,"s3://crabby-images/ade95/ade956cd11e2f9b13978d788830ab6e7ae7cbbec" alt=""
Visual Regression Tools are used to compare your site to its previous versions visually by using image comparison techniques.
data:image/s3,"s3://crabby-images/e6b9d/e6b9d1e0c22c3fec7fef762f1654eefb55473536" alt=""
data:image/s3,"s3://crabby-images/892f5/892f540b8ffcfe7f0ead2269ba2709b2a1208665" alt=""
visual regression tools
- Applitools (Selenium, not free)
- Percy.io (Selenium, not free)
- BackstopJS (Puppeteer, free, 4.8k ⭐️)
- Gemini (Selenium, free, 1.5k ⭐️)
- Differencify (Puppeteer, free, 475 ⭐️)
Wraith (Phantom JS)
More info here: "Awesome Visual Regression Testing"
visual regression tools challenges
- test execution environment
- different version of browsers
- different operation systems
- different screen sizes
- different DPI
- ...
- storing pictures in the repository
- git-lfs
- external storage
- ...
at the end of the DAY 4
some visual regression tool
=
Icon made by Freepik from www.flaticon.com
day 4 overtime
data:image/s3,"s3://crabby-images/ea8a4/ea8a4b136ace0ab90e16d9bf2a011730d056fc3c" alt=""
Source: memegenerator.net
Dom nodes number
data:image/s3,"s3://crabby-images/615ab/615aba017a51bc1f08c96ba4db17aab720ee17a0" alt=""
Deploy
data:image/s3,"s3://crabby-images/43ea7/43ea7c54f9adeb07216f18a8fd1e319019a510fd" alt=""
data:image/s3,"s3://crabby-images/b98c6/b98c6de5c1e2927f0b0986fb1b89a12d3c9492db" alt=""
- Jest for React,
- Ava for React,
- Snap-shot-it for Mocha and BDD test runners.
data:image/s3,"s3://crabby-images/4b578/4b578cdeb573d293999b321b3a0d57d34624f55b" alt=""
snapshot testing
Source: Jest snapshot testing
bundlesize
$ npm install bundlesize --save-dev
$
{
"name": "your-project",
"version": "1.0.0",
"bundlesize": [{
"path": "./dist/main.css",
"maxSize": "3 kB"
},
{
"path": "./dist/chunk.js",
"maxSize": "10 kB"
}
]
}
at the end of the DAY 4 + overtime
visual regression
snapshot testing
data:image/s3,"s3://crabby-images/521bd/521bdc8178510268d09010e469eba0e3ea528b81" alt=""
Source: blog.debugme.eu
=
+
Day 5
save your fingers
data:image/s3,"s3://crabby-images/d685e/d685eadfcce90a1593781a9c7b7b0724d0fcd953" alt=""
// icon component
import { html } from 'escape-html-template-tag';
import * as styles from './index.css';
export default function renderIcon(iconUrl) {
return html`
<svg class="${styles.icon}">
<image xlink:href="${html.safe(iconUrl)}"></image>
</svg>
`;
}
// head component
import { html } from 'escape-html-template-tag';
import classnames from '../classnames';
import * as styles from './index.css';
export default function renderHead(title, isAlignRight) {
const headClassNames = classnames([
styles.head,
isAlignRight && styles.headRight,
]);
return html`
<span class="${headClassNames}">${title}</span>
`;
}
COde snippets
VScode extensions
GIT and command line aliases
# ...
[alias]
#LAZY VERSIONS OF BASIC COMMANDS
co = checkout
br = branch
ci = commit
st = status
#BETTER VERSIONS OF BASIC COMMANDS
purr = pull --rebase
puff = pull --ff-only
difff = diff --color-words
bbranch = branch -v
branches = branch -avvl
# ...
Source: git-kurwa
# Git aliases
alias gc="git commit"
alias ga="git add"
alias gr="git rm"
alias gp="git push"
alias gs="git status"
alias gpull="git pull"
alias gpullm="git pull origin master"
alias gpm="git push origin master"
alias clone="git clone"
Source: .bash_aliases
Oh My Zsh
data:image/s3,"s3://crabby-images/15d47/15d47ebf953ac0462ae047a4283d63b8e98fcb87" alt=""
Source: iterm2-solirized
itermocil
data:image/s3,"s3://crabby-images/36371/36371c1586b6890af9294ba9eba09b35127a5fff" alt=""
Source: Itermocil
NPM check updates
data:image/s3,"s3://crabby-images/33c31/33c315e7a455234385d95382be3301c658b84ce9" alt=""
Source: npm-check-updates
greenkeeper
Source: Greenkeeper
at the end of the week
data:image/s3,"s3://crabby-images/99408/9940805c0126beaa84aa406ad298915a3dea8ca1" alt=""
data:image/s3,"s3://crabby-images/66410/66410b01a31a81f5c33ef5f226d6566131fdf527" alt=""
data:image/s3,"s3://crabby-images/9cfc3/9cfc35db26df5a07d3f94e9724ead86bf9d2b438" alt=""
data:image/s3,"s3://crabby-images/8adfd/8adfd7d07117e75d69a169ad2272c6e140fd67db" alt=""
data:image/s3,"s3://crabby-images/fa0fd/fa0fdcff09b1f29c28cd96cf9a92e4a1e84d38d5" alt=""
data:image/s3,"s3://crabby-images/ca552/ca5529f9e2e14270bc0682115127e3b61c8d3a48" alt=""
Sinon
data:image/s3,"s3://crabby-images/fffd3/fffd3049b311cc94b9b464e3acc86ded412dca55" alt=""
data:image/s3,"s3://crabby-images/c3ca5/c3ca5a4d52d9c1a3bf4f9ae865b68ba7e3c79d6f" alt=""
VISUAL REGRESSION
SNAPSHOT
TESTS
data:image/s3,"s3://crabby-images/4f0d3/4f0d3ab5f5e2ffeed39a300dc678acc0dacdc00f" alt=""
data:image/s3,"s3://crabby-images/0a4d7/0a4d76df4de753a5f281ae72e42e3e8f193255cd" alt=""
data:image/s3,"s3://crabby-images/13f03/13f031a0c0f5a06d846c4cb79d2a45fa171416e1" alt=""
data:image/s3,"s3://crabby-images/c71b6/c71b69d2fcf8224b28dcef8321bc3406c3ba8359" alt=""
thank you
Perfect working environment in 5 days (remastered 2020)
By Sergey Bolshov
Perfect working environment in 5 days (remastered 2020)
- 978