В мире JavaScript существуют две фракции. Первая из них — технари, которые все проблемы стараются решать «технично». Вообще технари ребята суровые, я бы даже сказал строгие, и потому любят такую же суровую и строгую типизацию, и везде суют TypeScript, Dependency Injection и другой IoC.

Вторая же — маги. Кто-то, считает их шарлатанами, и уж никто точно не понимает как работает их код. Но он работает. На строгую типизацию у них табу, а про(от) DI у них есть простая отговорка:

«Зачем мне уродовать свой код, смешивая ужа с ежом, если это нужно исключительно для тестов?».

И ведь на самом деле — добавлять в проект DI исключительно чтобы мокать зависимости в тестах — идея не самая умная. Особенно если DI и на самом деле редкий зверь за пределами экосистемы Angular.

Есть только одно но — если технари от своей профдеформации не страдают, то маги… ну как сказать…

В общем пару месяцев назад один добрый человек создал мне в proxyquire-webpack-alias issue. Суть была проста — «не работает». Мне потребовался день чтобы изменить ЧТО не работает, на ГДЕ.



PS: Зачем нужно заменять (мокать) зависимости в тестах? Чтобы тесты были более «юнит», более изолированные, и не дергали реальные команды (ендпоинты), которые могут быть и очень медленными, и очень одноразовыми. В общем не надо трогать их в тестах.

Суть проблемы очень проста – для моканья зависимостей в nodejs придумано ОЧЕНЬ много библиотек: proxyquire, rewire, mockery и так далее. Все они паразитируют на внутренем представлении nodejs о модулях и их шупальца пробираются куда-то в начинку require.

Если же ваши тесты запускаются не в nodejs, а в браузере – то все меняется. Банально — нет никакого nodejs environment, только тот суррогат, что предоставил использованный бандлер. Но, так как бандеров вообще неограниченное колличество — будем раcсматривать только один — webpack.

Тем более, что некоторые бандлеры, типа browseryfy или (особенно) rollup «модульной» системы не имеют вообще. И не надо.

Webpack


Исторически сложилось, что есть только один подход к моканью зависимостей в webpack — использовать inject-loader или rewire, который тоже «loader».

Лоадеры просто «меняют» запрашиваемый файл на уровне исходных кодов. К принципе особых притензий к таким загрузчикам нету — вы просто говорите

const stuff = require('inject-loader!stuff')({
  'fs': mockFS,
  'someOtherDep': mock
});

И зависимости будут замоканы. Ну просто это немного каменный век, и использовать все это дело не так чтобы всегда просто. «Старшие братья» из nodejs (особенно mockery) умеют сильно сильно больше.

Со rewire сложнее. Я бы лично ломал бы пальцы тем кто его использует — «мокать» используя rewire это тоже самое, что «мокать» используя sinon.

С другой стороны — rewire-webpack это единственный(!) правильный на уровне исходных кодов плагин(не лоадер) для webpack. Просто потому что автор rewire зафейлил этот плагин написать, и его писал автор webpack. Хотя этот плагин в итоге просто добавляет loader. Я вот тоже запутался.
Большой плюс rewire, несмотря на его кракозяблость,- одинаковый интерфейс для webpack и node окружения. Он был один такой хороший.

Rewiremock


Несколько месяцев назад я написал чуть более «правильный», чем остальные, инструмент для моканья зависимостей — Rewiremock (github, статья на хабре). И мне было прямо «спортивно интересно» завести rewiremock не только под nodejs(с который вообще проблем нет), но и под webpack.

И так чтобы API не изменился, и все тесты работали. Сейчас не работает один тест, потому что не должен. А все остальные — зеленые.

1. Как оно работает?


Вся работа свелась буквально к трем пунтам:

1. Добавить «хоть какие-то» мозги модульной системе webpack. А именно требуется добавить два плагина, и оба с некой вероятностью уже есть — NamedModulesPlugin(который вернет файлам имена) и HotModuleReplacementPlugin(который предоставит некий сурогат модульной системы), плюс подключить плагин от rewiremock(который заменит require на свой вариант).
2. Дописать недостающий тулинг — очистка кеша, работа с чуть чуть другим «module».
3. Собственно нарисовать плагин, который как-то внедрит возможность перегрузки require.

Проблема возникла только с третьим пунктом — никакого хепла, доки или примера о том как можно сделать желаемое я найти не смог. Документация webpack часто отправляет курить сорсы, что тоже было проделано, только ясности не внесло.

Дальнейшее ознакомление с исходниками rewire-webpack, который, как я уже говорил, единственно правильный, решение подсказало. Точнее стало понятно, что просто все очень плохо.

Webpack большей частью основан на Tappable — маленькой библиотечке, которая вызывает хуки в некой последовательности. Это вроде как lifecycle… но что и когда она вызывает, какие аргументы передать, и (главное!) что с ними можно сделать — информации ноль. В общем webpack писали маги, а не технари.

Сейчас все думаю — оставиль свою реализацию плагина как есть, или заменить на более «правильное» из rewire. Оно просто раз в 5 длинее и смысла я в этом покуда не вижу.

2. Что в итоге?


В итоге — оно просто работает. Где-то внутри зашито многовато магии по нормализации имен файлов, чтобы все прозрачно работало в обоих экосистемах, но наружу точит достаточно простой и удобный API. Точнее целый букет, который не изменился.

Одна из «проблем» rewiremock — универсальность API.

Он может работать как mockery (базовый синтаксис как у mockery, включая режим изоляции):

rewiremock('fs')
    .with({
        readFile: yourFunction
    });
rewiremock.enable();

Может как proxyquire (хеплеры proxy и module):

rewiremock.proxy('somemodule', {
   'dep1': { name: 'override' },
   'dep2': { name: 'override' }
 }));

Умеет разные чтуки из Jest (например динамическое создание мока):

rewiremock('fs')
    .by(({requireActual}) => requireActual('fs'));

И есть кой какие свои приемы (расширенный синтаксис proxy):

const mock = await rewiremock.module(() => import('somemodule'), r => ({
   'dep1': r.with({ name: 'override' }).calledFromMock(),
}));

В общем, как я уже писал выше, rewiremock — инструмент чуть более лучший, чем все остальные. И первый, из «нормальных», который одинаково умеет работать как как под nodejs, так и под webpack.



Хотя. Кому он нужен под webpack то? Вот честно — поднимите руки, а то у меня знакомых которые сидят на Karma/Headless/Webpack/Angular, и которые могут проверить это все в деле – как-то не завелось.

PS: Он и под ноду то особенно не востребован. Старый добрый proxyquire, несмотря на все свои ограничения, справляется с 99% задач. О значимой разнице знаю только я…

Комментарии (9)


  1. justboris
    08.10.2017 16:51

    По умолчанию Karma-webpack собирает независимый бандл для каждого теста. Поэтому проблема очищения моков после каждого теста не стоит.


    Сейчас собрал примитивный пример с inject-loader, работает как надо.


    1. vasIvas
      08.10.2017 19:01

      А Вам не кажется, что это похоже на открывание консервов гидравлическими ножницами?
      Ведь это разные слои, webpack для внешних зависимостей, DI для зависимостей приложения.
      Зачем смешивать? [ прошу прощения, ни туда написал ]


      1. justboris
        08.10.2017 19:07

        На эту тему автор уже отвечал в комментариях к прошлой статье: https://habrahabr.ru/post/329740/#comment_10258456


        если у вам повезло, и у вас есть нормальный DI, то у вас все хорошо. А если не повезло — то proxyquire и компания — неплохой способ выкрутиться без переписывания кода целиком.


    1. kashey Автор
      09.10.2017 01:26

      У меня и самого есть проект на inject-loader, и я им в принципе доволен. Но иногда по пути попадаются грабли, на которые редко, да наступаю.

      Давайте придумаем немного синтетический пример:

      import function1 from 'module1';
      export default () => function1()+1;
      

      После чего мы просто мокаем module1 и expect(function1).to.be.called();
      Проходит время, и Вася добавляет еще одну строчку:
      import function1 from 'module1';
      import function2 from 'module2';
      export default () => function1()+function2()+1;
      

      Но он не меняет тесты, а тесты как работали, так и работают.

      Бывает и обратная ситуация — вы убираете строчку, а тесты как работали, так и работают.

      Беда в том, что тесты должны не только «тестировать» поведение функции, те проверять результат — они должны служить вторым контуром проверки, обеспечивать еще один «момент подумать».
      И, по хорошему, должны падать при _любом_ изменении логики програмы. Просто потому что они должны его детектить и подносить вам на блюдечке.

      Да — совсем не круто править 10 тестов после того как изменил одну запятую. Но еще более не круто – не догадываться, что вы изменили чуть больше логики, чем думаете, этой одной запятой.

      В свое время я пытался пропихнуть API для этого в proxyquire, но не срослось.
      В rewiremock для мока можно определить то как он может быть вызван — toBeUsed, directChildOnly, calledFromMock, плюс режим isolation который будет ругаться на каждый неожиданный import.

      Раз уж units tests are production code — относитесь к ним соотвествующе. Хотя конечно можно добавить тесты для тестов, но нельзя добавить тесты для моков, те заставить моки соотвествовать интерфейсам того, кого они собой замещают – тут нужен DI. Но он тоже не сахар.


  1. AlexPu
    09.10.2017 09:12

    >> DI и на самом деле редкий зверь за пределами экосистемы Angular

    Дальше читать не стал


    1. kashey Автор
      09.10.2017 10:22
      +1

      Но «нормальный» DI(уровня SpringBoot/ng-di) на самом деле редкий зверь в мире JavaScript. Есть взять пяток самых популярный библиотек, убрать то что для ng, и посмотреть на статистику скачек — будет не очень. Плюс многие путают DI с более широким IoT.

      DI хорош, когда в одну и туже розетку можно засунуть разные вилки. Если пара вилка/розетка всегда одна, и DI — исключительно для тестов — может его и не надо?

      И, самое главное, многие вещи просто принято использовать «как есть». Именно поэтому 90% примеров моканья чего либо, мокают: fs, сеть, session storage, селекторы в редаксе и другой environment.


      1. AlexPu
        09.10.2017 13:59

        >>может его и не надо?

        Что и когда надо или НЕ надо это вопрос за который я, как и большинство моих коллег получаем заработную плату. Поэтому, безапелляционные утверждения, основаные в основном на недостаточной информированности вызывают у меня легкое чуство раздражения (с пониманием того факта, что когда опять все станет совсем плохо все равно позовут меня или какого нибудь из моих коллег, которые не склонны к радикальным взглядам на что бы то ни было)


        1. kashey Автор
          09.10.2017 14:10
          +2

          У вас метод индукции сбоит. В данный момент безаппеляционные утверждения исходят только с вашей стороны.
          Вроде как приверженность только одного взгляда — и есть проявление радикального.
          У меня каждый день есть возможности видеть плюсы и минусы разных подходов на разных языках… и в большинстве случаев использование «классического» DI — это или Java, или объектно ориентированный говнокод 5-то летней давности.


          1. AlexPu
            09.10.2017 15:23

            Ну… как скажете — вам виднее :)