Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

CGMES metadata models extension #2898

Merged
merged 30 commits into from
Mar 26, 2024
Merged
Show file tree
Hide file tree
Changes from 16 commits
Commits
Show all changes
30 commits
Select commit Hold shift + click to select a range
39b04b0
CGMES metadata models extension
zamarrenolm Feb 13, 2024
5348e0f
checkstyle: whitespace
zamarrenolm Feb 13, 2024
01dd060
Merge branch 'main' into cgmes_metadata_models
zamarrenolm Feb 19, 2024
53f5051
Merge branch 'main' into cgmes_metadata_models
zamarrenolm Mar 7, 2024
10c32b9
Merge branch 'main' into cgmes_metadata_models
zamarrenolm Mar 7, 2024
c1dfc40
remove previous sv, ssh extensions and replace with new generic exten…
zamarrenolm Mar 7, 2024
04d3863
fixes from failing unit tests
zamarrenolm Mar 8, 2024
db2bb00
names in extension
zamarrenolm Mar 8, 2024
d43c1ae
always export models sorted; adjust names in test expected data
zamarrenolm Mar 8, 2024
fc6defd
remove debug code
zamarrenolm Mar 8, 2024
88d7f50
Merge branch 'main' into cgmes_metadata_models
zamarrenolm Mar 8, 2024
8b85452
add source of model (import or export); use enum for part (cgmes subset)
zamarrenolm Mar 11, 2024
3a8ba5e
Merge branch 'main' into cgmes_metadata_models
zamarrenolm Mar 11, 2024
7510cb1
ignore unknown subset in test
zamarrenolm Mar 11, 2024
0ca3f24
Merge branch 'main' into cgmes_metadata_models
zamarrenolm Mar 12, 2024
f8d998a
remove source of metadata model
zamarrenolm Mar 12, 2024
5bd9507
use a single class for storing CGMES metadata model information
zamarrenolm Mar 13, 2024
54fe16e
Merge branch 'main' into cgmes_metadata_models
zamarrenolm Mar 13, 2024
9d39929
Merge branch 'main' into cgmes_metadata_models
zamarrenolm Mar 14, 2024
705e84c
we do not need to store multiple identifiers in a metadata model
zamarrenolm Mar 14, 2024
17a2425
query fullModel for profile replace by fullModels
zamarrenolm Mar 15, 2024
9759a96
fix license data
zamarrenolm Mar 15, 2024
e1707e4
Merge branch 'main' into cgmes_metadata_models
zamarrenolm Mar 15, 2024
3c80f77
Merge branch 'main' into cgmes_metadata_models
zamarrenolm Mar 18, 2024
feeef19
Model part is an enum
zamarrenolm Mar 18, 2024
7ebd5c0
Merge branch 'main' into cgmes_metadata_models
zamarrenolm Mar 19, 2024
6e535d4
naming: uniform use of cgmes "subset" instead of "part"
zamarrenolm Mar 22, 2024
78e0c49
Merge branch 'main' into cgmes_metadata_models
zamarrenolm Mar 22, 2024
514e137
Add javadoc
rcourtier Mar 26, 2024
d27b2bd
Merge branch 'main' into cgmes_metadata_models
rcourtier Mar 26, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Supplier;
import java.util.stream.Stream;

import static com.powsybl.cgmes.conversion.CgmesReports.importedCgmesNetworkReport;
import static com.powsybl.cgmes.conversion.Conversion.Config.StateProfile.SSH;
Expand Down Expand Up @@ -156,8 +157,7 @@ public Network convert(ReportNode reportNode) {
Network network = createNetwork();
Context context = createContext(network, reportNode);
assignNetworkProperties(context);
addCgmesSvMetadata(network, context);
addCgmesSshMetadata(network, context);
addMetadataModels(network, context);
addCimCharacteristics(network);
BaseVoltageMappingAdder bvAdder = network.newExtension(BaseVoltageMappingAdder.class);
cgmes.baseVoltages().forEach(bv -> bvAdder.addBaseVoltage(bv.getId("BaseVoltage"), bv.asDouble("nominalVoltage"), isBoundaryBaseVoltage(bv.getLocal("graph"))));
Expand Down Expand Up @@ -438,36 +438,48 @@ private void assignNetworkProperties(Context context) {
LOG.info("network forecastDistance : {}", context.network().getForecastDistance());
}

private void addCgmesSvMetadata(Network network, Context context) {
PropertyBags svDescription = cgmes.fullModel(CgmesSubset.STATE_VARIABLES.getProfile());
if (svDescription != null && !svDescription.isEmpty()) {
CgmesSvMetadataAdder adder = network.newExtension(CgmesSvMetadataAdder.class)
.setDescription(svDescription.get(0).get("description"))
.setSvVersion(readVersion(svDescription, context))
.setModelingAuthoritySet(svDescription.get(0).get("modelingAuthoritySet"));
svDescription.pluckLocals("DependentOn").forEach(adder::addDependency);
adder.add();
private void addMetadataModels(Network network, Context context) {
PropertyBags ps = cgmes.fullModels();
if (ps.isEmpty()) {
return;
}
CgmesMetadataModelsAdder modelsAdder = network.newExtension(CgmesMetadataModelsAdder.class);
for (PropertyBag p : ps) {
CgmesMetadataModelsAdder.ModelAdder modelAdder = modelsAdder.newModel()
.setId(p.getId("FullModel"))
.setPart(partFromGraph(p.getLocal("graph")))
.setDescription(p.getId("description"))
.setVersion(readVersion(p, context))
.setModelingAuthoritySet(p.getId("modelingAuthoritySet"));
addMetadataModelReferences(p, "profileList", modelAdder::addProfile);
addMetadataModelReferences(p, "dependentOnList", modelAdder::addDependentOn);
addMetadataModelReferences(p, "supersedesList", modelAdder::addSupersedes);
modelAdder.add();
}
modelsAdder.add();
}

private void addCgmesSshMetadata(Network network, Context context) {
PropertyBags sshDescription = cgmes.fullModel(CgmesSubset.STEADY_STATE_HYPOTHESIS.getProfile());
if (sshDescription != null && !sshDescription.isEmpty()) {
CgmesSshMetadataAdder adder = network.newExtension(CgmesSshMetadataAdder.class)
.setId(sshDescription.get(0).get("FullModel"))
.setDescription(sshDescription.get(0).get("description"))
.setSshVersion(readVersion(sshDescription, context))
.setModelingAuthoritySet(sshDescription.get(0).get("modelingAuthoritySet"));
sshDescription.pluckLocals("DependentOn").forEach(adder::addDependency);
adder.add();
private void addMetadataModelReferences(PropertyBag p, String refsProperty, Function<String, CgmesMetadataModelsAdder.ModelAdder> adder) {
String refs = p.get(refsProperty);
if (refs != null && !refs.isEmpty()) {
for (String ref : refs.split(" ")) {
adder.apply(ref);
}
}
}

private int readVersion(PropertyBags propertyBags, Context context) {
private CgmesSubset partFromGraph(String graph) {
return Stream.of(CgmesSubset.values())
.filter(subset -> subset.isValidName(graph))
.findFirst()
.orElse(CgmesSubset.UNKNOWN);
}

private int readVersion(PropertyBag propertyBag, Context context) {
try {
return propertyBags.get(0).asInt("version");
return propertyBag.asInt("version");
} catch (NumberFormatException e) {
context.fixed("Version", "The version is expected to be an integer: " + propertyBags.get(0).get("version") + ". Fixed to 1");
context.fixed("Version", "The version is expected to be an integer: " + propertyBag.get("version") + ". Fixed to 1");
return 1;
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@
import com.powsybl.cgmes.extensions.*;
import com.powsybl.cgmes.model.CgmesNames;
import com.powsybl.cgmes.model.CgmesNamespace;
import com.powsybl.cgmes.model.CgmesSubset;
import com.powsybl.commons.report.ReportNode;
import com.powsybl.iidm.network.*;
import com.powsybl.iidm.network.Identifiable;
Expand Down Expand Up @@ -272,21 +273,20 @@ public CgmesExportContext(Network network, ReferenceDataProvider referenceDataPr
topologyKind = networkTopologyKind(network);
}
scenarioTime = network.getCaseDate();
// TODO CgmesSvMetadata and CgmesSshMetadata could be in fact same class
// Add multiple instances of CgmesMetadata to Network, one for each profile
CgmesSvMetadata svMetadata = network.getExtension(CgmesSvMetadata.class);
CgmesMetadataModels models = network.getExtension(CgmesMetadataModels.class);
CgmesMetadataModels.Model svMetadata = models != null ? models.getModelForPart(CgmesSubset.STATE_VARIABLES).orElse(null) : null;
if (svMetadata != null) {
svModelDescription.setDescription(svMetadata.getDescription());
svModelDescription.setVersion(svMetadata.getSvVersion() + 1);
svModelDescription.addDependencies(svMetadata.getDependencies());
svModelDescription.setVersion(svMetadata.getVersion() + 1);
svModelDescription.addDependencies(svMetadata.getDependentOn());
svModelDescription.setModelingAuthoritySet(svMetadata.getModelingAuthoritySet());
}
CgmesSshMetadata sshMetadata = network.getExtension(CgmesSshMetadata.class);
CgmesMetadataModels.Model sshMetadata = models != null ? models.getModelForPart(CgmesSubset.STEADY_STATE_HYPOTHESIS).orElse(null) : null;
if (sshMetadata != null) {
sshModelDescription.setDescription(sshMetadata.getDescription());
sshModelDescription.setVersion(sshMetadata.getSshVersion() + 1);
sshModelDescription.setVersion(sshMetadata.getVersion() + 1);
sshModelDescription.setSupersedes(sshMetadata.getId());
sshModelDescription.addDependencies(sshMetadata.getDependencies());
sshModelDescription.addDependencies(sshMetadata.getDependentOn());
sshModelDescription.setModelingAuthoritySet(sshMetadata.getModelingAuthoritySet());
}
addIidmMappings(network);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,12 +16,11 @@
import com.powsybl.cgmes.conversion.Conversion;
import com.powsybl.cgmes.conversion.test.network.compare.Comparison;
import com.powsybl.cgmes.conversion.test.network.compare.ComparisonConfig;
import com.powsybl.cgmes.model.CgmesModel;
import com.powsybl.cgmes.model.CgmesModelException;
import com.powsybl.cgmes.model.GridModelReference;
import com.powsybl.cgmes.model.*;
import com.powsybl.commons.datasource.DataSource;
import com.powsybl.commons.datasource.FileDataSource;
import com.powsybl.commons.datasource.ReadOnlyDataSource;
import com.powsybl.commons.datasource.ZipFileDataSource;
import com.powsybl.iidm.network.Network;
import com.powsybl.iidm.network.impl.NetworkFactoryImpl;
import com.powsybl.iidm.serde.XMLExporter;
Expand All @@ -43,6 +42,7 @@
import java.util.List;
import java.util.Properties;
import java.util.function.Consumer;
import java.util.stream.Stream;

import static org.junit.jupiter.api.Assertions.assertTrue;
import static org.junit.jupiter.api.Assertions.fail;
Expand Down Expand Up @@ -177,31 +177,40 @@ private static void exportCgmes(String name, String impl, Network network) throw
new CgmesExport().export(network, null, new FileDataSource(path, "foo"));
}

private static String partFromName(String name) {
return Stream.of(CgmesSubset.values())
.filter(s -> s.isValidName(name))
.map(CgmesSubset::getIdentifier)
.findFirst()
.orElse("unknown");
}

private void testExportImportCgmes(Network network, ReadOnlyDataSource originalDs, FileSystem fs, CgmesImport i, Properties iparams,
ComparisonConfig config) throws IOException {

// We copy everything from the original data source to the temporary destination with a normalized name
// And then export the requested instance files to the same temporary destination
// We will overwrite some of the files, with the expected content

// Create a temporary directory to store the exported files
Path path = fs.getPath("temp-export-cgmes");
Files.createDirectories(path);
String baseName = "bar";
new CgmesExport().export(network, exportParams, new FileDataSource(path, baseName));

DataSource ds = new FileDataSource(path, "bar");
String expected = originalDs.listNames(".*EQ.*").stream().filter(name -> !name.contains("BD")).findFirst().orElseThrow(() -> new CgmesModelException("Should contain EQ profile"));
try (OutputStream out = new BufferedOutputStream(ds.newOutputStream(baseName + "_EQ.xml", false));
InputStream in = originalDs.newInputStream(expected)) {
ByteStreams.copy(in, out);
}
expected = originalDs.listNames(".*TP.*").stream().filter(name -> !name.contains("BD")).findFirst().orElseThrow(() -> new CgmesModelException("Should contain TP profile"));
try (OutputStream out = new BufferedOutputStream(ds.newOutputStream(baseName + "_TP.xml", false));
InputStream in = originalDs.newInputStream(expected)) {
ByteStreams.copy(in, out);
}
for (String boundary : originalDs.listNames(".*BD.*")) {
try (OutputStream out = new BufferedOutputStream(ds.newOutputStream(baseName + boundary, false));
InputStream in = originalDs.newInputStream(boundary)) {
String baseName = originalDs.getBaseName();
DataSource ds = new ZipFileDataSource(path, baseName);

// Copy the original files to the temporary destination, ensuring a normalized name
for (String name : new CgmesOnDataSource(originalDs).names()) {
String normalizedName = baseName + "_" + partFromName(name) + ".xml";
try (OutputStream out = new BufferedOutputStream(ds.newOutputStream(normalizedName, false));
InputStream in = originalDs.newInputStream(name)) {
ByteStreams.copy(in, out);
}
}

// Export the requested instance files to the temporary destination, overwriting some of the original files
new CgmesExport().export(network, exportParams, ds);

// Import the exported files and compare with the original network
Network actual = i.importData(ds, new NetworkFactoryImpl(), iparams);
new Comparison(network, actual, config).compare();
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -116,7 +116,7 @@ void microGridWithAndWithoutTpSv() {

resetBusVoltageAndAngleBeforeComparison(network);
resetTerminalPQofLoadsAndGeneratorsBeforeComparison(network);
new Comparison(network, networkwithoutTpSv, new ComparisonConfig()).compare();
new Comparison(network, networkwithoutTpSv, new ComparisonConfig().ignoreMissingMetadata()).compare();
assertTrue(true);
}

Expand Down Expand Up @@ -199,7 +199,7 @@ void miniGridWithAndWithoutTpSv() {

resetBusVoltageAndAngleBeforeComparison(network);
resetTerminalPQofLoadsAndGeneratorsBeforeComparison(network);
new Comparison(network, networkwithoutTpSv, new ComparisonConfig()).compare();
new Comparison(network, networkwithoutTpSv, new ComparisonConfig().ignoreMissingMetadata()).compare();
assertTrue(true);
}

Expand Down Expand Up @@ -263,7 +263,7 @@ void smallGridWithAndWithoutTpSv() {

resetBusVoltageAndAngleBeforeComparison(network);
resetTerminalPQofLoadsAndGeneratorsBeforeComparison(network);
new Comparison(network, networkwithoutTpSv, new ComparisonConfig()).compare();
new Comparison(network, networkwithoutTpSv, new ComparisonConfig().ignoreMissingMetadata()).compare();
assertTrue(true);
}

Expand Down Expand Up @@ -345,7 +345,7 @@ void svedalaWithAndWithoutTpSv() {

resetBusVoltageAndAngleBeforeComparison(network);
resetTerminalPQofLoadsAndGeneratorsBeforeComparison(network);
new Comparison(network, networkwithoutTpSv, new ComparisonConfig()).compare();
new Comparison(network, networkwithoutTpSv, new ComparisonConfig().ignoreMissingMetadata()).compare();
assertTrue(true);
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@
import com.powsybl.cgmes.conversion.test.network.compare.ComparisonConfig;
import com.powsybl.cgmes.model.CgmesModel;
import com.powsybl.cgmes.model.CgmesNames;
import com.powsybl.cgmes.model.CgmesSubset;
import com.powsybl.iidm.network.*;
import com.powsybl.iidm.network.extensions.ActivePowerControl;
import com.powsybl.triplestore.api.TripleStoreFactory;
Expand Down Expand Up @@ -95,7 +96,11 @@ void microGridBaseCaseBERoundtripBoundary() throws IOException {
ConversionTester t = new ConversionTester(
importParams, exportParams,
TripleStoreFactory.onlyDefaultImplementation(),
new ComparisonConfig().tolerance(1e-5).checkNetworkId(false).incrementVersions(true));
new ComparisonConfig()
.tolerance(1e-5)
.checkNetworkId(false)
.incrementVersions(true)
.exportedParts(Set.of(CgmesSubset.STEADY_STATE_HYPOTHESIS, CgmesSubset.STATE_VARIABLES)));
t.setTestExportImportCgmes(true);
Network expected = null;
t.testConversion(expected, CgmesConformity1Catalog.microGridBaseCaseBE());
Expand All @@ -110,7 +115,11 @@ void microGridBaseCaseBERoundtrip() throws IOException {
exportParams.put(CgmesExport.MODELING_AUTHORITY_SET, "http://elia.be/CGMES/2.4.15");
ConversionTester t = new ConversionTester(importParams, exportParams,
TripleStoreFactory.onlyDefaultImplementation(),
new ComparisonConfig().tolerance(1e-5).checkNetworkId(false).incrementVersions(true));
new ComparisonConfig()
.tolerance(1e-5)
.checkNetworkId(false)
.incrementVersions(true)
.exportedParts(Set.of(CgmesSubset.STEADY_STATE_HYPOTHESIS, CgmesSubset.STATE_VARIABLES)));
t.setTestExportImportCgmes(true);
t.testConversion(CgmesConformity1NetworkCatalog.microBaseCaseBE(), CgmesConformity1Catalog.microGridBaseCaseBE());
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,9 +16,10 @@
import com.powsybl.cgmes.conversion.test.ConversionUtil;
import com.powsybl.cgmes.extensions.CgmesControlArea;
import com.powsybl.cgmes.extensions.CgmesControlAreas;
import com.powsybl.cgmes.extensions.CgmesSvMetadata;
import com.powsybl.cgmes.extensions.CgmesMetadataModels;
import com.powsybl.cgmes.model.CgmesModel;
import com.powsybl.cgmes.model.CgmesModelException;
import com.powsybl.cgmes.model.CgmesSubset;
import com.powsybl.cgmes.model.GridModelReference;
import com.powsybl.commons.PowsyblException;
import com.powsybl.commons.config.InMemoryPlatformConfig;
Expand All @@ -41,6 +42,7 @@
import java.time.Duration;
import java.time.ZonedDateTime;
import java.util.Map;
import java.util.Optional;
import java.util.Properties;
import java.util.stream.Collectors;
import java.util.stream.Stream;
Expand Down Expand Up @@ -509,9 +511,11 @@ void microBEIncorrectDate() {
NetworkFactory.findDefault(), importParams);
assertEquals(0, network.getForecastDistance());
assertTrue(Duration.between(ZonedDateTime.now(), network.getCaseDate()).toMinutes() < 10);
CgmesSvMetadata cgmesSvMetadata = network.getExtension(CgmesSvMetadata.class);
assertNotNull(cgmesSvMetadata);
assertEquals(1, cgmesSvMetadata.getSvVersion());
CgmesMetadataModels cgmesMetadata = network.getExtension(CgmesMetadataModels.class);
assertNotNull(cgmesMetadata);
Optional<CgmesMetadataModels.Model> svModel = cgmesMetadata.getModelForPart(CgmesSubset.STATE_VARIABLES);
assertTrue(svModel.isPresent());
assertEquals(1, svModel.get().getVersion());
}

@Test
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,10 +8,11 @@

import com.powsybl.cgmes.conversion.CgmesExport;
import com.powsybl.cgmes.conversion.export.CgmesExportContext;
import com.powsybl.cgmes.extensions.CgmesSvMetadataAdder;
import com.powsybl.cgmes.extensions.CgmesMetadataModelsAdder;
import com.powsybl.cgmes.extensions.CgmesTopologyKind;
import com.powsybl.cgmes.extensions.CimCharacteristicsAdder;
import com.powsybl.cgmes.model.CgmesNamespace;
import com.powsybl.cgmes.model.CgmesSubset;
import com.powsybl.iidm.network.*;
import com.powsybl.iidm.network.test.EurostagTutorialExample1Factory;
import java.time.ZonedDateTime;
Expand Down Expand Up @@ -56,13 +57,18 @@ void networkConstructor() {
.setCimVersion(14)
.setTopologyKind(CgmesTopologyKind.NODE_BREAKER)
.add();
network.newExtension(CgmesSvMetadataAdder.class)
.setDescription("test")
.setSvVersion(2)
.addDependency("powsybl.test.org")
.addDependency("cgmes")
.setModelingAuthoritySet("cgmes.org")
.add();
network.newExtension(CgmesMetadataModelsAdder.class)
.newModel()
.setId("testId")
.setPart(CgmesSubset.STATE_VARIABLES)
.setDescription("test")
.setVersion(2)
.addProfile("testProfile")
.addDependentOn("otherModel1")
.addDependentOn("otherModel2")
.setModelingAuthoritySet("cgmes.org")
.add()
.add();

CgmesExportContext context2 = new CgmesExportContext(network);

Expand All @@ -73,8 +79,8 @@ void networkConstructor() {
assertEquals("test", context2.getSvModelDescription().getDescription());
assertEquals(3, context2.getSvModelDescription().getVersion());
assertEquals(2, context2.getSvModelDescription().getDependencies().size());
assertTrue(context2.getSvModelDescription().getDependencies().contains("powsybl.test.org"));
assertTrue(context2.getSvModelDescription().getDependencies().contains("cgmes"));
assertTrue(context2.getSvModelDescription().getDependencies().contains("otherModel1"));
assertTrue(context2.getSvModelDescription().getDependencies().contains("otherModel2"));
assertEquals("cgmes.org", context2.getSvModelDescription().getModelingAuthoritySet());
}

Expand Down
Loading