The GNU social [Component-based architecture](https://en.wikipedia.org/wiki/Component-based_software_engineering)
provides a clear distinction between what can not be changed (**[core](http://foldoc.org/core)**), what is replaceable
but must be always present (**[component](http://foldoc.org/component)**), and what can be removed or
added (**[plugin](http://foldoc.org/plugin)**).
This architecture has terminology differences when compared to the one that was [introduced in v2](https://agile.gnusocial.rocks/doku.php?id=v2modules).
In fact, back in v2 - as the term `modules` is not necessarily non-essential - we would keep a "modules" directory near
"plugins", to make the intended difference between both self-evident.
Now in v3, **[Module](http://foldoc.org/module)** is the name we give to the core system managing all the `modules` (as
it is broad enough to include both components and plugins). N.B.: there are not `modules` in the same sense as
there are `components` and `plugins`, the latter descend from the former.
### Components
The most fundamental modules are the components. These are non-core functionality expected to be always available.
Unlike the core, it can be exchanged with equivalent components.
We have components for two key reasons:
- to make available internal higher level APIs, i.e. more abstract ways of interacting with the Core;
- to implement all the basic/essential GNU social functionality in the very same way we would implement plugins.
Currently, GNU social has the following components:
* The Core only depends on Symfony. We wrote wrappers for all the Symfony functionality we use, making it possible to
replace Symfony in the future if needed and to make it usable under our
[programming paradigms, philosophies and conventions](./paradigms.md). V2 tried to do this with PEAR.
* Components only depend on the Core. The Core never depends on Components.
* Components never have inter-dependencies.
* Plugins can depend both on the Core and on Components.
* A plugin may recognize other plugin existence and provide extra functionality via events.
N.B.: "depend on" and "allowing to" have different implications. A plugin can throw an event and other plugins may
handle such event. On the other hand, it's **wrong** if:
* two plugins are inter-dependent in order to provide all of their useful functionality - consider adding [configuration to your plugin](./plugins/configuration.md);
* a component depends on or handles events from plugins - consider throwing an event from your component replacement and
then handling it from a plugin.
This "hierarchy" makes the flow of things perceivable and predictable, that helps to maintain sanity.