- Developer Readme for 4Minitz
- We use this successful git branching strategy
- For generating the UML figures in this doc we use PlantUML (respective the PlantUML plugin for JetBrains products)
- We use ES2015 (ES6) as we make progress in learning it.
- Before merging new features, read our "Definition of Done"
- Translations are managed via Crowdin. Don't manually edit any language file other than both/i18n/en.i18n.yml (see below)!
- Acceptance critera: from the user story are checked and work OK
- Responsiveness: Your user story shall look great on small (<= 480px width) and large (>1000px) screens
- Unit tests: do cover your new code
- Documentation updated: the developer doc and the user doc are updated
- Code Quality: the code passes our eslint rules and no eslint errors will be thrown
- No client-side exceptions: Check your user story and watch the browser console
- No server-side exceptions: Check your user story and the meteor console
- No open issues: on your user story
- Create issues: if your tests discover issues in other already implemented user stories: write a github issue
- ES2015: Where possible we use ES2015 language features
- Localization: All UI strings are added to both/i18n/en.i18n.yml with their corresponding ressource key
We use several approaches to test as many aspects of our application as possible.
- Make sure you have meteor installed. Run
meteor --version
and check which node / npm meteor uses internallymeteor node --version
thenmeteor npm --version
(note that down!) - Install the node version manager
nvm
to quickly switch between node versions. - Install a global node & npm version from https://nodejs.org/en/ (so that you can run our package.json test runner scripts)
- Install C/C++ commandline compiler (so that you can build native node modules with node-gyp)
- Windows: Install Microsoft's windows-build-tools using:
npm install --global --production windows-build-tools
from an elevated PowerShell or CMD.exe (run as Administrator) - MacOS: Run in shell:
xcode-select --install
- Windows: Install Microsoft's windows-build-tools using:
- If not yet done, install all 4Minitz dependencies:
meteor npm install
- Install chrome-driver
- Check version number of you installed chrome browser
- If your
node
version is less than 12.x temporarily switch withnvm use 12
npm i chromedriver
or specific versionnpm i chromedriver@^84.0.0
./node_modules/chromedriver/bin/chromedriver --version
check your chrome driver version
Unit tests are stored in tests/unit
. They are implemented so they do not use any meteor dependencies at all. In order
to execute unit tests run
npm run test:unit
End-to-End tests are stored in directory tests/end2end/
.
To run the e2e tests, you need to run the server in "end2end" mode.
`npm run test:end2end:server`
This will set some specific e2e settings from settings-test-end2end.json
.
After the end2end server is up and running you may run all(!) end2end tests in either of the following ways:
npm run test:end2end # visible browser
npm run test:end2end:headless # invisible browser
npm run test:end2end:watch # visible browser, tests re-run when any test code changes
npm run test:end2end:watchheadless # invisible browser, tests re-run when any test code changes
To run any of the above scripts with a specific test file, you may append that test spec like so:
npm run test:end2end:watch -- --spec=tests/end2end/MeetingSeries-test.js
To debug headless testcases that run red, you can create screenshots at interesting timepoints in your tests by either calling WebDriver.io's:
browser.saveScreenshot("my-testcase");
or by calling our convenience wrapper function:
E2EGlobal.saveScreenshot("my-testcase");
where the later one adds a timestamp (YYYYMMDDHHMMSS plus milliseconds) in front
of the file name to create unique names. The screenshots of the headless
browser are stored in /tests/snapshots
.
Headless end2end tests are executed via GitHub Actions. Screenshots are exported as github action artifacts with every build run.
We use eslint to make static code analysis to keep our code quality constantly high. Our set of rules is defined in the .eslintrc.json configuration file.
You can run eslint to check the code by executing the npm script eslint
:
npm run eslint
Some rules can be applied automatically like correcting file indention or transforming double quoted literals into
single quoted ones (yes we use single quotes like recommended in the
Google JavaScript Style Guide
together with template strings if necessary). To automatically fix all those errors which can be fixed automatically
run the npm script eslint:fix
:
npm run eslint:fix
We use Crowdin for managing translations. Resource keys are added to both/i18n/en.i18n.yml following the naming convention:
- PascalCase for nodes
- camelCase for leafs
After your changes were merged to the develop
branch, you will be able to translate the UI strings directly within Crowdin.
Any changes on Crowdin are synced every ten minutes to GitHub. Our faithful servant 4minitz-bot will
automatically commit these changes to the branch l10n_develop
and create a PR to develop
. Don't manually edit any translation file or it will be overwritten with the next PR.
Sometimes it is necessary that there are some test data available in the database. Therefore you can use our minutes seeding tool. This tool allows to insert a meeting series with dummy entries. The series contains a custom amount of minutes containing a custom amount of topics and items.
Simply execute:
npm run fixtures:seed-minutes
The parameter --help
displays all command-line options:
npm run fixtures:seed-minutes -- --help
Our work-horses are the classes in /imports/ (e.g. meetingseries.js, minutes.js). They build a facade for the underlying MongoCollections and enrich them with convenience methods.
We provide different ways for sending emails. Currently sending mails via smtp or the mail delivery service mailgun is supported. Via the configuration file settings.json you can define globally which service should be used (for configuration go to Admin Doc).
To achieve this you should not use Meteor.mail() for sending mails directly. Instead you can use our MailFactory to get a concrete instance of the abstract Mail-Class. This class defines the API for sending mails and is currently implemented by a class called MeteorMail which uses the meteor api for sending emails and a class MailgunMail which uses the mailgun api for sending emails.
You can get a concrete Mail-Instance via the static method getMailer(replyTo, recipients) of the MailFactory. You have to provide the two parameters replyTo (string) and recipients. Recipients can either be a string representing an email address or an array of email addresses. As a result you will ge a Mail-Object.
With this object you can prepare your email by providing a subject, setting a email text as either plain text or html or both. After that you can send the email by calling the Method send() of the Mail-Object.
When sending the mail by calling the send method, the abstract class Mail takes care about some general things like checking if the necessary parameters are set. Then it calls the abstract method _sendMail() which will be implemented by the concrete implementation either MeteorMail or MailgunMail which takes care about actually sending the mail. Afterwords the send-method of the Mail-Class takes care about logging the result.
The following sequence diagram shows the process of sending mails using the concrete meteor mail object as an example. But you should never use the constructor of a Mail class directly. Instead you should always use the MailFactory to get an instance of a Mailer.
If you want to add an additional email service you have to write your own class which derives from the abstract class Mail. In your class you have to implement the method _sendMail() only. There you can access the attributes _replyTo, _recipients, _from, _html and _text for building and sending the mail. Then you have to adjust the MailFactory so that it will return your implementation depending on the current configuration. Do not forget to adjust the settings_sample.json file to draw attention to your new email delivery service.
For sending info items or action items you can use the classes InfoItemsMailHandler and ActionItemsMailHandler, respectively. Both classes are derivations from the abstract class TopicItemsMailHandler. This class provides the base features for sending action or info items.
The following sequence diagram shows the process of sending info items. If you pass multiple recipients to the InfoItemsMailHandler depending on the isTrustedIntranetEnvironment configuration the info items will be sent to all recipients in one mail (all recipients in the TO-field) or if the system is not in a trusted intranet environment the info items will sent to all recipients separately. Sending action items works analog.
The email layout will be provided as a spacebars template. You can pass a template name to the TopicItemsMailHandler which must be found under the path "/private/server_side_templates/email" (e.g InfoItemsMailHandler uses the template "publishInfoItems"). In a concrete derivation of the base class you have to call the method _buildMail() of the base class and pass the subject and the data required by the template as parameters. The method _buildMail() will then take care about rendering the template and sending the email.
See *.schema.js files in imports/collections.
Migrations are realized with the percolate:migrations package. To adjust existing databases to a new schema add new migration up() and down() functions to /server/migrations.js.
If you have to generate an absolute URL (e.g. for links in emails) do not use Meteor.absoluteUrl() ! Instead use GlobalSettings.getRootUrl(path [optional]). This method does all the necessary stuff for you ;-)