/* * Copyright (C) 2015 Cloudius Systems, Ltd. */ #pragma once class no_such_class : public std::runtime_error { public: using runtime_error::runtime_error; }; // BaseType is a base type of a type hierarchy that this registry will hold // Args... are parameters for object's constructor template class class_registry { using creator_type = std::function(Args...)>; public: static void register_class(sstring name, creator_type creator); template static void register_class(sstring name); static std::unique_ptr create(const sstring& name, Args&&...); static std::unordered_map& classes() { static std::unordered_map _classes; return _classes; } }; template void class_registry::register_class(sstring name, class_registry::creator_type creator) { classes().emplace(name, std::move(creator)); } template template void class_registry::register_class(sstring name) { register_class(name, [](Args&&... args) { return std::make_unique(std::forward(args)...); }); } template struct class_registrator { class_registrator(const sstring& name) { class_registry::template register_class(name); } class_registrator(const sstring& name, std::function(Args...)> creator) { class_registry::register_class(name, creator); } }; template std::unique_ptr class_registry::create(const sstring& name, Args&&... args) { auto it = classes().find(name); if (it == classes().end()) { throw no_such_class(sstring("unable to find class '") + name + sstring("'")); } return it->second(std::forward(args)...); } template std::unique_ptr create_object(const sstring& name, Args&&... args) { return class_registry::create(name, std::forward(args)...); }