Разработка плагинов для Discourse — Часть 6: Добавление тестов принятия

Предыдущее руководство: Developing Discourse Plugins - Part 5 - Add an admin interface


Знаете ли вы, что в коде Discourse есть две большие наборы тестов? На стороне сервера наш Ruby-код имеет набор тестов, использующий rspec. Для браузерного приложения у нас есть набор qunit, в который включено ember-testing.

Если у вас настроена среда разработки, перейдите по адресу http://localhost:3000/tests, чтобы запустить набор JavaScript-тестов в браузере. Один из интересных аспектов — вы можете наблюдать за тестированием приложения в миниатюрном окне в правом нижнем углу:

Приложение Discourse содержит очень много тестов, которые начнут выполняться при переходе по адресу /tests. Поэтому может быть полезно отфильтровать тесты по плагину, над которым вы работаете. Это можно сделать в интерфейсе, щёлкнув по выпадающему списку Plugin и выбрав свой плагин:

Добавление acceptance-теста в ваш плагин

Сначала убедитесь, что у вас установлена последняя версия Discourse. Возможность запуска acceptance-тестов из плагинов — относительно новая функция, и если вы не используете последнюю версию, ваши тесты не появятся.

В этой статье я напишу acceptance-тест для плагина purple-tentacle, который мы создали в части 5 этой серии.

Добавление acceptance-теста так же просто, как создание одного файла в вашем плагине. Создайте следующий файл:

test/javascripts/acceptance/purple-tentacle-test.js

import { acceptance, exists } from "discourse/tests/helpers/qunit-helpers";
import { click, visit } from "@ember/test-helpers";
import { test } from "qunit";

acceptance("Purple Tentacle", function (needs) {
  needs.settings({ purple_tentacle_enabled: true });
  needs.user();

  test("Purple tentacle button works", async function (assert) {
    await visit("/admin/plugins/purple-tentacle");
    assert.ok(exists("#show-tentacle"), "it shows the purple tentacle button");
    assert.ok(!exists(".tentacle"), "the tentacle is not shown yet");
    await click("#show-tentacle");
    assert.ok(exists(".tentacle"), "the tentacle wants to rule the world!");
  });
});

Я постарался написать тест максимально понятно, но он может показаться немного запутанным, если вы никогда не писали acceptance-тесты. Я настоятельно рекомендую вам прочитать документацию Ember по acceptance-тестированию, так как в ней содержится много полезной информации.

В каждом тесте нам нужно сделать assert (утверждение). В нашем тесте мы хотим проверить несколько утверждений: скрыт ли щупальце изначально и появляется ли оно только после нажатия кнопки.

Мы хотим определить набор действий, которые должны быть выполнены перед проверкой утверждения. Для этого используется ключевое слово await. Используя его, мы ждём завершения выполнения каждого асинхронного помощника.

Наше первое важное действие: await visit("/admin/plugins/purple-tentacle");. Это говорит тесту перейти по указанному URL в нашем приложении. Именно этот URL отображает щупальце.

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

Это делается с помощью следующих утверждений:

assert.ok(exists("#show-tentacle"), "it shows the purple tentacle button");
assert.ok(!exists(".tentacle"), "the tentacle is not shown yet");

P.S. В предыдущей версии плагина purple-tentacle в шаблоне Handlebars не было идентификатора элемента #show-tentacle. Проверьте последнюю версию, чтобы следовать инструкции!

После того как эти тесты пройдут, пришло время проверить взаимодействие.

Следующая команда — await click('#show-tentacle');, которая сообщает нашему фреймворку тестирования, что мы хотим нажать кнопку и показать щупальце.

После симуляции нажатия на кнопку мы можем проверить, появилось ли щупальце, сделав утверждение:

assert.ok(exists(".tentacle"), "the tentacle wants to rule the world!");

Не так уж и сложно, правда? Вы можете попробовать запустить тест самостоятельно, перейдя по адресу http://localhost:3000/tests?qunit_single_plugin=purple-tentacle&qunit_skip_core=1 на вашей машине с настроенной средой разработки. Вы очень быстро увидите появление фиолетового щупальца, и все тесты пройдут успешно.

Если вы хотите запустить qunit-тесты плагина из командной строки с помощью PhantomJS, выполните команду:

rake plugin:qunit['purple-tentacle']

(где purple-tentacle — имя папки вашего плагина)

Отладка ваших тестов

При написании плагинов тесты помогают выявлять проблемы в коде. Когда вы разрабатываете тесты или вносите изменения в код плагина, тесты могут не проходить. Чтобы понять причину, в Ember есть удобные помощники: pauseTest() и resumeTest().

Чтобы воспользоваться ими, добавьте await pauseTest() в нужном месте вашего тестового кода, где вы хотите приостановить выполнение теста. Теперь при запуске теста в браузере он автоматически остановится в точке, где вы добавили pauseTest(). Это даст вам возможность проверить страницу или увидеть любые ошибки, чтобы помочь в отладке.

Куда двигаться дальше

Мне не хочется звучать как заезженная пластинка, но документация Ember по тестированию превосходна. Также стоит посмотреть, как Discourse тестирует различные функции, просматривая тесты в нашем каталоге JavaScript-тестов. Там есть множество примеров, которые помогут вам научиться.

Удачного тестирования!


Больше в этой серии

Часть 1: Основы плагинов
Часть 2: Plugin Outlets
Часть 3: Настройки сайта
Часть 4: Настройка git
Часть 5: Административные интерфейсы
Часть 6: Эта тема
Часть 7: Опубликуйте ваш плагин


Этот документ находится под версионным контролем — предлагайте изменения на GitHub.

30 лайков

OMG, have I put off writing tests for this long?!

A few things
I noticed you put an “id” into the template file that wasn’t there in the previous tutorial
I added 3 more tests to deal with the “Hide Tentacle” button I added
I needed to be logged in it is an Admin page
I needed to use port 4000 for my VirtualBox Vagrant set-up

Happily, quick success !

5 лайков

I’ve noticed a push to implement testing for plugins lately and I completely agree that this is important. I’ve been terrible at writing these in the past and want to get caught up.

I was considering the setup of Travis CI on a plugin repo as I noticed the ability to use Docker with Travis. But having never touched Travis before, is it possible to do this with a plugin? It would be nice to submit PRs to my own repo and have the tests run automatically.

5 лайков

how can I run single ember tests on the browser?

http://localhost:3000/qunit?module=Acceptance%3A%20Purple%20Tentacle

I mean, if I have another acceptance test called “another test”, how would I run it? Thank you!

If you click “rerun” beside a test, you’ll get a URL that is just for that particular test that you can refresh over and over.

You can also run by module by clicking in the “All Module” box and checking off just the modules you want to run.

2 лайка

This tutorial needs updating to replace ok(...) with assert.ok(...).

Have put the necessary changes in a PR - I’m happy to edit the OP if it’s made into a wiki :slight_smile:

https://github.com/eviltrout/purple-tentacle/pull/2

2 лайка

OK it is wikified! PM me if anything else needs this treatment any time.

1 лайк

A note after writing some basic acceptance tests for the Elections Plugin:

  • If you use the default loggedIn option, you’re logged in as eviltrout with the properties in this session fixture

  • You can login as a custom user (necessary if you’ve added custom user properties) by defining your own session fixture and setting it as the current user using the beforeEach hook. See here for an example.

@eviltrout is there a better way of using a custom user for acceptance tests?

5 лайков

What we should probably do is change the acceptance method in discourse to take an argument for the user to log in as. Then your test could say { loggedIn: true, logInAs: angus }

If you’ve got time for that PR we’d accept it!

4 лайка

q: how to run the ruby tests tests for a plugin?

i am trying to run the tests for discourse-github-linkback, but i have no idea how.

After some stackoverflow i tried this (and i got some errors):

~/discourse$ bundle exec rspec ./plugins/discourse-github-linkback/spec/lib/github_linkback_spec.rb 

An error occurred while loading ./plugins/discourse-github-linkback/spec/lib/github_linkback_spec.rb.
Failure/Error: CoreExt::ActiveSupport.without_bootsnap_cache { super }

NameError:
  uninitialized constant GithubLinkback
# ./plugins/discourse-github-linkback/spec/lib/github_linkback_spec.rb:3:in `<top (required)>'
# ------------------
# --- Caused by: ---
# NameError:
#   uninitialized constant GithubLinkback
#   ./plugins/discourse-github-linkback/spec/lib/github_linkback_spec.rb:3:in `<top (required)>'
No examples found.

Randomized with seed 60987


Finished in 0.02025 seconds (files took 3.36 seconds to load)
0 examples, 0 failures, 1 error occurred outside of examples

Randomized with seed 60987

Note that i see the plugin in the local discourse dev admin page, so i hope it is correctly installed (into ~discourse/plugins/

Simplest way, run bin/rake autospec, and save the file

1 лайк

From what i see, doing cd ~/discourse and then running

runs all the tests for discourse.

On my ubuntu 18.04 VM with i5 6500, this has been running for half an hour already.
It feels wrong to run the whole tests suite when i only want to run the tests in that one plugin.
(this is especially so when i probably will modify my tests again, at least a few times)

Did i do something wrong?

Let me guess, you are symlinking your plugin vs checking out directly to the path under plugins

no,

tbp@tbp-linux-dev:~$ cd ~/discourse/plugins/
git clone -b allow-all-user-repositories-wildcard https://github.com/TheBestPessimist/discourse-github-linkback
tbp@tbp-linux-dev:~/discourse/plugins$ ll
total 40
drwxr-xr-x 10 tbp tbp 4096 May 13 21:48 ./
drwxr-xr-x 20 tbp tbp 4096 May 13 21:18 ../
drwxr-xr-x  6 tbp tbp 4096 May 13 20:29 discourse-details/
drwxr-xr-x  6 tbp tbp 4096 May 13 21:48 discourse-github-linkback/
drwxr-xr-x  6 tbp tbp 4096 May 13 20:29 discourse-local-dates/
drwxr-xr-x  9 tbp tbp 4096 May 13 20:29 discourse-narrative-bot/
drwxr-xr-x  7 tbp tbp 4096 May 13 20:29 discourse-nginx-performance-report/
drwxr-xr-x  5 tbp tbp 4096 May 13 20:29 discourse-presence/
drwxr-xr-x  3 tbp tbp 4096 May 13 20:29 lazyYT/
drwxr-xr-x  9 tbp tbp 4096 May 13 20:29 poll/
tbp@tbp-linux-dev:~/discourse/plugins$ 

Can you confirm you tried saving the spec file once it was running, cause it is meant to interrupt stuff and start running the plugin spec?

since the initial bin/rake autospec is still running, there was/is no “save spec file” message yet.
also, by running bin/rake autospec --help, i saw no --save option.

I didn’t understand what save the file meant, so my full command was bin/rake autospec > spec.rb ← probably this is wrong

I meant in Vim or whatever you use, save the spec file, there is a watcher on it that will invoke the spec runner once it is saved.

Also see: How do I run only discourse/plugins/poll/spec? - #6 by tgxworld

1 лайк

Finally i did it :^).

With the help of that link, i ran the tests by using this command:
bundle exec rake plugin:spec["discourse-github-linkback"]
and, ofc, i was in ~/discourse when doing that.

Thanks for all your help @sam!

(1st PR for discourse-plugin babyyyyy :tada:)

8 лайков

My Qunit-Tests aren’t executed or at least they don’t show me any output.
They finish with the following:

No output has been received in the last 10m0s, this potentially indicates a stalled build or something wrong with the build itself.
Check the details on how to adjust your build configuration on: https://docs.travis-ci.com/user/common-build-problems/#Build-times-out-because-no-output-was-received

Travis.CI Result - Travis CI
Test-File - GitHub - kokoro-ko/discourse-humble-box: humble-bundle onebox · GitHub

I am not that skilled with CI yet and haven’t found my mistake by now.

Do your tests pass if you visit /qunit on your development environment? How about if you run rake plugin:qunit[discouse-humble-box]?

2 лайка