Live share
WHO I AM
dmtrKovalenko
@dmtrKovalenko
Пирамида
Для кого?
Для кого?
Для кого?
Для кого?
Unit testing
Для кого?
Integration testing
Для кого?
e2e testing
e2e
Integration
Unit
Вопрос
?=
Давайте разбираться!
Где граница?
e2e
Integration
Unit
непонятно
Rest API
e2e
Front-end
Cypress.Commands.add('login', () => {
cy.visit('/login')
cy.get('#email').type('admin@cypress.io')
cy.get('#password').type('security')
cy.get('#submit-btn').click()
})
cy.login()
Cypress.Commands.add('login', () => {
cy.request('/testRoutes/loginAsAdmin')
})
Где граница?
e2e
Integration
Unit
Назад
В
Прошлое
Мы тут
React
Angular 1.0
Vue
JQuery
2006
2012
2013
2014
2019
Timeline
Testing Pyramid
2008
Google it
https://benhutchison.wordpress.com/2008/03/20/automated-testing-2-the-test-pyramid/
Почему
- Долгаааа
- Сложнааа
(запускать и писать)
(писать и поддерживать)
не e2e?
чо
так
сложно долго
Мы тут
React
Angular 1.0
Vue
JQuery
2006
2012
2013
2014
2019
Timeline
Testing 🔼
2008
2004
Selenium
Если у вас нету тестов
То билд у вас не упадет
Если вы не запускаете тесты
То у вас нету
тестов
e2e
is painfull
Build Time: 42:14
Мы тут
React
Angular 1.0
Vue
JQuery
2006
2012
2013
2014
2019
Timeline
Testing 🔼
2008
2004
Selenium
version: 2
jobs:
build:
docker:
- image: cypress/base:10
steps:
- checkout
- restore_cache:
name: Restore Yarn Package Cache
keys:
- yarn-packages-{{ checksum "yarn.lock" }}
- run:
name: Install Dependencies
command: yarn install --frozen-lockfile
- save_cache:
name: Save Yarn Package Cache
key: yarn-packages-{{ checksum "yarn.lock" }}
paths:
- ~/.cache
- run: cypress run --record
Проще не бывает
version: 2
jobs:
build:
docker:
- image: cypress/base:10
parallelism: 15
steps:
- checkout
- restore_cache:
name: Restore Yarn Package Cache
keys:
- yarn-packages-{{ checksum "yarn.lock" }}
- run:
name: Install Dependencies
command: yarn install --frozen-lockfile
- save_cache:
name: Save Yarn Package Cache
key: yarn-packages-{{ checksum "yarn.lock" }}
paths:
- ~/.cache
- run: cypress run --record
Еще и быстро
e2e
e2e
Integration
Unit
Ну-ка напиши тест
import db from './db'
const router = express.Router();
router.get('/users', (req, res) => {
db.getAllUsers().then(res.json)
})
Dependency Injection
const router = express.Router();
router.get('/users', (req, res) => {
req.db.getAllUsers().then(res.json)
})
Я
Unit tests
TRIGGERED
Rest API
Frontend
GraphQL
Overmocking
import { fetchFromApi } from '~/fetcher'
function fetchUsers() {
return fetchFromApi('/users')
}
it('Should fetch users', async () => {
jest.mock(fetchFromApi, () => Promise.resolve([{ id: 1 }]))
const result = await fetchUsers()
expect(result).toEqual([{ id: 1 }])
})
И правда
Чем больше тесты знают про реализацию, тем...
Но ведь e2e тоже не огонь
/root/div/descendant::*[not (@class='bad')] //h1/following-sibling::ul[1]
<div data-test-id="user-list-name"/>
Убираем хрупкость
e2e
One more thing
Зачем мы пишем тесты?
Зачем?
Надежность
e2e
Integration
Unit
Надежность
Скорость
>.< Бесполезные тесты
UI components unit testing
it("renders three <Foo /> components", () => {
const wrapper = shallow(
<MyComponent>
<Foo />
<Foo />
<Foo />
</MyComponent>
);
expect(wrapper.find(Foo)).to.have.lengthOf(3);
});
const Component = () => (
<MyComponent>
<Foo />
<Foo />
<Foo />
</MyComponent>
)
const jsx = React.createElement
const Component = () => jsx(
MyComponent,
null,
jsx(Foo, null),
jsx(Foo, null),
jsx(Foo, null)
);
===
UI components unit testing
it("renders three <Foo /> components", () => {
const wrapper = shallow(jsx(
MyComponent,
null,
jsx(Foo, null),
jsx(Foo, null),
jsx(Foo, null)
));
const arguments = jsx.mock.calls[0]
expect(arguments.filter((element) => element.type === Foo).to.have.lengthOf(3)
});
Text
Тестирование Реализации
vs
export function calculateUsagePeriods(
currentUsagePeriodStartsAt,
currentUsagePeriodEndsAt,
organizationCreatedAt,
n = 6
) {
if (!currentUsagePeriodStartsAt) return []
const usagePeriods = []
let startTime = moment.utc(currentUsagePeriodStartsAt)
let endTime = moment.utc(currentUsagePeriodEndsAt)
for (let i = 0; i < n; i++) {
if (endTime.isBefore(organizationCreatedAt)) {
break
}
usagePeriods.push({
startTime: startTime.clone(),
endTime: endTime.clone(),
})
startTime.subtract(1, 'month')
endTime.subtract(1, 'month')
}
return usagePeriods
}
test("Should calculate usage periods when n = 5", () => {
const result = calculateUsagePeriods(new Date(), addDays(new Date(), 5), 5)
expect(result).toEqual([...])
})
test.each`
startsAt | endsAt | n | expected
${new Date()} | ${new Date()} | ${10} | ${[15, 16]}
${new Date()} | ${addDays(new Date(), 10)} | ${6} | ${[15, 16]}
${new Date()} | ${addDays(new Date(), -10)} | ${15} | ${[15, 16]}
`(
"Should calculate usage periods when startsAt = $startsAt, endsAt = $endsAt n = $n",
({ startsAt, endsAt, n, expected }) => {
expect(calculateUsagePeriods(startsAt, endsAt, n)).toEqual(expected);
}
);
Unit Tests
But useless
Users ❤️ Quality app
Business
Dev ❤️ Coding
Dev ❤️ Create quality apps
❤️ Happy users
e2e
Integration
Unit
===
Illuminati confirmed
Links
Спасибо за внимание
Unit tests are useless
By dkovalenko
Unit tests are useless
- 1,970