Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/ebellocchia/factories_injector
C++ Template-based abstract factories injector
https://github.com/ebellocchia/factories_injector
abstract-factory abstract-factory-pattern cpp-library cpp14 factories-injector
Last synced: 2 months ago
JSON representation
C++ Template-based abstract factories injector
- Host: GitHub
- URL: https://github.com/ebellocchia/factories_injector
- Owner: ebellocchia
- License: mit
- Created: 2020-03-25T20:55:07.000Z (almost 5 years ago)
- Default Branch: master
- Last Pushed: 2024-02-09T21:33:06.000Z (11 months ago)
- Last Synced: 2024-10-15T22:21:16.149Z (3 months ago)
- Topics: abstract-factory, abstract-factory-pattern, cpp-library, cpp14, factories-injector
- Language: C++
- Homepage:
- Size: 54.7 KB
- Stars: 2
- Watchers: 2
- Forks: 5
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# C++ Factories injector
| |
|---|
| [![GitHub License](https://img.shields.io/github/license/ebellocchia/factories_injector?label=License)](https://github.com/ebellocchia/factories_injector?tab=MIT-1-ov-file) |
| [![Build & Test](https://github.com/ebellocchia/factories_injector/actions/workflows/built-test.yml/badge.svg)](https://github.com/ebellocchia/factories_injector/actions/workflows/built-test.yml) |
| [![Codecov](https://img.shields.io/codecov/c/github/ebellocchia/factories_injector?label=Code%20Coverage)](https://codecov.io/gh/ebellocchia/factories_injector) [![Codacy grade](https://img.shields.io/codacy/grade/a9ba229090be499fb27380aabcbc635b?label=Codacy%20Grade)](https://app.codacy.com/gh/ebellocchia/factories_injector/dashboard?utm_source=gh&utm_medium=referral&utm_content=&utm_campaign=Badge_grade) [![CodeFactor Grade](https://img.shields.io/codefactor/grade/github/ebellocchia/factories_injector?label=CodeFactor%20Grade)](https://www.codefactor.io/repository/github/ebellocchia/factories_injector) |
| |## Introduction
This is a template-based utility library composed of just a couple of header files, hence it doesn't need to be compiled and linked.\
It allows run-time injection of factories, making the code completely independent from specific factory types and only dependent on factory interfaces.\
In this way, you can use a real factory producing real objects in the production code, while in the testing code you can dynamically inject a factory producing test objects (e.g. stubs or mocks) without modifying a single line.\
For being injected by the library, the factories shall follow the [abstract factory pattern](https://en.wikipedia.org/wiki/Abstract_factory_pattern) and inherit from the *FactoryTraits* structure defined by the library.## Installation
As already said, you just have to include the header files. There is no dependency and nothing to compile, since the entire code is based on templates.\
The code is compliant to C++14 standard, so you need a compiler supporting C++14.To compile the example and the unit tests, just run:
cmake .
makeIn the folder where the *CMakeLists.txt* file is located.\
A *bin* folder will be created inside with the compiled files.**NOTE:** for compiling unit tests, you need *googletest* to be installed.
## Usage
To inject a factory, you have to create a factory interface (*abstract factory*) that inherits from the *FactoryTraits* structure, by specifying its type and the interface of the object to be created.
**Example**
// Object interface
class IObj
{};// Abstract factory for IObj
class IObjFactory : public factory_injector::FactoryTraits
{
public:
virtual ~IObjFactory(void) = default;// tObjectPtr is defined by FactoryTraits as std::unique_ptr
virtual tObjectPtr Create(/* Some parameters */) const = 0;
};Then you can create your real factory classes (*concrete factories*) inheriting from the abstract factory and creating real objects.
**Example**
// A real object, used for production
class MyRealObj : public IObj
{};// A test object, used for testing
class MyTestObj : public IObj
{};// Another test object, used for some other types of tests
class AnotherTestObj : public IObj
{};// Concrete factory producing MyRealObj
class MyRealObjFactory : public IObjFactory
{
public:
tObjectPtr Create(/* Some parameters */) const override
{
return std::make_unique(/* Some parameters */);
}
};// Concrete factory producing MyTestObj
class MyTestObjFactory : public IObjFactory
{
public:
tObjectPtr Create(/* Some parameters */) const override
{
return std::make_unique(/* Some parameters */);
}
};// Concrete factory producing AnotherTestObj
class AnotherTestObjFactory : public IObjFactory
{
public:
tObjectPtr Create(/* Some parameters */) const override
{
return std::make_unique(/* Some parameters */);
}
};Finally, you have to register the concrete factory you want to use with one of the following methods:
- *FactoryInjector::RegisterFactory()* will register the factory type only if not already registered, otherwise a *FactoryAlreadyRegisteredEx* exception is thrown
- *FactoryInjector::OverwriteFactory()* will register the factory type in any case, overwriting it if already registeredTo get back a factory, you can use the *FactoryInjector::GetFactory()* method. Here you can either specify the concrete or the abstract factory type, but it's suggested to use the abstract one so that the code is completely independent from concrete factories.\
In case you try to get a factory type that is not registered, a *FactoryNotRegisteredEx* will be thrown.**Example**
// You can wrap the factory injector in a singleton to make it globally accessible, or pass it around the classes that need it
factory_injector::FactoryInjector fi;// Register MyRealObjFactory. From now on, all the code now will use MyRealObjFactory when calling GetFactory, this producing MyRealObj
// OverwriteFactory can also be used here, it's the same
fi.RegisterFactory();//
// Somewhere else in the production code...
//// Get the factory by the abstract type
// It will get MyRealObjFactory, MyTestObjFactory or AnotherTestObjFactory depending on what it's registered
auto& obj_factory = fi.GetFactory();
// Let's produce some object instances. MyRealObj, MyTestObj or AnotherTestObj will be produced depending on the registered factory.
// In this way, the code is independent from the concrete factory, which can be injected dynamically.
for (int i = 0; i < 10; i++)
{
// obj is of type std::unique_ptr
auto obj = obj_factory.Create(/* Some parameters */);
// Do something with the object
}// Alternatively, there is a helper method called CreateObject. It simply calls internally the factory Create method, if defined.
// It's just a shortcut, if you a different name for the method or you need to use the factory multiple times (e.g. in a loop),
// just use the GetFactory as described previously
auto obj = fi.CreateObject(/* Some parameters */);//
// In a test code, we need to use MyTestObj
//// Register MyTestObjFactory (MyRealObjFactory will be overwritten if registered)
fi.OverwriteFactory();
// Now, the previous code will use MyTestObj objects instead of MyRealObj//
// In another test code, we need to use AnotherTestObj
//// Register AnotherTestObjFactory
fi.OverwriteFactory();
// Now, the previous code will use AnotherTestObj objects instead of MyRealObjOf course, you can register as many factory types as you want, as long as they inherit from a different interface.
## How it works
The base concept is quite simple. The *FactoryInjector* class is keeping track of the registered types by means of a hash table.\
Using the *FactoryTraits*, the class deducts the interface type of the concrete factory. Then, it creates an instance of the concrete factory and inserts it in the hash table, associating it with the *typeid* of the abstract factory. Therefore, if you register another concrete factory inheriting from the same abstract factory, it will overwrite the existent one because the *typeid* of the abstract factory is always the same and it is used as a key of the hash table.\
When you get a factory, a reference to the concrete instance is simply returned by searching for the *typeid* of the abstract factory in the hash table.\
The library takes advantage of move semantics, perfect forwarding and variadic templates to be as most generic as possible.## License
This software is available under the MIT license.