Заметки в категории «JavaScript» (страница 3)

Node.js + mocha + selenium-webdriver

Для запуска функциональных тестов нам понадобятся:

  1. Node.js — надеюсь, он у вас уже установлен;
  2. Python 2.7.х — указан в зависимостях модуля selenium-webdriver и должен быть установлен в систему до загрузки остальных пакетов;
  3. Java Runtime Environment — среда для запуска Selenium Server;
  4. mocha — с помощью этой утилиты мы будем запускать тесты; устанавливается командой npm i -g mocha;
  5. selenium-webdriver — через него мы будем оправлять Selenium-серверу команды и получать от него данные со страницы; устанавливается в папку проекта командой npm i selenium-webdriver;
  6. Selenium server — запускает и закрывает браузер, отправляет браузеру команды и т.п.; не требует особой установки; скачиваем JAR-файл в папку рядом с тестами;
  7. Driver Provider — вспомогательный класс, в котором сосредоточен код для запуска и остановки сервера и конфигурация драйвера (например, выбор браузера для тестирования).

Когда всё перечисленное будет установлено, можно начинать писать и запускать тесты.

В каждом пакете тестов нужно создать экземпляр класса DriverProvider . Как я указал выше, с его помощью мы будем управлять сервером и драйвером.

var driverProvider = new DriverProvider();

Перед запуском пакета тестов выполняется метод before:

test.before(function () {
  return driverProvider.startUp();
});

По окончании работы тестов выполняется метод after:


test.after(function () {
  return driverProvider.tearDown();
});

DriverProvider — это хорошая точка расширения функциональности. В этом классе можно управлять конфигурацией драйвера без вмешательства в код самих тестов. Например, я указываю там браузер, в котором нужно запустить тест. Код можно слегка поправить для использования удалённого Selenium-сервера или запуск специального прокси-сервера.

В методах startUp() и tearDown() этого класса выполняются задачи по запуску и остановке соответствующих сервисов. Они должны возвращать promise-объект. По его состоянию будет определяться завершена-ли задача или нет.

Отличительной особенностью API selenium-webdriver является то, что большинство методов выполняются асинхронно. Однако, постоянно использоваться promise-объекты, чтобы обеспечить нужный порядок выполнения команд, вовсе не требуется. Для этого есть так называемые “control flow”. Внутри одного потока все команды будут исполняться синхронно по мере того как они были добавлены в поток. Это немного облегчает написание тестов. Но, если нужно получить значение из функции (например, список элементов, размеры элемента и т.д.), то без “обещаний” не обойтись.

Так же библиотека selenium-webdriver/testing тоже использует “control flow” чтобы последовательно исполнять содержимое тестов. Она дублирует методы mocha. Тест может вернуть promise-объект и он будет считаться завершённым, когда обещание будет установлено в какое-либо состояние. Если тест ничего не вернул, то он будет считаться завершенным, когда очередь заданий окажется пуста.

И в заключение команда, которая запустит на выполнение наш тест:

mocha page-spec.js

У команды mocha есть масса полезных параметров. Мне, например, нравятся отчёты в виде спецификации. Ещё можно указать максимальное время выполнения теста. Функциональные тесты, как правило, не быстрые и 2 секунды на тест (значение по-умолчанию) совсем не хватает, чтобы его выполнить.

В заключение, хочу обратить внимание на шаблон написания тестов Page Object . Этот подход позволяет выделить взаимодействие с WebDriver и повторяющиеся во время тестирования действия в изолированные классы. Практика показывает, что таким образом легче поддерживать тесты в актуальном состоянии.

Комментарии к заметке: 2

Ссылка на глобальный объект в JavaScript

Пожалуй, единственное, для чего может понадобиться глобальный объект, это экспорт каких-либо данных из модуля. Получить этот объект не составляет труда:


function myCoolFunction() {
  var global = function () { return this; }();
}

Переменная global внутри myCoolFunction() будет соответствовать window, если код выполняется в браузере, и global для окружения Node.js.

Однако, когда функция выполняется в строгом режиме , такой трюк не пройдёт. В «strict mode» значение this будет равно undefined.


function myCoolFunction() {
  'use strict';
  var global = Function('return this')();
}

Этот приём работает корректно из-за того, что функция, созданная с помощью конструктора Function, не наследует «строгость».

Оставте свой комментарий

Безопасное получение свойств вложенных объектов

Доступ к свойствам вложенных объектов через «.» или «[]» опасен тем, что если в цепочке не будет указанного объекта, то будет выброшено исключение.


var obj = {a: {b: {c: 42} } };
obj.a.x.y;

TypeError: Cannot read property ‘y’ of undefined

От таких ситуаций можно обезопаситься, если воспользоваться простой функцией.


function getProperty(obj, prop) {
  return prop.split('.')
    .reduce(function (m, i) {
      return m && typeof m == 'object' ? m[i] : null;
    }, obj);
}

var obj = {a: {b: {c: 42} } };
getProperty(obj, 'a.x.y');

null

Комментарии к заметке: 9

Автоматизируем запуск QUnit тестов

QUnit очень старый фреймворк для тестирования браузерного JavaScript. Чтобы запустить тесты, нужно создать небольшой HTML-файл с несколькими div-ами, стилями и скриптами QUnit и, собственно, тестовыми скриптами. Затем этот файл можно открывать в разных браузерах и смотреть отчёт о выполнении тестов.

Запуск этих тестов можно поручить PhantomJS. Для этого нужно воспользоваться скриптом QUnit PhantomJS Runner.

$ phantomjs runner.js test/index.html

Если у вас используется Grunt для запуска различных задач, то для него есть плагин grunt-contrib-qunit. Он так же использует PhantomJS для прогона тестов.

grunt.initConfig({
  qunit: {
    all: ['test/index.html']
  }
});
Оставте свой комментарий

Динамическая загрузка «отзывчивых» ресурсов

Любой медиа-запрос можно выполнить в реальном времени с помощью JavaScript. Создать экземпляр объекта, который возвращал бы результат проверки Media Query или оповещал бы об изменении этого результата, можно при помощи метода .matchMedia(). Все современные браузеры уже реализовали этот метод в объекте window. Если же вам нужна поддержка старых браузеров, то можно воспользоваться полифилом matchMedia.js.


var mql = window.matchMedia("(min-width: 720px)");
if (mql.matches) {
  // десктоп или планшет
} else {
  // мобильник
}

Теперь, когда мы знаем как организовать ветвление, основываясь на результате выполнения медиа-запроса, подгрузить «тяжелые» стили и скрипты можно только если они действительно необходимы.


var Environment = (function () {
  var _mqDesktop;

  function getDesktopMediaQuery() {
    return _mqDesktop ||
      (_mqDesktop = window.matchMedia("(min-width: 720px)"));
  }

  function isDesktop() {
    return !!(window.matchMedia && getDesktopMediaQuery().matches);
  }

  function once(mq, callback) {
    function handler() {
      mq.removeListener(handler);
      callback();
    }
    mq.addListener(handler);
  }

  return {
    /**
     * Test the environment against “desktop” media query
     */
    isDesktop: isDesktop,
    /**
     * Invoke callback when the environment becomes like “desktop”.
     */
    waitForDesktop: function (callback) {
      if (isDesktop()) {
        callback();
      } else {
        once(getDesktopMediaQuery(), callback);
      }
    }
  };

})();

Далее в коде приложения сделать вызов:


Environment.waitForDesktop(function () {
  Modernizr.load([
    "/styles/desktop.css",
    "/scripts/desktop.js"
  ]);
});

Таким образом, файл стилей /styles/desktop.css и скрипт /scripts/desktop.js загрузятся только когда ширина экрана будет больше 720 пикселей.

Из-за того, что файлы загружаются асинхронно появляется побочный эффект — нет никакой гарантии, что выполнение скрипта начнётся только после применения стилей. Например, габариты блока будут не верны или блок вообще может быть невидимым во время инициализации компоненты. Это нужно учесть и дождаться полной загрузки CSS файла с перерисовкой страницы.

Проверить, что файл стилей гарантировано применился можно с помощью специальных стилей-маркеров.

mobile.css:

.style-detector {
  z-index: 1;
}

desktop.css:


@media (min-width: 720px) {
  .style-detector {
    z-index: 2;
  }
}

А колбек обернуть в функцию, которая ожидает нужного значения свойства z-index .


function waitUntil(value, cb) {
  var body, detector = document.querySelector(".style-detector");
  if (!detector) {
    detector = document.createElement("div");
    detector.className = "style-detector";
    body = document.getElementsByTagName("body")[0];
    body.insertBefore(detector, body.firstChild);
  }

  function getStyle(el, cssProperty) {
    return (getComputedStyle !== "undefined") ?
      getComputedStyle(el)[cssProperty] : el.currentStyle[cssProperty];
  }

  var i = 100; // timeout = 25 sec
  (function loop() {
    var v = getStyle(detector, "zIndex");
    if (v != value) {
      i = i - 1;
      if (i >= 0) {
        setTimeout(loop, 250);
      } else {
        cb();
      }
    } else {
      cb();
    }
  }());
}

Полный код модуля доступен на Гитхабе.

Комментарии к заметке: 5