-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathorchestration.hpp
198 lines (168 loc) · 6.13 KB
/
orchestration.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
/**
* \file
* Interfaces for orchestration of remote as well as local simulations.
*
* \copyright
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at https://mozilla.org/MPL/2.0/.
*/
#ifndef COSIM_ORCHESTRATION_HPP
#define COSIM_ORCHESTRATION_HPP
#include <cosim/file_cache.hpp>
#include <cosim/fmi/importer.hpp>
#include <cosim/model_description.hpp>
#include <cosim/slave.hpp>
#include <cosim/uri.hpp>
#include <memory>
#include <string_view>
#include <vector>
namespace cosim
{
/// A model, i.e., a blueprint from which slaves can be instantiated.
class model
{
public:
virtual ~model() noexcept = default;
/// Returns a description of this model.
virtual std::shared_ptr<const model_description> description() const noexcept = 0;
/// Instantiates a slave.
virtual std::shared_ptr<slave> instantiate(std::string_view name) = 0;
};
/**
* An interface for classes that resolve model URIs of one or more specific
* URI schemes.
*
* Examples could be a file URI resolver which handles URIs like
* `file:///models/my_model.fmu` or an FMU-proxy URI resolver which handles
* URIs like `fmu-proxy://sim.example.com:6345`.
*
* Client code will normally not use this directly to resolve URIs, but rather
* as one of many sub-resolvers in a `model_uri_resolver`.
*/
class model_uri_sub_resolver
{
public:
virtual ~model_uri_sub_resolver() noexcept = default;
/**
* Tries to resolve a model URI relative to some base URI.
*
* Returns a `model` object for the model referred to by the resulting
* URI, or null if this resolver is not designed to handle such URIs.
* May also throw an exception if the URI would normally be handled,
* but the address resolution failed (e.g. due to I/O error).
*
* \note
* The default implementation of this function resolves
* `modelUriReference` relative to `baseUri` in an RFC 3986
* compliant manner using `cosim::resolve_uri_reference()` and
* forwards to `lookup_model(const uri&)`. Specific sub-resolvers
* may override it to use non-standard resolution mechanisms.
*
* \param [in] baseUri
* An (absolute) base URI.
* \param [in] modelUriReference
* A model URI reference that will be resolved relative to `baseUri`.
*/
virtual std::shared_ptr<model> lookup_model(
const uri& baseUri,
const uri& modelUriReference);
/**
* Tries to resolve a model URI.
*
* Returns a `model` object for the model referred to by `modelUri`,
* or null if this resolver is not designed to handle such URIs.
* May also throw an exception if the URI would normally be handled,
* but the address resolution failed (e.g. due to I/O error).
*
* \param [in] modelUri
* An (absolute) model URI.
*/
virtual std::shared_ptr<model> lookup_model(const uri& modelUri) = 0;
};
/**
* A generic model URI resolver.
*
* This class groups resolvers for multiple model URI schemes into one.
* Use `default_model_uri_resolver()` to create one which handles all
* schemes that have built-in support in libcosim.
*
* A custom URI resolver can be created by starting with a default-constructed
* object and adding scheme-specific resolvers using `add_sub_resolver()`.
*/
class model_uri_resolver
{
public:
/// Constructs an empty URI resolver.
model_uri_resolver() noexcept;
model_uri_resolver(model_uri_resolver&&) noexcept;
model_uri_resolver& operator=(model_uri_resolver&&) noexcept;
model_uri_resolver(const model_uri_resolver&) = delete;
model_uri_resolver& operator=(const model_uri_resolver&) = delete;
~model_uri_resolver() noexcept;
/// Adds a sub-resolver.
void add_sub_resolver(std::shared_ptr<model_uri_sub_resolver> sr);
/**
* Tries to resolve a model URI reference relative to some base URI.
*
* The URIs will be passed to each of the sub-resolvers in turn,
* in the order they were added, until one of them succeeds (or throws).
*
* \param [in] baseUri
* An (absolute) base URI.
* \param [in] modelUriReference
* A model URI reference that will be resolved relative to `baseUri`.
*
* \returns
* The model referred to by `modelUriReference`.
*
* \throws std::invalid_argument
* if neither of `baseUri` or `modelUriReference` are absolute.
* \throws std::runtime_error
* if URI resolution failed.
*/
std::shared_ptr<model> lookup_model(
const uri& baseUri,
const uri& modelUriReference);
/**
* Tries to resolve the given model URI.
*
* The URI will be passed to each of the sub-resolvers in turn,
* in the order they were added, until one of them succeeds (or throws).
*
* \param [in] modelUri
* An (absolute) model URI.
*
* \returns
* The model referred to by `modelUri`.
*
* \throws std::invalid_argument
* if `modelUri` is not absolute.
* \throws std::runtime_error
* if URI resolution failed.
*/
std::shared_ptr<model> lookup_model(const uri& modelUri);
private:
std::vector<std::shared_ptr<model_uri_sub_resolver>> subResolvers_;
};
/// A resolver for `file://` model URIs with .fmu file extension.
class fmu_file_uri_sub_resolver : public model_uri_sub_resolver
{
public:
fmu_file_uri_sub_resolver();
explicit fmu_file_uri_sub_resolver(std::shared_ptr<file_cache> cache);
std::shared_ptr<model> lookup_model(const uri& modelUri) override;
private:
std::shared_ptr<fmi::importer> importer_;
};
/**
* Returns a resolver for all URI schemes supported natively by libcosim.
*
* If `cache` is not null, it will be used for caching by the URI
* resolvers that support it (e.g. for unpacking of FMUs by the `file`
* URI resolver).
*/
std::shared_ptr<model_uri_resolver> default_model_uri_resolver(
std::shared_ptr<file_cache> cache = nullptr);
} // namespace cosim
#endif // header guard