- Главная
- Шоу-биз
Юлия КАЦУН
29 ноября 18:16
0
Фото: «1+1»
В этом сезоне победный кубок танцевального шоу канала «1+1» получили звезда сериала «Моя улюблена Страшко» Артур Логай и танцовщица Анна Карелина. Впрочем, пара была фаворитами как судей, так и зрителей с первых выпусков «Танцев со звездами». Только во время прямых эфиров удалось сделать 49 сложнейших поддержек, и это не считая репетиций.
Мы позвонили Артуру Логаю, чтобы поздравить и расспросить, как они шли к своей победе.
Сложно было даже понять, как просто стоять
— Артур, поздравляю! Успели чуточку отпраздновать или сразу эфиры, интервью?
— Да, успели немножечко отпраздновать сразу после окончания эфира со всей командой, которая создает это замечательное шоу.
— Помните наше интервью на старте проекта? Вы тогда загадали, что через два месяца будете держать с Аней в руках победный кубок. Вы не сомневались в себе ни дня?
— Мы никогда не были уверены, что победим именно мы. В разговорах, понятно, мы с вами гадали, желали, но внутренне всегда готовились прицельно неделя за неделей, от воскресенья к воскресенью. Вот такая была история. Подготовились, сделали, прошли – супер. Каждую неделю – та же схема. Не было такого: ах, сейчас мы прямо в финал зайдем. Нет.
Я сам себя всегда мысленно спрашивал: а если ты сейчас вылетишь, что будешь делать?! Но как-то все срослось.
— Что вам давалось сложнее всего в танцах? Возможно, были какие-то страхи? Мне кажется, все завидовали вашим поддержкам.
— Что могу сказать — я никому не завидовал. Что сложного? Для меня, как для человека, который танцами занялся с нуля, все сложно. Сложно было даже понять, как просто стоять. Потому что это не то же самое, как стоять на улице.
Очень много разных нюансов, было сложно, но интересно. Что-то не сразу получалось, но если со 150-й попытки все же получалось, становилось веселее и радостнее.
— Просто глядя на вас, создавалось впечатление, что вам все дается легко.
— Мы очень много времени уделяли танцам, в зале проводили в среднем по 10 и больше часов в зависимости от надобности. Пробовали сделать все так, как вы говорите, чтобы это выглядело легко.
Артур с Аней работали в зале в среднем по 10 часов в день. И – победили. Фото: «1+1»
Мне больше импонировала Оля Харлан
— Вы признавались, что хотите, чтобы победили или вы, или Оля Харлан. Костя Войтенко в этом треугольнике был третьим лишним?
— Ну я же не могу болеть за всех сразу, правильно?! Болеть можно только за одного. Мы свой выбор сделали в пользу себя, и если не мы, то Оля.
— Почему именно Оля? Чем она вам импонировала?
— Я очень уважаю ее труд, с ее именем ассоциируется Украина, на мировой арене Украину знают и благодаря Ольге Харлан. Она олимпийская чемпионка, это очень многого стоит. Я на первых эфирах читал какие-то комментарии в духе «Боже, а кто это», и мне так хотелось написать человеку: «А тебе не стыдно?!»
Благодаря ей в мире знают, что на карте есть страна под названием Украина.
И тот путь, который Оля прошла, как она старалась… Мне она больше импонировала. Но самая главная, наверное, ее привлекательная черта – это большое доброе сердце.
— Всегда есть несогласные, недовольные… Сейчас на странице проекта в Инстаграме, помимо похвалы вам, пишут, что финал несправедливый, что слили Войтенко, продвигают своих, что должна была победить Оля Харлан… Вы как относитесь к таким комментариям? Они не огорчают вкус победы?
— Всегда такое будут писать. Это же мнение людей, и, мне кажется, им просто стоило бы не писать гневные комментарии, а голосовать за тех, кого они хотели видеть победителем.
Я не читаю эти комментарии, по крайней мере сейчас. Я точно знаю, что всегда будут люди, которым что-то будет не нравиться. К сожалению, наверное, или к счастью, но всем я нравиться не могу.
Каждый имеет право на свою точку зрения, если она конструктивная.
— А вас хейтеры могут как-то зацепить или умеете абстрагироваться?
— Я учусь понемногу фильтровать информацию, которую хочу получать. Опять-таки, так будет всегда, всем не понравишься.
Мне, наверное, никогда не будет понятно, почему, если человек человеку не нравится, нельзя просто пройти мимо, а обязательно надо что-то сказать плохое. Это мне непонятно, но я учусь постепенно не обращать на это внимания.
Думаю, кубок поедет к Ане
— Вы передали Анне Карелиной подарок, который вас попросил выбрать сын. Сами выбирали, что ей подойдет?
— Аня носит браслет с подвесками, кулончиками, и это был кулон на этот браслет. Я отправил Герману на утверждение: показал – ему понравилось, он ответил, что подходит.
— Он сам это решил?
— Герман еще с репетиции, когда мы делали семейный номер, много расспрашивал об Ане, ему понравилось танцевать, работать на сцене. Аня сказала, что он способный, может заниматься танцами. Вот как-то так и закружилось.
— Место для кубка дома нашли?
— Думаю, кубок поедет к Ане, если она захочет его забрать. Пока что он у меня, я обещал сыновьям показать, а потом отдам Ане.
Подписывайтесь на нас в соц. сетях
«Тесты — это лишняя работа», «тесты писать необязательно» — такие мнения часто можно услышать в разговорах о тестировании. В этой статье мы постараемся развеять этот миф и рассмотрим плюсы тестирования и минусы его отсутствия.
Тесты делают код более прочным и живучим. Одновременно с этим тесты — это отличная документация, которая не врёт и не устаревает. Также тесты можно использовать как инструмент разработки программы.
Для тестов нужно закладывать больше времени на разработку, это правда. Но время, потраченное в начале работы над проектом, окупится в дальнейшем.
Тест — это код, который проверяет предположения о работе другого кода.
Представим, что у нас есть функция add
, которая складывает одно число с другим:
function add(a, b) { return a + b;}
function add(a, b) { return a + b; }
Мы предполагаем, что функция прибавляет аргумент b
к аргументу a
и возвращает нам результат. Мы можем проверить это, вызвав её:
const result = add(10, 5);// result === 15
const result = add(10, 5);
Но что будет, если мы передадим не два числа, а одно? А если передадим не числа? Или функция за время жизни проекта изменится? Чтобы проверить такие предположения, мы пишем тесты.
Самый простой тест, который мы можем написать — ручной. Сравним руками результат работы функции и ожидаемое значение:
function testAdd() { const result = add(10, 5); const expected = 15; console.assert( result === expected, `The result ${result} doesn't match the expected value ${expected}.` );}
function testAdd() { const result = add(10, 5); const expected = 15; console.assert( result === expected, `The result ${result} doesn't match the expected value ${expected}.` ); }
При запуске функции testAdd
она проверит, что вернёт функция add
. Если результат не будет соответствовать ожиданию, консоль покажет ошибку.
Конечно, такой тест никуда не годится ?
- Ему сильно не хватает описания – как понять, что именно мы проверяем?
- Не хватает выразительности и лаконичности — сравнивать значения и выбрасывать ошибки руками не круто.
- Не хватает интерактивности — чтобы перезапустить тест, нужно запустить функцию заново.
В идеале хотелось бы, чтобы при изменении теста он перезапускался сам. Чтобы было место для описания того, что мы тестируем. Специально для этого придумали инструменты для тестирования.
Инструментов для тестирования много. Чтобы подобрать подходящий, нам надо определиться, какие тесты мы хотим писать. Подробнее о видах тестирования мы поговорим в конце статьи, а пока что посмотрим на самый часто используемый инструмент — Jest.
Попробуем, используя его, переписать тест нашей функции add
:
describe('When given 2 numbers', () => { it('returns the sum of those 2 numbers', () => { const result = add(10, 5); const expected = 15; expect(result).toEqual(expected); })})
describe('When given 2 numbers', () => { it('returns the sum of those 2 numbers', () => { const result = add(10, 5); const expected = 15; expect(result).toEqual(expected); }) })
Разберём по строкам:
- На первой строке мы указываем описание теста — в каких условиях мы собираемся тестировать функцию.
- На второй строке указываем само предположение о результате — что функция должна нам вернуть.
- На строчках 3–6 выполняем сам тест.
Функция expect
помогает избежать работы с ошибками напрямую и предоставляет удобные методы для сравнения аргументов друг с другом.
Обвязка из describe
и it
помогает нам описать тест в виде самого настоящего текстового предположения, которое код теста проверит.
Теперь разберём, собственно, код теста.
Наш тест состоит из 3 строк:
const result = add(10, 5);const expected = 15;expect(result).toEqual(expected);
const result = add(10, 5); const expected = 15; expect(result).toEqual(expected);
Его можно разделить на 3 стадии, которые можно запомнить по мнемоникам:
- ПВП: Подготовка, Выполнение, Проверка;
- или по-английски AAA: Arrange, Act и Assert.
На стадии подготовки мы готовим исходные данные для функции и ожидаемый результат. В более сложных тестах на этой стадии мы бы готовили зависимости для функции и фиктивные объекты.
В нашем случае подготовкой можно назвать выбор аргументов 10
и 5
, а также обозначение ожидаемого значения const expected = 15
.
На второй стадии мы запускаем функцию, чтобы получить результат. В этот момент мы отрабатываем тестируемый сценарий и получаем значение, которое потом будем проверять.
В нашем случае это строчка с вызовом функции: const result = add(10, 5)
.
На стадии проверки мы сверяем полученный результат с ожидаемым. Хотя проверка может состоять из нескольких утверждений, хорошей практикой считается внутри одного теста проверять только одно предположение.
В нашем случае проверка — это сравнение результатов на последней строке:
expect(result).toEqual(expected);
expect(result).toEqual(expected);
Идеальный тест состоит из всех трёх стадий ПВП. Часто — такой тест даже состоит из 3 строчек.
Кроме этого, идеальный тест проверяет только одно утверждение. Для проверки разных утверждений лучше написать отдельные тесты.
Идеальный тест не зависит от других тестов. Если тесты зависят друг от друга, они могут влиять и на результаты проверки друг друга. Правильно написанный тест можно запустить где и как угодно, и его результат не изменится.
Может показаться, что тестирование это пустая трата времени, и лучше выделить это время на другие задачи. И правда, на начальных этапах жизни проекта тесты отнимают время. Но в будущем написанные тесты сохранят больше времени, потому что будут играть роль и документации, и инструмента разработки, и проверки ранее написанного кода.
Рассмотрим основные плюсы тестирования.
Когда мы пишем программу, мы чаще думаем об основном сценарии работы (happy path), часто забывая о крайних случаях.
Тесты смещают фокус с основного сценария на «что может пойти не так». Когда мы пишем тесты, мы больше склонны искать ошибки и неадекватную работу функции. Чем больше крайних случаев мы обработаем, тем надёжнее будет код.
Регрессия — это ошибка, которая возникает в уже работающей части системы после изменений в коде. Это может быть добавление новой функциональности в старый код или доработка общих функций.
В нашей голове может уместиться лишь небольшой кусочек системы, которую мы программируем. Большая часть системы всегда находится вне поля нашего зрения. Это значит, что при добавлении функциональности мы можем не учесть особенности работы уже существующего кода.
Тесты закрывают такие ошибки, потому что падают при возникновении неучтённой ситуации и не дают ей отправиться в продакшен.
Когда мы рефакторим код, мы изменяем его структуру. Ошибки при рефакторинге появляются по той же причине — мы не можем держать всё в голове.
Даже если мы уверены, что полностью знаем кусок, который рефакторим, мы не застрахованы от более простых ошибок:
// 1.let a = 15;if (a == 20) {}// 2.let a = 15;if (a = 20) {}
let a = 15; if (a == 20) {} let a = 15; if (a = 20) {}
Второй случай в примере выше всегда будет истинным, потому что в условии вместо сравнения — присваивание. Тесты уберегут от подобных ошибок.
Это частный случай регрессий. Обновление зависимостей не только исправляет баги и несёт новые фичи, но иногда и ломает логику работы. Если наш код протестирован, то такие ломающие обновления мы определим сразу.
Тесты не врут ?
Они действительно показывают, как работает система.
Дополнительная документация может устареть, особенно часто это случается с комментариями. Если документация устарела, у нас появляется два источника правды: документация и код. Это плохо, потому что непонятно, чему верить, и как программа должна работать. Тесты же точно говорят, как программа работать должна и работает ли.
Кроме этого, хорошо написанные тесты читаются как текст на английском языке. Из их описаний при желании можно автоматически сгенерировать и текстовую документацию.
Тестирование не бесплатное, за надёжность кода приходится платить.
На старте проекта издержки ощущаются больнее всего. Чем моложе проект, тем субъективно большую долю времени будут занимать тесты. В этот момент желание плюнуть и не писать тесты сильнее всего, потому что время как будто уходит на бесполезную работу.
Это, конечно, не так. Чем проект старше, тем больше мы получаем выгоды от написанных тестов.
Избавиться от ощущения ненужности тестов можно, если заранее закладывать время на них в план, а также считать их базовой гигиеной при работе. Часть команд и проектов сейчас не работают без тестов вовсе. Такие команды — это хорошая среда для того, чтобы привить себе привычку к тестированию на уровне автоматизма.
При работе с тестами приходится использовать фиктивные объекты и данные. Например, для проверки аутентификации мы не поднимаем настоящий сервер, а используем заглушку для него, которая отвечает нужными нам в конкретном тесте данными.
Чтобы не запутаться во всех фиктивных объектах и данных, для них нужно организовать структуру. Чем удобнее структура, тем проще писать тесты.
Грамотно организовать систему с фиктивными объектами сложно. Это требует навыков проектирования, знаний о хорошей архитектуре и опыта.
Тесты в проекте существуют, как правило, вместе с автоматическими задачами, которые их запускают. Они, например, могут отменять релиз, если при обновлении кода тесты не проходят.
Настройка таких задач — это тоже дополнительная работа.
Хорошо, вот мы взвесили все преимущества и недостатки тестирования. Если мы хотим внедрить тестирование в своём проекте, то какие тесты нам писать?
Тесты бывают разных видов и проверяют они тоже разные вещи. Рассмотрим основные типы тестов в виде пирамиды и поговорим о каждом.
Пирамида тестирования: в основании Unit-тесты, чуть выше интеграционные, на вершине — End-to-End.
В основании пирамиды лежат юнит-тесты. Их ещё называют модульными тестами или блочными тестами.
Такие тесты проверяют работу конкретного модуля, функции или части программы. Когда мы писали тест для функции add
выше, мы писали именно юнит-тест.
Особенность таких тестов в том, что они затрагивают лишь одну часть программы, никак не проверяя остальные. Если тестируемый модуль ссылается на другие модули, то вместо них мы будем использовать фиктивные объекты. Поведение таких объектов мы смоделируем таким, как нам требуется в каждом конкретном тесте.
Как правило, большая часть тестов в проекте — это модульные тесты. Их проще писать, для них не требуется слишком сложной структуры. Они быстро проверяются и их можно запускать параллельно, потому что они не зависят друг от друга.
Интеграционные тесты проверяют, как между собой взаимодействуют две и более части программы, которые работают над одной задачей (то есть как они интегрированы друг с другом, отсюда название).
Для интеграционных тестов нужна структура чуть сложнее, чем для модульных. Чтобы запустить интеграционный тест, надо изолированно запустить несколько частей программы.
Часто для интеграционного тестирования нужен отдельный фреймворк, который умеет читать спецификации задач и проверять, что программа работает правильно. Важно, что в таком тестировании мы проверяем не порядок вызова частей программы, а именно результат их совместной работы.
Интеграционными тестами проверяют связующие модули (так называемое middleware):
- юнит-тесты проверяют прямое назначение middleware в целом;
- интеграционные — как middleware взаимодействует с конкретными модулями.
Они же End-to-End тесты, они же системные тесты — это проверка работы программы в целом.
Такие тесты играют роль пользователей, которые ходят по приложению и выполняют разные задачи. Для этих тестов тоже необходим фреймворк, который умеет манипулировать приложением, как это делают люди.
Например, мы захотели полностью проверить, как работает регистрация на сайте. Мы можем написать сценарий для «робота», который будет ходить по сайту, заполнять формы и проверять, как прошла регистрация.
Приёмочные тесты — это проверка программы перед сдачей клиенту или релизом. Как правило, такие тесты ручные и включают в себя сложные сценарии из спецификаций или технического задания.
Во фронтенд-разработке они встречаются реже всех остальных, но часто используются при разработке десктопных программ или приложений для телефонов. У таких приложений более долгий цикл доставки до пользователя, поэтому приёмочные тесты обычно проходят перед релизом. В вебе цикл поставки короткий, и тесты стараются максимально полно автоматизировать.
Даже при наличии издержек тестирование экономит силы, время и деньги в долгосрочной перспективе. Отказаться от тестирования можно, если:
- мы разрабатываем прототип, который не пойдёт в продакшен — скорость важнее;
- или проект точно не будет жить долго — предсказать такое сложно, но если мы уверены, что тесты не успеют окупиться, ими можно пренебречь.
В остальных случаях тесты лучше писать с самого начала.
Несколько дней назад попалась мне одна статья про то, что Vim достаточно хорош как инструмент для разработчика. В комментариях разгорелось очередное противостояние приверженцев vim против остального мира. Мыслей по данному вопросу накопилось достаточно много, поэтому считаю разумным сформулировать их уже в виде статьи. Адепты vim говорят о том что, якобы, освоив vim и полностью настроив его под себя можно получить ощутимую прибавку в скорости разработки. Я же считаю иначе. Лично я в работе использую PhpStorm, поэтому буду сравнивать с vim именно его. Так же оговоримся, что рассматривать инструменты я буду в контексте, деятельности вэб-разработчика. А для наглядности рассмотрим как одни и те же задачи будут решаться в IDE и в Vim. Рабочие инструменты берем в состоянии «из коробки». Предполагаем что на компьютере уже установлен vim или IDE в коробочном варианте.
Начало
Итак есть проект к работе над которым нужно подключиться. Есть репозиторий в котором ведется работа над проектом и дамп базы данных. Какие действия требуются для начала работы? В шторме создается проект сразу из репозитория и автоматически загружается в рабочий каталог. Остается только добавить локальную базу данных, импортировать дамп, подтянуть зависимости и настроить подключение к локальному Xdebug. Все это делается встроенными в IDE средствами. Единственная сложность с которой придется столкнуться — нам придется удерживать повышенный контроль над ходом мыслей во время перекладывания руки с клавиатуры на мышь/тачпад, ведь по заверениям пользователей vim мысль резко обрывается, стоит только оторвать руки от клавиатуры.
В случае с vim придется установить плагин для работы с git, а уже потом скачивать проект. Не знаю есть ли плагин для работы с базами в vim, но даже если и есть время на его поиск, установку и настройку может оказаться больше чем просто развернуть дамп из командной строки в терминале. И это будет первый раз когда нам придется покинуть рабочее окружение. В дальнейшем мы будем вынуждены закрывать рабочее окружение каждый раз, когда нам потребуется какой-то плагин, а требоваться они будут неприлично часто.
Первая задача
Итак получаем первую задачу по проекту. Какой-то функционал ведет себя совсем не так, как должен, и то и вовсе падает с ошибкой. Приступаем. Скорость навигации по каталогам в поисках нужных файлов будет примерно одинаковой потому, что файловый менеджер — он и в Африке файловый менеджер, однако в PhpStorm он уже есть, а для vim снова нужен плагин. И вновь мы тратим драгоценные минуты на установку необходимого функционала.
Если в процессе работы мы встретим в legacy-код, то для начала его надо хоть как-то бы привести в читаемый вид. В IDE придется исполнить аккорд на клавиатуре Alt+Shit+L
, однако в этом вопросе vim отличается только аккордом. Точнее у него их несколько.
В IDE, благодаря умному анализу кода, ничего особенного не предпринимая, мы сразу увидим какие переменные не используются, какие циклы окажутся вечными, какие условия никогда не изменятся, а значит мы увидим большинство проблемных мест, просто открыв файл. Vim же предлагает нам внимательно вникать в каждую строчку и выискивать все самостоятельно, проявляя весь свой талант разработчика. Сколько уйдет на это времени и насколько сильно утомится разработчик от такой работы сказать сложно. Видимо именно из-за этого перенос рук с клавиатуры на мышь сбивает vim-овцев c мысли.
Что мы имеем? Мы еще не написали ни одной строчки кода, а разработчик, использующий vim уже потратил ощутимое количество своего ресурса.
Следующим шагом логично посмотреть в дебагере как ведет себя приложение, какие данные откуда и куда ходят, какие значения пишутся в переменные и как они используются. Расставляем брекпоинты (в шторме это тоже можно делать с клавиатуры) и нажимаем Shift+F9. Перед нами откроется полная картина происходящего. IDE выведет все значения переменных не только в окне дебагера, но и наложит их на редактор кода. Можно будет отследить все подключаемые файлы и цепочки вызовов. Сможет ли сделать это vim? Пускай знатоки подскажут в комментариях.
На этапе рефакторинга пользователь vim столкнется с непреодолимыми проблемами. Он не сможет одной кнопкой провалиться в конкретный метод или класс, не сможет быстро переименовать что-то, что используется больше чем в одном месте. Если что-то будет удаляться, vim не подскажет ему, что этот файл используется в другом месте и удаление может поломать систему еще больше. Не узнает он и о том, что и переименование может потянуть за собой цепочку ошибок в других местах.
А если что-то пошло не так? Сможет ли пользователь vim откатиться по локальной истории и вернуть все взад? Или предлагаете коммитить построчно?
А что если мы разрабатываем интерфейс API и нам нужно покидать на него запросы и посмотреть какие приходят ответы? Без проблем — в IDE есть встроенный HTTP-клиент и мы можем баловаться с запросами так, как захотим, а что предлагает нам vim-комьюнити? Использовать штатный консольный клиент операционной системы? Но для этого мы вновь вынуждены покинуть рабочее окружение. И пока ведется разработка, мы так и будем скакать из одного окружения в другое.
Что дальше?
А дальше будут другие задачи. И знаете что? Их тоже можно подключить в свою IDE. PhpStorm из коробки умеет подключаться к таким сервисам, как Jira, Trello, YouTrack, Redmine, Bugzilla, Git, Asana и другим. Вы даже можете отслеживать время затраченное на решение задач. Здесь vim уверенно проигрывает, потому что не рассчитан на командную работу.
Пока что используя, vim мы не получили какой-то существенной прибавки в производительности, а скорее даже наоборот. Так чем же могут похвастаться пользователи vim? Предлагаю рассмотреть аргументацию самих vim-еров чтобы попытаться разобраться в вопросе. Аргументы я искал в комментариях к указанной выше статье, в других статьях и на просторах интернета.
Почему vim?
Vim производительный и может легко открыть файл в несколько гигов.
Да, это правда. А еще правда то, что разработчику не нужно открываться многогиговые файлы. Единственное для чего может пригодиться эта возможность — просмотр логов. Но зачем, когда всю необходимую информацию можно получить в отладчике?
Владея десяти-пальцевым методом вы получаете прибавку в скорости.
Да, но причем здесь vim? Или тот факт что управляется он только с клавиатуры как-то ускорит вам процесс рефакторинга? Современная разработка это по большей части чтение и анализ кода. Владея слепым методом вы просто будете писать быстрее и в vim, и в IDE. А учитывая то, что непосредственно на набор текста уходит меньше всего времени разработчика суммарно вы ничего не выигрываете. А для тех у кого руки приклеены к клавиатуре есть бесчисленное множество шорт-комбинаций и в IDE. Если хотите, можете даже vim-плагин поставить в IDE и радоваться. Для чего нужно отказываться от остальных возможностей IDE, мне не понятно.
Весь функционал vim направлен на то, чтобы пользоваться им без использования мыши. Отсюда многообразие команд. А из многообразия следует их изощренность, которая таковая лишь на первый взгляд.
Отнюдь. Nano тоже работает без мыши. Я, помню, даже DOS-среда разработки Turbo Pascal позволяла комфортно работать без мыши. Просто вместо вменяемого Text User Interface в vim применяется шаманская клавиатурная магия, возникшая стихийно на заре программирования, когда самого понятия «User Interface» еще существовало.
Со своим конфигом от vim можно прийти куда угодно и сразу же получить свою сборку.
Для этих же целей служат файлы .idea
Если чего-то нет в vim, то всегда можно написать самому.
Отлично, но тогда надо признать что редактор нужен только для того чтобы потешить ваше самолюбие, похвастаться тем что вы написали очередной крутой плагин для vim, вместо того чтобы решать задачи. По сути вы просто умело решаете проблемы, которых у вас никогда не было бы, если бы вы сразу использовали IDE.
Вместо заключения
В заключение расскажу одну притчу, которую придумал сам. Жили-были два мужика. Иван, да Матвей. И каждый год они вспахивали поля плугом, а осенью собирали с полей урожай. И чтобы урожай был хорошим — каждый из них как мог усовершенствовал свой плуг. Добавляли в него новые фичи, улучшали эргономику. Иван даже стул к нему приделал на колесиках, чтобы пахать сидя можно было. Но в какой-то момент их пути разошлись. Матвей купил себе трактор и стал собирать урожая кратно больше чем Иван. А Иван обозлился на Матвея и каждый раз проходя мимо трактора Матвея плевал в него, недовольно бормоча про то, что лучше плуга ничего нет и все эти тракторы жутко неудобны и громоздки, да и заправлять их все время надо и запчасти покупать. И как не пытался Матвей убедить Ивана купить себе трактор, но ничего у него не вышло.