Skip to content

Commit 8f64719

Browse files
authored
Merge pull request #2660 from GMLC-TDC/develop
2 parents 4e1c24e + 60644d4 commit 8f64719

19 files changed

+254
-85
lines changed

.clang-tidy

+3
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@ Checks: "
2828
misc-*,
2929
-misc-non-private-member-variables-in-classes,
3030
-misc-no-recursion,
31+
-misc-use-anonymous-namespace,
3132
cert-*,
3233
-cert-err58-cpp,
3334
portability-*,
@@ -67,6 +68,8 @@ WarningsAsErrors: "
6768
misc-*,
6869
-misc-non-private-member-variables-in-classes,
6970
-misc-no-recursion,
71+
-misc-include-cleaner,
72+
-misc-use-anonymous-namespace,
7073
cert-*,
7174
-cert-err58-cpp,
7275
portability-*,

.pre-commit-config.yaml

+3-3
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ repos:
88
- id: dockerfile_lint
99
args: [--rulefile, ./config/Docker/docker_rules.yml, --dockerfile]
1010
- repo: https://github.com/psf/black
11-
rev: 24.3.0
11+
rev: 24.4.2
1212
hooks:
1313
- id: black
1414
args: ["--line-length=100"]
@@ -22,7 +22,7 @@ repos:
2222
hooks:
2323
- id: remove-tabs
2424
- repo: https://github.com/pre-commit/pre-commit-hooks
25-
rev: v4.5.0
25+
rev: v4.6.0
2626
hooks:
2727
- id: check-added-large-files
2828
- id: mixed-line-ending
@@ -74,7 +74,7 @@ repos:
7474
"--exclude-file=./config/spelling_ignorelines.txt",
7575
]
7676
- repo: https://github.com/pre-commit/mirrors-clang-format
77-
rev: v18.1.2
77+
rev: v18.1.5
7878
hooks:
7979
- id: clang-format
8080
types:

CHANGELOG.md

+14
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,20 @@ This project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.htm
99
A note on future revisions.
1010
Everything within a major version number should be code compatible (with the exception of experimental interfaces). The most notable example of an experimental interface is the support for multiple source inputs. The APIs to deal with this will change in future minor releases. Everything within a single minor release should be network compatible with other federates on the same minor release number. Compatibility across minor release numbers may be possible in some situations but we are not going to guarantee this as those components are subject to performance improvements and may need to be modified at some point. Patch releases will be limited to bug fixes and other improvements not impacting the public API or network compatibility. Check the [Public API](./docs/Public_API.md) for details on what is included and excluded from the public API and version stability.
1111

12+
## [3.5.3][] - 2024-07-08
13+
14+
Patch release with fixes for potential interface definitions and some compiler warnings
15+
16+
### Fixed
17+
18+
- Fixed some compiler warnings on the connector
19+
20+
### Added
21+
22+
- Added support for "potential_interface_templates" object in json configuration
23+
- Added support for arrays of tags
24+
- Added support for "fields" object in potential interface template definitions
25+
1226
## [3.5.2][] - 2024-04-08
1327

1428
Patch release with fixes for certain compiler builds, a fix to the test core leading to some sporadic test failures, and fixing a discrepancy in the handing of config files with the helics_apps.

CMakeLists.txt

+2-2
Original file line numberDiff line numberDiff line change
@@ -17,14 +17,14 @@ if(DEFINED ENV{VCPKG_ROOT} AND NOT DEFINED CMAKE_TOOLCHAIN_FILE AND NOT HELICS_D
1717
set(CMAKE_TOOLCHAIN_FILE "$ENV{VCPKG_ROOT}/scripts/buildsystems/vcpkg.cmake" CACHE STRING "")
1818
endif()
1919

20-
project(HELICS VERSION 3.5.2)
20+
project(HELICS VERSION 3.5.3)
2121

2222
# -----------------------------------------------------------------------------
2323
# HELICS Version number
2424
# -----------------------------------------------------------------------------
2525
set(HELICS_VERSION_BUILD)
2626
# use ISO date YYYY-MM-DD
27-
set(HELICS_DATE "2024-04-08")
27+
set(HELICS_DATE "2024-07-08")
2828

2929
set(HELICS_VERSION_UNDERSCORE
3030
"${HELICS_VERSION_MAJOR}_${HELICS_VERSION_MINOR}_${HELICS_VERSION_PATCH}"

appveyor.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ branches:
55
- main
66
- develop
77

8-
version: 3.5.2.{build}
8+
version: 3.5.3.{build}
99

1010
image: Visual Studio 2019
1111

docs/ROADMAP.md

-1
Original file line numberDiff line numberDiff line change
@@ -27,5 +27,4 @@ This document contains tentative plans for changes and improvements of note in u
2727
- Some sort of rollback operations
2828
- Remote procedure call type of federate
2929
- Plugin architecture for user defined cores
30-
- Separate octave interface
3130
- Enable mesh networking in HELICS

docs/developer-guide/typeConversions.md

+3
Original file line numberDiff line numberDiff line change
@@ -105,6 +105,7 @@ There are defined conversions from all known available types to all others.
105105
- BOOL -> (val!=0)?"1":"0"
106106

107107
[^1]: conversion to double is lossless only if the value actually fits in a double mantissa value.
108+
108109
[^2]: for a named point conversion, if the value doesn't fit in double the string translation is placed in the string field and a NaN value in the value segment to ensure no data loss.
109110

110111
### Conversion from String
@@ -155,7 +156,9 @@ Similar to getDoubleFromString in conversion of vectors. It will convert most re
155156
- BOOL -> (vectorNorm(val)!=0)?"1":"0"
156157

157158
[^3]: vectorNorm is the sqrt of the inner product of the vector.
159+
158160
[^4]: vectorString is comma-separated string of the numerical values enclosed in `[]`, for example `[45.7,22.7,17.8]`. This is a JSON compatible string format.
161+
159162
[^5]: if the vector is a single element the NAMED_POINT translation is equivalent to a double translation.
160163

161164
### Conversion from Complex Vector

src/helics/application_api/Federate.cpp

+5-3
Original file line numberDiff line numberDiff line change
@@ -132,6 +132,8 @@ Federate::~Federate()
132132
// LCOV_EXCL_START
133133
catch (...) // do not allow a throw inside the destructor
134134
{
135+
// finalize may throw but we can't allow that
136+
;
135137
}
136138
// LCOV_EXCL_STOP
137139
}
@@ -1542,7 +1544,7 @@ void Federate::registerConnectorInterfacesJsonDetail(Json::Value& json)
15421544
registerConnectorInterfacesJsonDetail(json["helics"]);
15431545
}
15441546

1545-
if (json.isMember("potential_interfaces")) {
1547+
if (json.isMember("potential_interfaces") || json.isMember("potential_interface_templates")) {
15461548
if (!potManager) {
15471549
potManager = std::make_unique<PotentialInterfacesManager>(coreObject.get(), this);
15481550
}
@@ -2060,9 +2062,9 @@ void Federate::logMessage(int level, std::string_view message) const
20602062
if (coreObject) {
20612063
coreObject->logMessage(fedID, level, message);
20622064
} else if (level <= HELICS_LOG_LEVEL_WARNING) {
2063-
std::cerr << message << std::endl;
2065+
std::cerr << message << '\n';
20642066
} else {
2065-
std::cout << message << std::endl;
2067+
std::cout << message << '\n';
20662068
}
20672069
}
20682070

src/helics/application_api/PotentialInterfacesManager.cpp

+46-25
Original file line numberDiff line numberDiff line change
@@ -24,34 +24,55 @@ void PotentialInterfacesManager::loadPotentialInterfaces(Json::Value& json)
2424
{
2525
static const std::set<std::string> interfaceTypes{
2626
"publications", "inputs", "endpoints", "filters", "translators", "datasinks"};
27-
if (!json.isMember("potential_interfaces")) {
28-
return;
29-
}
30-
const auto& interfaces = json["potential_interfaces"];
31-
for (const auto& itype : interfaceTypes) {
32-
if (interfaces.isMember(itype)) {
33-
auto tInterface = interfaces[itype];
34-
auto& pMap = potInterfaces[itype];
35-
for (auto& ispec : tInterface) {
36-
auto name = fileops::getName(ispec);
37-
pMap[name] = ispec;
27+
if (json.isMember("potential_interfaces")) {
28+
const auto& interfaces = json["potential_interfaces"];
29+
for (const auto& itype : interfaceTypes) {
30+
if (interfaces.isMember(itype)) {
31+
auto tInterface = interfaces[itype];
32+
auto& pMap = potInterfaces[itype];
33+
for (auto& ispec : tInterface) {
34+
auto name = fileops::getName(ispec);
35+
pMap[name] = ispec;
36+
}
37+
}
38+
std::string tempString = itype;
39+
tempString.pop_back();
40+
tempString += "_templates";
41+
if (interfaces.isMember(tempString)) {
42+
auto templateInterfaces = interfaces[tempString];
43+
auto& tMap = potInterfaceTemplates[itype];
44+
for (auto& tspec : templateInterfaces) {
45+
auto name = fileops::getName(tspec);
46+
if (name.find("}${") != std::string::npos) {
47+
throw(helics::InvalidParameter(
48+
std::string(
49+
"template key definitions must not be adjacent, they must have separator characters [") +
50+
name + ']'));
51+
}
52+
tMap[name] = tspec;
53+
}
3854
}
3955
}
40-
std::string tempString = itype;
41-
tempString.pop_back();
42-
tempString += "_templates";
43-
if (interfaces.isMember(tempString)) {
44-
auto templateInterfaces = interfaces[tempString];
45-
auto& tMap = potInterfaceTemplates[itype];
46-
for (auto& tspec : templateInterfaces) {
47-
auto name = fileops::getName(tspec);
48-
if (name.find("}${") != std::string::npos) {
49-
throw(helics::InvalidParameter(
50-
std::string(
51-
"template key definitions must not be adjacent, they must have separator characters [") +
52-
name + ']'));
56+
}
57+
if (json.isMember("potential_interface_templates")) {
58+
const auto& interfaces = json["potential_interface_templates"];
59+
for (const auto& itype : interfaceTypes) {
60+
std::string tempString = itype;
61+
tempString.pop_back();
62+
tempString += "_templates";
63+
if (interfaces.isMember(tempString)) {
64+
auto templateInterfaces = interfaces[tempString];
65+
auto& tMap = potInterfaceTemplates[itype];
66+
for (auto& tspec : templateInterfaces) {
67+
auto name = fileops::getName(tspec);
68+
if (name.find("}${") != std::string::npos) {
69+
throw(helics::InvalidParameter(
70+
std::string(
71+
"template key definitions must not be adjacent, they must have separator characters [") +
72+
name + ']'));
73+
}
74+
tMap[name] = tspec;
5375
}
54-
tMap[name] = tspec;
5576
}
5677
}
5778
}

src/helics/apps/Connector.cpp

+21-10
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@ SPDX-License-Identifier: BSD-3-Clause
2020
#include <optional>
2121
#include <regex>
2222
#include <set>
23+
#include <string_view>
2324
#include <tuple>
2425
#include <type_traits>
2526
#include <unordered_map>
@@ -119,13 +120,20 @@ bool TemplateMatcher::loadTemplate(const Json::Value& iTemplate)
119120
intermediaries.push_back(templateName.substr(0, tnameIndex));
120121
}
121122
std::vector<std::string> valueNames;
122-
int index{0};
123+
std::size_t index{0};
123124
while (tnameIndex != std::string::npos) {
124125
auto close = templateName.find_first_of('}', tnameIndex);
125126
const std::string tname = templateName.substr(tnameIndex + 2, close - tnameIndex - 2);
126-
if (!iTemplate.isMember(tname)) {
127-
return false;
127+
if (iTemplate.isMember("fields")) {
128+
if (!iTemplate["fields"].isMember(tname)) {
129+
return false;
130+
}
131+
} else {
132+
if (!iTemplate.isMember(tname)) {
133+
return false;
134+
}
128135
}
136+
129137
valueNames.push_back(tname);
130138

131139
tnameIndex = templateName.find("${", close + 1);
@@ -143,8 +151,10 @@ bool TemplateMatcher::loadTemplate(const Json::Value& iTemplate)
143151

144152
templatePossibilities.resize(valueNames.size());
145153
keys.resize(valueNames.size());
154+
const Json::Value& fieldRoot = (iTemplate.isMember("fields")) ? iTemplate["fields"] : iTemplate;
155+
146156
for (index = 0; index < valueNames.size(); ++index) {
147-
for (const auto& val : iTemplate[valueNames[index]]) {
157+
for (const auto& val : fieldRoot[valueNames[index]]) {
148158
std::pair<std::string, std::string> typeAndUnits;
149159
std::string_view keyval;
150160
if (val.isArray()) {
@@ -167,6 +177,7 @@ bool TemplateMatcher::loadTemplate(const Json::Value& iTemplate)
167177
templatePossibilities[index].emplace(keyval, typeAndUnits);
168178
}
169179
}
180+
170181
initialize();
171182
return true;
172183
}
@@ -594,15 +605,15 @@ Connector::Connector(std::string_view appName, const FederateInfo& fedInfo):
594605
}
595606

596607
Connector::Connector(std::string_view appName,
597-
const std::shared_ptr<Core>& core,
608+
const std::shared_ptr<Core>& coreObj,
598609
const FederateInfo& fedInfo):
599-
App(appName, core, fedInfo), core((fed) ? fed->getCorePointer() : nullptr)
610+
App(appName, coreObj, fedInfo), core((fed) ? fed->getCorePointer() : nullptr)
600611
{
601612
initialSetup();
602613
}
603614

604-
Connector::Connector(std::string_view appName, CoreApp& core, const FederateInfo& fedInfo):
605-
App(appName, core, fedInfo), core((fed) ? fed->getCorePointer() : nullptr)
615+
Connector::Connector(std::string_view appName, CoreApp& coreObj, const FederateInfo& fedInfo):
616+
App(appName, coreObj, fedInfo), core((fed) ? fed->getCorePointer() : nullptr)
606617
{
607618
initialSetup();
608619
}
@@ -645,7 +656,7 @@ bool Connector::addConnectionVector(const std::vector<std::string>& connection)
645656
newTags.push_back(connection[2]);
646657
}
647658

648-
for (int ii = 3; ii < connection.size(); ++ii) {
659+
for (std::size_t ii = 3; ii < connection.size(); ++ii) {
649660
newTags.push_back(connection[ii]);
650661
}
651662
addConnection(connection[0], connection[1], direction, newTags);
@@ -664,7 +675,7 @@ void Connector::addConnection(std::string_view interface1,
664675
}
665676
auto iview1 = addInterface(interface1);
666677
auto iview2 = addInterface(interface2);
667-
Connection conn{iview1, iview2, direction, std::move(svtags)};
678+
Connection conn{iview1, iview2, direction, std::move(svtags), nullptr};
668679
if (iview1.compare(0, 6, "REGEX:") == 0) {
669680
switch (direction) {
670681
case InterfaceDirection::TO_FROM:

src/helics/apps/Connector.hpp

+4-4
Original file line numberDiff line numberDiff line change
@@ -57,18 +57,18 @@ class HELICS_CXX_EXPORT Connector: public App {
5757
explicit Connector(std::string_view name, const FederateInfo& fedInfo);
5858
/**constructor taking a federate information structure and using the given core
5959
@param name the name of the federate (can be empty to use defaults from fedInfo)
60-
@param core a pointer to core object which the federate can join
60+
@param coreObj a pointer to core object which the federate can join
6161
@param fedInfo a federate information structure
6262
*/
6363
Connector(std::string_view name,
64-
const std::shared_ptr<Core>& core,
64+
const std::shared_ptr<Core>& coreObj,
6565
const FederateInfo& fedInfo);
6666
/**constructor taking a federate information structure and using the given core
6767
@param name the name of the federate (can be empty to use defaults from fedInfo)
68-
@param core a coreApp object that can be joined
68+
@param coreObj a coreApp object that can be joined
6969
@param fedInfo a federate information structure
7070
*/
71-
Connector(std::string_view name, CoreApp& core, const FederateInfo& fedInfo);
71+
Connector(std::string_view name, CoreApp& coreObj, const FederateInfo& fedInfo);
7272
/**constructor taking a file with the required information
7373
@param appName the name of the app
7474
@param configString JSON, TOML or text file or JSON string defining the federate information and

src/helics/apps/helicsApp.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -181,7 +181,7 @@ std::vector<int> AppTextParser::preParseFile(const std::vector<char>& klines)
181181
}
182182

183183
++counts[0];
184-
for (int ii = 0; ii < klines.size(); ++ii) {
184+
for (std::size_t ii = 0; ii < klines.size(); ++ii) {
185185
if (str[fc] == klines[ii]) {
186186
++counts[ii + 1];
187187
}

0 commit comments

Comments
 (0)