Skip to content

Commit e288593

Browse files
Merge branch 'dev' into df/#1131-psdm-update-for-tap-water-demand
2 parents ad01e5f + 29a7278 commit e288593

24 files changed

+261
-67
lines changed

.github/dependabot.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ updates:
99
target-branch: dev
1010
reviewers:
1111
- t-ober
12-
- sensarmad
12+
- staudtMarius
1313
- sebastian-peter
1414
- danielfeismann
1515
- jo-bao

CHANGELOG.md

+4
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
99
### Added
1010
- Enhance `TimeSeriesSource` with method to retrieve all time keys after a given key [#543](https://github.com/ie3-institute/PowerSystemDataModel/issues/543)
1111
- Enhance `WeatherSource` with method to retrieve all time keys after a given key [#572](https://github.com/ie3-institute/PowerSystemDataModel/issues/572)
12+
- Adding timeseries for voltage values [#1128](https://github.com/ie3-institute/PowerSystemDataModel/issues/1128)
13+
- Added Staudt to list of reviewers [#1190](https://github.com/ie3-institute/PowerSystemDataModel/issues/1190)
1214

1315
### Fixed
1416

@@ -18,6 +20,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
1820
- Improving usage of streams on sql fetches [#827](https://github.com/ie3-institute/PowerSystemDataModel/issues/827)
1921
- Improving error message when using the outdated csv format [#1112](https://github.com/ie3-institute/PowerSystemDataModel/issues/1112)
2022
- Changed ThermalUnitValidation: Ensure that thermal boundaries of thermal house are not the same [#1186](https://github.com/ie3-institute/PowerSystemDataModel/issues/1186)
23+
- Converted `MappingEntry` into a normal class [#1087](https://github.com/ie3-institute/PowerSystemDataModel/issues/1087)
24+
- Renamed timeseries mapping `participant` column to `asset` [#1191](https://github.com/ie3-institute/PowerSystemDataModel/issues/1191)
2125

2226
## [5.1.0] - 2024-06-24
2327

build.gradle

+6-6
Original file line numberDiff line numberDiff line change
@@ -5,11 +5,11 @@ plugins {
55
id 'signing'
66
id 'pmd' // code check, working on source code
77
id 'com.diffplug.spotless' version '6.25.0' //code format
8-
id 'com.github.spotbugs' version '6.0.26' // code check, working on byte code
8+
id 'com.github.spotbugs' version '6.0.27' // code check, working on byte code
99
id 'de.undercouch.download' version '5.6.0'
1010
id 'kr.motd.sphinx' version '2.10.1' // documentation generation
1111
id 'jacoco' // java code coverage plugin
12-
id "org.sonarqube" version "6.0.0.5145" // sonarqube
12+
id "org.sonarqube" version "6.0.1.5171" // sonarqube
1313
id 'net.thauvin.erik.gradle.semver' version '1.0.4' // semantic versioning
1414
}
1515

@@ -70,10 +70,10 @@ dependencies {
7070
// testing
7171
testImplementation "org.apache.groovy:groovy:$groovyBinaryVersion"
7272

73-
testImplementation 'org.junit.jupiter:junit-jupiter:5.11.3'
73+
testImplementation 'org.junit.jupiter:junit-jupiter:5.11.4'
7474
testImplementation "org.spockframework:spock-core:2.3-groovy-$groovyVersion"
7575
testImplementation 'org.objenesis:objenesis:3.4' // Mock creation with constructor parameters
76-
testImplementation 'net.bytebuddy:byte-buddy:1.15.10' // Mocks of classes
76+
testImplementation 'net.bytebuddy:byte-buddy:1.15.11' // Mocks of classes
7777

7878
// testcontainers (docker framework for testing)
7979
testImplementation "org.testcontainers:testcontainers:$testcontainersVersion"
@@ -83,14 +83,14 @@ dependencies {
8383
testImplementation "org.testcontainers:couchbase:$testcontainersVersion"
8484

8585
// logging
86-
implementation platform('org.apache.logging.log4j:log4j-bom:2.24.2')
86+
implementation platform('org.apache.logging.log4j:log4j-bom:2.24.3')
8787
implementation 'org.apache.logging.log4j:log4j-api' // log4j
8888
implementation 'org.apache.logging.log4j:log4j-core' // log4j
8989
implementation 'org.apache.logging.log4j:log4j-slf4j-impl' // log4j -> slf4j
9090

9191
// Databases
9292
implementation 'org.influxdb:influxdb-java:2.24'
93-
implementation 'com.couchbase.client:java-client:3.7.5'
93+
implementation 'com.couchbase.client:java-client:3.7.6'
9494
runtimeOnly 'org.postgresql:postgresql:42.7.4' // postgresql jdbc driver required during runtime
9595

9696
implementation 'commons-io:commons-io:2.18.0' // I/O functionalities

docs/readthedocs/io/csvfiles.md

+11-7
Original file line numberDiff line numberDiff line change
@@ -122,30 +122,34 @@ This is the UUID from the example above `2fcb3e53-b94a-4b96-bea4-c469e499f1a1`.
122122
The following keys are supported until now:
123123
```{list-table}
124124
:widths: auto
125+
:class: wrapping
125126
:header-rows: 1
126127
127128
* - Key
128-
- Information and supported head line
129+
- Information and supported head line.
129130
* - c
130131
- An energy price (e.g. in €/MWh; c stands for charge).
131132
Permissible head line: ``time,price``
132133
* - p
133-
- Active power
134+
- Active power.
134135
Permissible head line: ``time,p``
135136
* - pq
136-
- Active and reactive power
137+
- Active and reactive power.
137138
Permissible head line: ``time,p,q``
138139
* - h
139-
- Heat power demand
140+
- Heat power demand.
140141
Permissible head line: ``time,h``
141142
* - ph
142-
- Active and heat power
143+
- Active and heat power.
143144
Permissible head line: ``time,p,h``
144145
* - pqh
145-
- Active, reactive and heat power
146+
- Active, reactive and heat power.
146147
Permissible head line: ``time,p,q,h``
148+
* - v
149+
- Voltage mangnitude in pu and angle in °.
150+
Permissible head line: ``time,vMag,vAng``
147151
* - weather
148-
- Weather information
152+
- Weather information.
149153
Permissible head line: ``time,coordinate,direct_irradiation,diffuse_irradiation,temperature,wind_velocity,wind_direction``
150154
151155
```

docs/readthedocs/models/input/additionaldata/timeseries.md

+4-1
Original file line numberDiff line numberDiff line change
@@ -48,7 +48,10 @@ The following different values are available:
4848
4949
* - `WindValue`
5050
- Combination of wind direction and wind velocity
51-
51+
52+
* - `VoltageValue`
53+
- Combination of voltage magnitude in p.u. and angle in °
54+
5255
* - `WeatherValue`
5356
- Combination of irradiance, temperature and wind information
5457

src/main/java/edu/ie3/datamodel/exceptions/EntityProcessorException.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@
66
package edu.ie3.datamodel.exceptions;
77

88
/**
9-
* Is thrown, when an something went wrong during entity field mapping creation in a {@link
9+
* Is thrown, when something went wrong during entity field mapping creation in a {@link
1010
* edu.ie3.datamodel.io.processor.EntityProcessor}
1111
*/
1212
public class EntityProcessorException extends Exception {

src/main/java/edu/ie3/datamodel/io/factory/FactoryData.java

+16
Original file line numberDiff line numberDiff line change
@@ -78,6 +78,22 @@ public <Q extends Quantity<Q>> ComparableQuantity<Q> getQuantity(String field, U
7878
return Quantities.getQuantity(getDouble(field), unit);
7979
}
8080

81+
/**
82+
* Returns field value for given field name, or empty Optional if field does not exist.
83+
*
84+
* @param field field name
85+
* @param unit unit of Quantity
86+
* @param <Q> unit type parameter
87+
* @return field value
88+
*/
89+
public <Q extends Quantity<Q>> Optional<ComparableQuantity<Q>> getQuantityOptional(
90+
String field, Unit<Q> unit) {
91+
return Optional.ofNullable(fieldsToAttributes.get(field))
92+
.filter(str -> !str.isEmpty())
93+
.map(Double::parseDouble)
94+
.map(value -> Quantities.getQuantity(value, unit));
95+
}
96+
8197
/**
8298
* Returns int value for given field name. Throws {@link FactoryException} if field does not exist
8399
* or parsing fails.

src/main/java/edu/ie3/datamodel/io/factory/timeseries/TimeBasedSimpleValueFactory.java

+13
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,10 @@ public class TimeBasedSimpleValueFactory<V extends Value>
2525
private static final String REACTIVE_POWER = "q";
2626
private static final String HEAT_DEMAND = "heatDemand";
2727

28+
/* voltage */
29+
private static final String VMAG = "vMag";
30+
private static final String VANG = "VAng";
31+
2832
public TimeBasedSimpleValueFactory(Class<? extends V> valueClasses) {
2933
super(valueClasses);
3034
}
@@ -35,6 +39,7 @@ public TimeBasedSimpleValueFactory(
3539
}
3640

3741
@Override
42+
@SuppressWarnings("unchecked")
3843
protected TimeBasedValue<V> buildModel(SimpleTimeBasedValueData<V> data) {
3944
ZonedDateTime time = timeUtil.toZonedDateTime(data.getField(TIME));
4045
V value;
@@ -64,6 +69,12 @@ protected TimeBasedValue<V> buildModel(SimpleTimeBasedValueData<V> data) {
6469
data.getQuantity(REACTIVE_POWER, REACTIVE_POWER_IN));
6570
} else if (PValue.class.isAssignableFrom(data.getTargetClass())) {
6671
value = (V) new PValue(data.getQuantity(ACTIVE_POWER, ACTIVE_POWER_IN));
72+
} else if (VoltageValue.class.isAssignableFrom(data.getTargetClass())) {
73+
value =
74+
(V)
75+
new VoltageValue(
76+
data.getQuantity(VMAG, VOLTAGE_MAGNITUDE),
77+
data.getQuantityOptional(VANG, VOLTAGE_ANGLE));
6778
} else {
6879
throw new FactoryException(
6980
"The given factory cannot handle target class '" + data.getTargetClass() + "'.");
@@ -88,6 +99,8 @@ protected List<Set<String>> getFields(Class<?> entityClass) {
8899
minConstructorParams.addAll(Arrays.asList(ACTIVE_POWER, REACTIVE_POWER));
89100
} else if (PValue.class.isAssignableFrom(entityClass)) {
90101
minConstructorParams.add(ACTIVE_POWER);
102+
} else if (VoltageValue.class.isAssignableFrom(entityClass)) {
103+
minConstructorParams.addAll(List.of(VMAG, VANG));
91104
} else {
92105
throw new FactoryException(
93106
"The given factory cannot handle target class '" + entityClass + "'.");

src/main/java/edu/ie3/datamodel/io/factory/timeseries/TimeSeriesMappingFactory.java

+4-8
Original file line numberDiff line numberDiff line change
@@ -8,16 +8,13 @@
88
import edu.ie3.datamodel.io.factory.EntityData;
99
import edu.ie3.datamodel.io.factory.EntityFactory;
1010
import edu.ie3.datamodel.io.source.TimeSeriesMappingSource;
11-
import java.util.Collections;
1211
import java.util.List;
1312
import java.util.Set;
1413
import java.util.UUID;
15-
import java.util.stream.Collectors;
16-
import java.util.stream.Stream;
1714

1815
public class TimeSeriesMappingFactory
1916
extends EntityFactory<TimeSeriesMappingSource.MappingEntry, EntityData> {
20-
private static final String PARTICIPANT = "participant";
17+
private static final String ASSET = "asset";
2118
private static final String TIME_SERIES = "timeSeries";
2219

2320
public TimeSeriesMappingFactory() {
@@ -26,14 +23,13 @@ public TimeSeriesMappingFactory() {
2623

2724
@Override
2825
protected List<Set<String>> getFields(Class<?> entityClass) {
29-
return Collections.singletonList(
30-
Stream.of(PARTICIPANT, TIME_SERIES).collect(Collectors.toSet()));
26+
return List.of(newSet(ASSET, TIME_SERIES));
3127
}
3228

3329
@Override
3430
protected TimeSeriesMappingSource.MappingEntry buildModel(EntityData data) {
35-
UUID participant = data.getUUID(PARTICIPANT);
31+
UUID asset = data.getUUID(ASSET);
3632
UUID timeSeries = data.getUUID(TIME_SERIES);
37-
return new TimeSeriesMappingSource.MappingEntry(participant, timeSeries);
33+
return new TimeSeriesMappingSource.MappingEntry(asset, timeSeries);
3834
}
3935
}

src/main/java/edu/ie3/datamodel/io/naming/timeseries/ColumnScheme.java

+3-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,8 @@ public enum ColumnScheme {
1919
HEAT_DEMAND("h", HeatDemandValue.class),
2020
ACTIVE_POWER_AND_HEAT_DEMAND("ph", HeatAndPValue.class),
2121
APPARENT_POWER_AND_HEAT_DEMAND("pqh", HeatAndSValue.class),
22-
WEATHER("weather", WeatherValue.class);
22+
WEATHER("weather", WeatherValue.class),
23+
VOLTAGE("v", VoltageValue.class);
2324

2425
private final String scheme;
2526
private final Class<? extends Value> valueClass;
@@ -57,6 +58,7 @@ public static <V extends Value> Optional<ColumnScheme> parse(Class<V> valueClass
5758
if (PValue.class.isAssignableFrom(valueClass)) return Optional.of(ACTIVE_POWER);
5859
if (HeatDemandValue.class.isAssignableFrom(valueClass)) return Optional.of(HEAT_DEMAND);
5960
if (WeatherValue.class.isAssignableFrom(valueClass)) return Optional.of(WEATHER);
61+
if (VoltageValue.class.isAssignableFrom(valueClass)) return Optional.of(VOLTAGE);
6062
return Optional.empty();
6163
}
6264
}

src/main/java/edu/ie3/datamodel/io/processor/EntityProcessor.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -55,7 +55,7 @@ protected EntityProcessor(Class<? extends T> registeredClass) throws EntityProce
5555
* during processing
5656
*/
5757
public LinkedHashMap<String, String> handleEntity(T entity) throws EntityProcessorException {
58-
if (!registeredClass.equals(entity.getClass()))
58+
if (!registeredClass.isAssignableFrom(entity.getClass()))
5959
throw new EntityProcessorException(
6060
"Cannot process "
6161
+ entity.getClass().getSimpleName()

src/main/java/edu/ie3/datamodel/io/source/TimeSeriesMappingSource.java

+16-10
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,6 @@
1111
import edu.ie3.datamodel.io.factory.EntityData;
1212
import edu.ie3.datamodel.io.factory.timeseries.TimeSeriesMappingFactory;
1313
import edu.ie3.datamodel.models.input.InputEntity;
14-
import edu.ie3.datamodel.models.input.system.SystemParticipantInput;
1514
import edu.ie3.datamodel.models.timeseries.TimeSeries;
1615
import edu.ie3.datamodel.utils.Try;
1716
import edu.ie3.datamodel.utils.Try.*;
@@ -47,7 +46,7 @@ public Map<UUID, UUID> getMapping() throws SourceException {
4746
.filter(Try::isSuccess)
4847
.map(t -> (Success<MappingEntry, FactoryException>) t)
4948
.map(Success::get)
50-
.collect(Collectors.toMap(MappingEntry::participant, MappingEntry::timeSeries));
49+
.collect(Collectors.toMap(MappingEntry::getAsset, MappingEntry::getTimeSeries));
5150
}
5251

5352
/**
@@ -80,12 +79,19 @@ private Try<MappingEntry, FactoryException> createMappingEntry(
8079

8180
// -=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
8281

83-
/** Class to represent one entry within the participant to time series mapping */
84-
public record MappingEntry(UUID participant, UUID timeSeries) implements InputEntity {
82+
/** Class to represent one entry within the asset to time series mapping */
83+
public static class MappingEntry implements InputEntity {
84+
private final UUID asset;
85+
private final UUID timeSeries;
8586

86-
/** Returns the {@link UUID} of the {@link SystemParticipantInput}. */
87-
public UUID getParticipant() {
88-
return participant;
87+
public MappingEntry(UUID asset, UUID timeSeries) {
88+
this.asset = asset;
89+
this.timeSeries = timeSeries;
90+
}
91+
92+
/** Returns the {@link UUID} of the {@link edu.ie3.datamodel.models.input.AssetInput}. */
93+
public UUID getAsset() {
94+
return asset;
8995
}
9096

9197
/** Returns the {@link UUID} of the {@link TimeSeries}. */
@@ -97,17 +103,17 @@ public UUID getTimeSeries() {
97103
public boolean equals(Object o) {
98104
if (this == o) return true;
99105
if (!(o instanceof MappingEntry that)) return false;
100-
return participant.equals(that.participant) && timeSeries.equals(that.timeSeries);
106+
return asset.equals(that.asset) && timeSeries.equals(that.timeSeries);
101107
}
102108

103109
@Override
104110
public int hashCode() {
105-
return Objects.hash(participant, timeSeries);
111+
return Objects.hash(asset, timeSeries);
106112
}
107113

108114
@Override
109115
public String toString() {
110-
return "MappingEntry{" + "participant=" + participant + ", timeSeries=" + timeSeries + '}';
116+
return "MappingEntry{" + "asset=" + asset + ", timeSeries=" + timeSeries + '}';
111117
}
112118
}
113119
}

0 commit comments

Comments
 (0)