Feedback and contributions are very welcome!
Here's help on how to make contributions, divided into the following sections:
The quick-read part:
- general information,
- vulnerability reporting,
- documentation.
The long-read part:
- code changes,
- how to check proposed changes before submitting them,
- reuse of other libraries, frameworks etc.
For specific proposals, please provide them as pull requests or issues For general discussion, feel free to use our Matrix room: #quotient:matrix.org.
If you're new to the project (or FLOSS in general), here's the list of issues that we consider simple enough for starters - they don't require much knowledge about the project and should be relatively straightforward (but not necessarily quick to fix). Welcome aboard!
Contributions are preferred in the form of pull requests at GitHub. See the respective articles about pull requests to learn how to deal with them.
We recommend creating different branches for different (logical)
changes, and creating a pull request when you're done; the development
integration branch is dev
. See the GitHub documentation on
creating branches
and
using pull requests.
We use GitHub to track all changes via its issue tracker and pull requests. Specific changes are proposed using those mechanisms. Issues are assigned to an individual who works on it and then marks it complete. If there are questions or objections, the conversation area of that issue or pull request is used to resolve it.
Unless a contributor explicitly specifies otherwise, we assume contributors to agree that all contributed code is released either under LGPL v2.1 or later. This is more than just LGPL v2.1 libQuotient now uses because the project plans to switch to LGPL v3 for library code in the near future.
Any components proposed for reuse should have a license that permits releasing a derivative work under LGPL v3 or later (that includes licenses permitting LGPL v2.1 or later but not LGPL v2.1 only). In any case, the component should be redistributable under a license from the list approved by OSI, no exceptions.
We use SPDX conventions for copyright statements. Please follow them when making a sizable contribution: add your name and year to the top of the file. New files should begin with the following preamble:
// SPDX-FileCopyrightText: 2021 Your Name <[email protected]>
// SPDX-License-Identifier: LGPL-2.1-or-later
Vulnerability reporting (security issues) - see SECURITY.md
Most of the documentation is in Markdown format. All Markdown files use the .md filename extension. Any help on fixing/extending these is more than welcome.
Where reasonable, limit yourself to Markdown that will be accepted by different markdown processors (e.g., what is specified by CommonMark or the original Markdown). In practice, as long as libQuotient is hosted at GitHub, GFM (GitHub-flavoured Markdown) is used to show those files in a browser, so it's fine to use its extensions. In particular, you can mark code snippets with the programming language used; blank lines separate paragraphs, newlines inside a paragraph do not force a line break.
Be mindful that this is not the same (albeit similar) markdown algorithm used by GitHub when it renders issue or pull comments; in those cases newlines in paragraph-like content are considered as real line breaks; unfortunately this other algorithm is also called GitHub-flavoured markdown. (Yes, it would be better if there were different names for different things.)
In your markdown, please don't use tab characters and avoid "bare" URLs. In a hyperlink, the link text and URL should be on the same line. Both in C/C++ code comments and Markdown documents, try to keep your lines within the 80-character limit except hyperlinks (wrapping breaks them). Some historical text may not follow that rule - feel free to reformat those parts when you edit them.
Do not use trailing two spaces for line breaks, since these cannot be seen and may be silently removed by some tools. If, for whatever reason, a blank line is not an option, use <br/> (an HTML line break).
If you don't plan/have substantial contributions, you can stop reading here. Further sections are for those who's going to actively hack on the library code.
The code should strive to be DRY (don't repeat yourself), clear, and obviously correct (i.e. buildable). Some technical debt is inevitable, just don't bankrupt us with it. Refactoring is welcome.
As of Quotient 0.7, the C++ standard for newly written code is C++20. Since none of the supported compilers (GCC 11, Clang 11, Apple Clang 12, MSVC 19.30 - see also the pre-requeisites in README) can handle the entire C++20 feature set, we have to stick with a subset. Most notably:
std::bind_front
, ranges,std::format
,std::source_location
can't be used yet;- while concepts and constraints as a language feature is there, some library
concepts are not available (
std::derived_from
, in particular); std::string
andstd::vector
are not constexpr.
This page at cppreference.com can be used to check whether a given feature is there.
The code style is defined by .clang-format
, and in general, all C++ files
should follow it. Files with minor deviations from the defined style are still
accepted in PRs; however, unless explicitly marked with // clang-format off
and // clang-format on
, these deviations will be rectified any commit soon
after.
Notable things from .clang-format:
- 4-space indents, no tabs, no trailing spaces, no last empty lines. If you spot the code abusing these - thank you for fixing it.
- Prefer keeping lines within 80 characters. Slight overflows are ok only
if that helps readability. Ideally, just use
.clang-format
to format lines.
Additionally:
-
Do not use
struct
next to protected or private members; try to use it with plain-old-data structures, with maybe just a function or two among public members. If you need access control or are adding yet another non-trivial (construction, assignment) member function to astruct
, make it aclass
instead. -
For newly created classes, keep to the rule of 3/5/0 - make sure to read about the rule of zero if you haven't before, it's not what you might think it is.
-
Qt containers are generally preferred to STL containers; however, there are notable exceptions, and libQuotient already uses them:
std::array
andstd::deque
have no direct counterparts in Qt.- Because of COW semantics, Qt containers cannot hold uncopyable classes.
Classes without a default constructor are a problem too. Examples of that
are
SyncRoomData
andEventsArray<>
. Use STL containers for structures having those but see the next point and also consider if you can supply a reasonable copy/default constructor. - STL containers can be freely used in code internal to a translation unit
(i.e., in a certain .cpp file) as long as that is not exposed in the API.
It's ok to use, e.g.,
std::vector
instead ofQVector
to tighten up code where you don't need COW, or when dealing with uncopyable data structures (see the previous point). However, exposing STL containers in the API is not encouraged (except where absolutely necessary, e.g. we usestd::deque
for a timeline). Especially when it comes to API intended for usage from QML (e.g.Q_PROPERTY
), STL containers or iterators are unlikely to work and therefore unlikely to be accepted intodev
. - Notwithstanding the above (you're not going to use these with QML anyway),
prefer
std::unique_ptr<>
overQScopedPointer<>
as it gives stronger guarantees; also, some features ofQScopedPointer
are deprecated in Qt 6.
-
Always use
QVector
instead ofQList
unless Qt's own API uses it - see the great article by Marc Mutz on Qt containers for details. With Qt 6, these two become the same type matching what used to beQVector
in Qt 5.(Note: unfortunately, QVector is a type alias in Qt 6 and that breaks templated code because type deduction doesn't work with aliases. This breakage will go away in C++23 but we are far from that yet; in the meantime, the fix boils down to specifying the template parameter of
QVector
explicitly.)
All non-inline symbols (functions, classes/structs, even namespace-level static
variables) that are intended for use in client code must be declared with
QUOTIENT_API
macro (the macro itself is defined in the dedicated
quotient_export.h
file). This is concerned with symbols visibility in
dynamic/shared libraries: the macro marks these symbols for exporting in
the library symbol table. Without that clients will get, usually, linkage errors
(but in some cases, only obscure runtime errors, such as split-brained singleton
instances) once they attempt to use those symbols. You only need to use
the macro on the namespace level; inner symbols (member functions, e.g.) are
exported if their class is exported.
On the other hand - it sounds obvious but calls, data structures and other
symbols not intended for usage in client code should not be exposed in
(public) .h files, unless they are necessary to declare other public symbols.
In particular, this involves private members (functions, typedefs, or variables)
in public classes; use pimpl idiom to hide implementation details as much
as possible. This is to make it easier to change the inside machinery without
breaking the API and ABI stability guarantees within a single major version.
For cases when a definition has to be in the header file but should not be used
in client code, use _impl
namespace - anything in that namespace is
not covered by API guarantees.
Note: As of Quotient 0.7, all header files of libQuotient are considered public; this may change eventually.
Whenever you add a new call to the library API that you expect to be used
from client code, make sure to supply a proper doc-comment along with the call.
Quotient uses the Doxygen C++-styled doc-comments (//!
, \brief
); some legacy
code may use Javadoc (/** ... */
, @brief
) or C-styled Doxygen (/*! ... */
)
but that is not encouraged any more. Some parts are not documented at all;
adding doc-comments to them and/or converting the existing ones to the assumed
style is highly encouraged; it's also a nice and easy first-time contribution.
Use \brief
for the summary, and follow with details after
an empty doc-comment line, using \param
, \return
etc. as necessary.
When commenting in-code:
- Don't restate what's happening in the code unless it's not really obvious. We assume the readers to have some command of C++ and Qt. If your code is not obvious, consider making it clearer itself before commenting.
- That said, both C++ and Qt have their arcane/novel features and dark corners, and education of code readers is a great thing. Use your experience to figure what might be less well-known to readers and comment such cases: leave references to web pages, Quotient wiki etc.
- Make sure to document not so much "what" but more "why" certain code is done the way it is. In the worst case, the logic of the code can be reverse-engineered; but you can almost never reverse-engineer the line of reasoning and the pitfalls avoided.
We gradually introduce autotests based on a combination of CTest and Qt Test
frameworks - see autotests/
directory. There are very few of those, as we
have just started adding those to the new code (you guessed it; adding more
tests to the old code is very welcome).
Aside from that, libQuotient comes with a command-line end-to-end test suite
called Quotest. Any significant addition to the library API should be
accompanied by a respective test in autotests/
and/or in Quotest.
To add a test to autotests:
- In a new .cpp file in
autotests/
, define a test class derived from QObject and write tests as member functions in itsprivate slots:
section. If you feel more comfortable using a header file to define the class, that's also fine but not necessary. If you're new to Qt Test framework, use existing tests as a guidance. - Add a
quotient_add_test
macro call with your test toautotests/CMakeLists.txt
To add a test to Quotest:
- In
quotest.cpp
, add a new test to theTestSuite
class. Similar to Qt Test, each test in Quotest is a private slot; unlike Qt Test, you should use special macros,TEST_DECL()
andTEST_IMPL()
, to declare and define the test (those macros conceal passing the testing handle inthisTest
variable to the test method). - In the test function definition, add test logic using
FINISH_TEST
macro to check for the test outcome and complete the test (be mindful thatFINISH_TEST
alwaysreturn
s, not only in case of error). ALL (even failing) branches should conclude with aFINISH_TEST
(orFAIL_TEST
that is a shortcut for a failingFINISH_TEST
) invocation, unless you want the test to fail with a "DID NOT FINISH" message in the logs under certain conditions.
The TestManager
class sets up some basic test fixture to help you with testing;
notably, the tests can rely on having an initialised Room
object with loaded
state for the test room in targetRoom
member variable. Note that it's normal
for tests to go async, which is not something Qt Test is easy with (and this
is why Quotest doesn't directly use Qt Test but rather fetches a few ideas
from it).
Pay attention to security, and work with, not against, the usual security hardening practices.
char *
and similar unchecked C-style read/write arrays are forbidden - use
Qt containers or at the very least std::array<>
instead. Where you see fit
(usually with data structures), try to use smart pointers, especially
std::unique_ptr<>
, instead of bare pointers. When dealing with QObject
s,
organise them in parent-child trees and let Qt manage object lifecycles
for you (this in turn is preferred to using smart pointers). If you find
a particular use case where the strict semantic of unique pointers doesn't help
and a shared pointer is necessary, feel free to step up with the working code
and it will be considered for inclusion.
Exercise the principle of least privilege where reasonable and appropriate. Prefer internally cohesive code while avoiding too much external coupling (such as calling into a bunch of classes at several points).
Protect private information, in particular passwords and email addresses. Absolutely don't spill around this information in logs, and only display those in UI where really needed. Do not forget about local access to data (in particular, be very careful when storing something in temporary files, let alone permanent configuration or state). Avoid mechanisms that could be used for tracking where possible (we do need to verify people are logged in but that's pretty much it), and ensure that third parties can't use interactions for tracking. Matrix protocols evolve towards decoupling the personally identifiable information from user activity entirely - follow this trend.
We want the software to have decent performance for users even on weaker
machines. At the same time we keep libQuotient single-threaded as much
as possible, to keep the code simple. That means being cautious about operation
complexity (read about big-O notation if you need a kickstart on the subject).
This especially refers to operations on the whole timeline and the list of
users - each of these can have tens of thousands of elements so even operations
with linear complexity, if heavy enough (with I/O or complex processing),
can produce noticeable GUI freezing or stuttering. When you don't see a way
to reduce algorithmic complexity, either split processing into isolated
pieces that can be individually scheduled as queued events (see the end of
Connection::consumeRoomData()
to get the idea) or uncouple the logic from
GUI and execute it outside of the main thread with QtConcurrent
facilities.
Having said that, there's always a trade-off between various attributes; in particular, readability and maintainability of the code is more important than squeezing every bit out of that clumsy algorithm. Beware of premature optimization and profile the code before before diving into hardcore tweaking that might not give the benefits you think it would.
Speaking of profiling logs (see README.md on how to turn them on) - if you
expect some code to take considerable (more than 10k "simple operations") time
you might want to setup a QElapsedTimer
and drop the elapsed time into logs
under PROFILER
logging category. See the existing code for examples -
room.cpp
has quite a few. In order to reduce small timespan logging spam,
PROFILER
log lines are usually guarded by a check that the timer counted big
enough time (200 microseconds by default, 20 microseconds for tighter parts);
this threshold can be altered at compile-time by defining PROFILER_LOG_USECS
preprocessor symbol (i.e. passing -DPROFILE_LOG_USECS=<usecs>
to the compiler
if you're on Linux/macOS).
The code in lib/csapi
, lib/identity
and lib/application-service
, although
stored in Git, is actually generated from the official Matrix Client-Server API
definition files. If you're unhappy with something in there and want to improve
that, you have to understand the way these files are produced and setup some
additional tooling. The shortest possible procedure resembling the below text
can be found in .github/workflows/ci.yml (regeneration of those files is tested
in CI). As described below, there is also a handy build target for CMake.
Because otherwise we have to do monkey business of writing boilerplate code, with the same patterns, types etc., literally, for every single API endpoint, and one of libQuotient authors got fed up with it at some point in time. By then about 15 job classes have been written; the whole API is about 100 endpoints, with new ones added and some existing changed in each Matrix protocol version. Other considerations can be found in this talk about API description languages that also briefly touches on GTAD - the tool written for the purpose.
-
Get the source code of GTAD and its dependencies. Since version 0.7, libQuotient includes GTAD as a submodule so you can get everything you need by updating gtad/gtad submodule in libQuotient sources:
git submodule update --init --recursive gtad/gtad
.You can also just clone GTAD sources to keep them separate from libQuotient:
git clone --recursive https://github.com/quotient-im/gtad.git
-
Configure and build GTAD: same as libQuotient, it uses CMake so this should be quite straightforward (if not - you're probably not quite ready for this stuff anyway).
-
Get Matrix CS API definitions from a matrix-spec repo. Although the official repo is at https://github.com/matrix-org/matrix-spec.git` (formerly https://github.com/matrix-org/matrix-doc.git), you may or may not be able to generate working code from it because the way it evolves is not necessarily in line with libQuotient needs. For that reason, a soft fork of the official definitions is kept at https://github.com/quotient-im/matrix-spec.git that guarantees buildability of the generated code. This repo closely follows the official one (but maybe not its freshest commit), applying a few adjustments on top. And of course you can use your own repository if you need to change the API definition.
-
If you plan to submit a PR with the generated code to libQuotient or just would like it to be properly formatted, you should either ensure you have clang-format (version 10 at least) in your PATH or pass
-DCLANG_FORMAT=<path>
to CMake, as mentioned in the next section.
- Pass additional configuration to CMake when configuring libQuotient:
-DMATRIX_SPEC_PATH=/path/to/matrix-spec/ -DGTAD_PATH=/path/to/gtad
. Note thatMATRIX_SPEC_PATH
should lead to the repo (not the API folder) whileGTAD_PATH
should have the path to GTAD binary. If you need to specify where your clang-format is (see the previous section) add-DCLANG_FORMAT=/path/to/clang-format
to the line above. If everything's right, the detected locations will be mentioned in CMake output and an additional build target calledupdate-api
will be configured. - Generate the code:
cmake --build <your build dir> --target update-api
. Building this target will create (overwriting without warning) source files inlib/csapi
,lib/identity
,lib/application-service
for all YAML files it can find in/path/to/matrix-spec/data/api/client-server
and their dependencies.
See the more detailed description of what GTAD is and how it works in the documentation on GTAD in its source repo. Only parts specific for libQuotient are described here.
GTAD uses the following three kinds of sources:
- OpenAPI files. Each file is treated as a separate source (unlike swagger-codegen, you do not need to have a single file for the whole API).
- A configuration file, in Quotient case it's
gtad/gtad.yaml
- common for all OpenAPI files GTAD is invoked on. - Source code template files:
gtad/*.mustache
- are also common.
The Mustache files have a templated (not in C++ sense) definition of a network
job class derived from BaseJob; if necessary, data structure definitions used
by this job are put before the job class. Bigger Mustache files look a bit
hideous for a newcomer; and the only known highlighter that can handle
the combination of Mustache (originally a web templating language) and C++ can
be found in CLion IDE. Fortunately, all our Mustache files are reasonably
concise and well-formatted these days.
To simplify things some reusable Mustache blocks are defined in gtad.yaml
-
see its mustache:
section. Adventurous souls that would like to figure
what's going on in these files should speak up in the Quotient room -
I (Kitsune) will be very glad to help you out.
The types
map in gtad.yaml
defines a mapping from OpenAPI types to C++/Qt.
It uses the following type attributes aside from pretty obvious imports:
:
avoidCopy
- this attribute defines whether a const ref should be used instead of a value. For basic types like int this is obviously unnecessary; but compound types likeQVector
should rather be taken by reference when possible.moveOnly
- some types are not copyable at all and must be moved instead (an obvious example is any structure that uses, directly or indirectly,std::unique_ptr<>
).useOmittable
- wrap types that have no value with "null" semantics (i.e. number types and custom-defined data structures) into a specialOmittable<>
template defined inconverters.h
, a drop-in upgrade overstd::optional
.omittedValue
- an alternative foruseOmittable
, just provide a value used for an omitted parameter. This is used for bool parameters which normally are considered false if omitted (or they have an explicit default value, passed in the "official" GTAD'sdefaultValue
variable).initializer
- this is a partial (see GTAD and Mustache documentation for explanations but basically it's a variable that is a Mustache template itself) that specifies how exactly a default value should be passed to the parameter. E.g., the default value for aQString
parameter is enclosed intoQStringLiteral
.
Instead of relying on the event structure definition in the OpenAPI files,
gtad.yaml
uses pointers to libQuotient's event structures: EventPtr
,
RoomEventPtr
and StateEventPtr
. Respectively, arrays of events, when
encountered in OpenAPI definitions, are converted to Events
, RoomEvents
and StateEvents
containers. When there's no way to figure the type from
the definition, an opaque QJsonObject
is used, leaving the conversion
to the library and/or client code.
If you changed the API definitions, the path to upstream becomes somewhat intricate, as you have to coordinate with two projects, making up to 4 PRs along the way. The recommended sequence depends on whether or not you have to write a Matrix Spec Change aka MSC. Usually you have to, unless your API changes keep API semantics intact. In that case:
- Submit an MSC before submitting changes to the API definition files and libQuotient.
- The MSC gets reviewed by the Spec Core Team. This can be a lengthy process but it's necessary for the Matrix ecosystem integrity.
- When your MSC has at least some approvals (not necessarily a complete
acceptance but at least some approvals should be there) submit a PR to
libQuotient, referring to your
matrix-spec
repo. Make sure that generated files are committed separately from non-generated ones (no need to make two PRs; just separate them in different commits). - If/when your libQuotient PR is approved and MSC is not there yet you'll
be asked to submit a PR with API definition files at
https://github.com/quotient-im/matrix-spec
. Note that this is not an official repo; but you can refer to your libQuotient PR as an implementation of the MSC - a necessary step before making a so-called "spec PR". - Once MSC is accepted, submit your
matrix-spec
changes as a PR tohttps://github.com/matrix-org/matrix-spec
(the "spec PR" mentioned above). This will require that your submission meets the standards set by this project (they are quite reasonable and not too hard to meet).
If your changes don't need an MSC, it becomes a more straightforward combination
of 2 PRs: one to https://github.com/matrix-org/matrix-spec
("spec PR") and one
to libQuotient (with the same guidance about putting generated and non-generated
files in different commits).
Checking the code on at least one configuration is essential; if you only have a hasty fix that doesn't even compile, better make an issue and put a link to your commit or gist into it (with an explanation what it is about and why).
The warnings configuration applied when using CMake can be found in
CMakeLists.txt
. Most warnings triggered by that configuration are not formally
considered errors (the compiler will keep going) but please treat them as such.
If you want the IDE to be really picky about your code you can use
the following line for the Clang analyzer code model to enable most compiler
warnings while keeping the number of false positives at bay (that does not
include clang-tidy
/clazy
warnings - see below on those):
-Weverything -Werror=return-type -Wno-c++98-compat -Wno-c++98-compat-pedantic -Wno-unused-macros -Wno-newline-eof -Wno-exit-time-destructors -Wno-global-constructors -Wno-gnu-zero-variadic-macro-arguments -Wno-documentation -Wno-missing-prototypes -Wno-shadow-field-in-constructor -Wno-padded -Wno-weak-vtables -Wno-unknown-attributes -Wno-comma -Wno-string-conversion -Wno-return-std-move-in-c++11
.
Recent versions of Qt Creator and CLion can automatically run your code through
clang-tidy. The source code contains .clang-tidy
file with the recommended
set of checks that doesn't give too many false positives.
Qt Creator in addition knows about clazy, a Qt-aware static analysis tool that
hunts for Qt-specific issues that are easy to overlook otherwise, such as
possible unintended copying of a Qt container. Most of clazy checks are relevant
to our code; here's the configuration line the author of this text is using:
level2,no-foreach,no-non-pod-global-static,no-range-loop-reference,no-ctor-missing-parent-argument,no-missing-qobject-macro,no-jni-signatures,no-qt-keywords,no-qt4-qstring-from-array
We use CI to check buildability and smoke-testing on Linux (GCC, Clang), MacOS (Clang), and Windows (MSVC). Every PR will go through these, and you'll see the traffic lights from them on the PR page. If your PR fails on any platform double-check that it's not your code causing it - and fix (or ask how to fix if you don't know) if it is.
When writing git commit messages, try to follow the guidelines in How to Write a Git Commit Message:
- Separate subject from body with a blank line.
- Be reasonable on the subject line length, because this is what we see in commit logs that are constrained in horizontal space. Try to fit in 50 characters whenever possible.
- Capitalize the subject line.
- Do not end the subject line with a period.
- Aim to use the imperative mood in the subject line (command form).
- Use the body to explain what and why vs. how (git tracks how it was changed in detail, don't repeat that). Sometimes a quick overview of "how" is acceptable if a commit is huge - but maybe split a commit into smaller ones, to begin with?
C++ is unfortunately not very coherent about SDK/package management, and we try to keep building the library as easy as possible. Because of that we are very conservative about adding dependencies to libQuotient. That relates to additional Qt components and even more to other libraries. Fortunately, even the Qt components now in use (Qt Core and Network) are very feature-rich and provide plenty of ready-made stuff.
Some cases need additional explanation:
- Before rolling out your own super-optimised container or algorithm written from scratch, take a good long look through documentation on Qt and C++ standard library. Please try to reuse the existing facilities as much as possible.
- libQuotient is a library to build Qt applications; for that reason, components from KDE Frameworks should be really lightweight and useful to be accepted as a dependency. If the intention is to better integrate libQuotient into KDE environment there's nothing wrong in building another library on top of libQuotient. Consider people who run LXDE and normally don't have KDE frameworks installed (some even oppose installing those) - libQuotient caters to them too.
- Never forget that libQuotient is aimed to be a non-visual library; QtGui in dependencies is only there to allow (entirely offscreen) handling of QImages. While there's a bunch of visual code (in C++ and QML) shared between Quotient-enabled applications, this is likely to end up in a separate (Quotient-backed) library, rather than libQuotient itself.
This text is based on CONTRIBUTING.md from CII Best Practices Badge project, which is a collective work of its contributors (many thanks!). The text itself is licensed under CC-BY-4.0.