diff --git a/CODEOWNERS b/CODEOWNERS
index c737e7c7e..a6d6e160d 100644
--- a/CODEOWNERS
+++ b/CODEOWNERS
@@ -92,6 +92,7 @@
/bundles/org.openhab.binding.elerotransmitterstick/ @vbier
/bundles/org.openhab.binding.elroconnects/ @mherwege
/bundles/org.openhab.binding.energenie/ @hmerk
+/bundles/org.openhab.binding.energidataservice/ @jlaur
/bundles/org.openhab.binding.enigma2/ @gdolfen
/bundles/org.openhab.binding.enocean/ @fruggy83
/bundles/org.openhab.binding.enphase/ @Hilbrand
diff --git a/bom/openhab-addons/pom.xml b/bom/openhab-addons/pom.xml
index 54161aed4..2a48a1987 100644
--- a/bom/openhab-addons/pom.xml
+++ b/bom/openhab-addons/pom.xml
@@ -456,6 +456,11 @@
org.openhab.binding.energenie
${project.version}
+
+ org.openhab.addons.bundles
+ org.openhab.binding.energidataservice
+ ${project.version}
+
org.openhab.addons.bundles
org.openhab.binding.enigma2
diff --git a/bundles/org.openhab.binding.energidataservice/NOTICE b/bundles/org.openhab.binding.energidataservice/NOTICE
new file mode 100644
index 000000000..38d625e34
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/NOTICE
@@ -0,0 +1,13 @@
+This content is produced and maintained by the openHAB project.
+
+* Project home: https://www.openhab.org
+
+== Declared Project Licenses
+
+This program and the accompanying materials are made available under the terms
+of the Eclipse Public License 2.0 which is available at
+https://www.eclipse.org/legal/epl-2.0/.
+
+== Source Code
+
+https://github.com/openhab/openhab-addons
diff --git a/bundles/org.openhab.binding.energidataservice/README.md b/bundles/org.openhab.binding.energidataservice/README.md
new file mode 100644
index 000000000..41bcbe7dc
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/README.md
@@ -0,0 +1,472 @@
+# Energi Data Service Binding
+
+This binding integrates electricity prices from the Danish Energi Data Service ("Open energy data from Energinet to society").
+
+This can be used to plan energy consumption, for example to calculate the cheapest period for running a dishwasher or charging an EV.
+
+## Supported Things
+
+All channels are available for thing type `service`.
+
+## Thing Configuration
+
+### `service` Thing Configuration
+
+| Name | Type | Description | Default | Required |
+|----------------|---------|---------------------------------------------------|---------------|----------|
+| priceArea | text | Price area for spot prices (same as bidding zone) | | yes |
+| currencyCode | text | Currency code in which to obtain spot prices | DKK | no |
+| gridCompanyGLN | integer | Global Location Number of the Grid Company | | no |
+| energinetGLN | integer | Global Location Number of Energinet | 5790000432752 | no |
+
+#### Global Location Number of the Grid Company
+
+The Global Location Number of your grid company can be selected from a built-in list of grid companies.
+To find the company in your area, you can go to [Find netselskab](https://greenpowerdenmark.dk/vejledning-teknik/nettilslutning/find-netselskab), enter your address, and the company will be shown.
+
+If your company is not on the list, you can configure it manually.
+To obtain the Global Location Number of your grid company:
+
+- Open a browser and go to [Eloverblik](https://eloverblik.dk/).
+- Click "Private customers" and log in with MitID (confirmation will appear as Energinet).
+- Click "Retrieve data" and select "Price data".
+- Open the file and look for the rows having **Price_type** = "Subscription".
+- In the columns **Name** and/or **Description** you should see the name of your grid company.
+- In column **Owner** you can find the GLN ("Global Location Number").
+- Most rows will have this **Owner**. If in doubt, try to look for rows __not__ having 5790000432752 as owner.
+
+## Channels
+
+### Channel Group `electricity`
+
+| Channel | Type | Description | Advanced |
+|-------------------------|--------|---------------------------------------------------------------------------------------|----------|
+| spot-price | Number | Current spot price in DKK or EUR per kWh | no |
+| net-tariff | Number | Current net tariff in DKK per kWh. Only available when `gridCompanyGLN` is configured | no |
+| system-tariff | Number | Current system tariff in DKK per kWh | no |
+| electricity-tax | Number | Current electricity tax in DKK per kWh | no |
+| transmission-net-tariff | Number | Current transmission net tariff in DKK per kWh | no |
+| hourly-prices | String | JSON array with hourly prices from 12 hours ago and onward | yes |
+
+_Please note:_ There is no channel providing the total price.
+Instead, create a group item with `SUM` as aggregate function and add the individual price items as children.
+This has the following advantages:
+
+- Full customization possible: Freely choose the channels which should be included in the total.
+- An additional item containing the kWh fee from your electricity supplier can be added also.
+- Spot price can be configured in EUR while tariffs are in DKK.
+
+#### Value-Added Tax
+
+VAT is not included in any of the prices.
+To include VAT for items linked to the `Number` channels, the [VAT profile](https://www.openhab.org/addons/transformations/vat/) can be used.
+This must be installed separately.
+Once installed, simply select "Value-Added Tax" as Profile when linking an item.
+
+#### Net Tariff
+
+Discounts are automatically taken into account for channel `net-tariff` so that it represents the actual price.
+
+The tariffs are downloaded using pre-configured filters for the different [Grid Company GLN's](#global-location-number-of-the-grid-company).
+If your company is not in the list, or the filters are not working, they can be manually overridden.
+To override filters, the channel `net-tariff` has the following configuration parameters:
+
+| Name | Type | Description | Default | Required | Advanced |
+|-----------------|---------|----------------------------------------------------------------------------------------------------------------------------|---------|----------|----------|
+| chargeTypeCodes | text | Comma-separated list of charge type codes | | no | yes |
+| notes | text | Comma-separated list of notes | | no | yes |
+| start | text | Query start date parameter expressed as either YYYY-MM-DD or dynamically as one of StartOfDay, StartOfMonth or StartOfYear | | no | yes |
+
+The parameters `chargeTypeCodes` and `notes` are logically combined with "AND", so if only one parameter is needed for the filter, only provide this parameter and leave the other one empty.
+Using any of these parameters will override the pre-configured filter entirely.
+
+The parameter `start` can be used independently to override the query start date parameter.
+If used while leaving `chargeTypeCodes` and `notes` empty, only the date will be overridden.
+
+Determining the right filters can be tricky, so if in doubt ask in the community forum.
+See also [Datahub Price List](https://www.energidataservice.dk/tso-electricity/DatahubPricelist).
+
+##### Filter Examples
+
+_N1:_
+| Parameter | Value |
+|-----------------|------------|
+| chargeTypeCodes | CD,CD R |
+| notes | |
+
+_Nord Energi Net:_
+| Parameter | Value |
+|-----------------|------------|
+| chargeTypeCodes | TA031U200 |
+| notes | Nettarif C |
+
+#### Hourly Prices
+
+The format of the `hourly-prices` JSON array is as follows:
+
+```json
+[
+ {
+ "hourStart": "2023-01-24T15:00:00Z",
+ "spotPrice": 1.67076001,
+ "spotPriceCurrency": "DKK",
+ "netTariff": 0.432225,
+ "systemTariff": 0.054000,
+ "electricityTax": 0.008000,
+ "transmissionNetTariff": 0.058000
+ },
+ {
+ "hourStart": "2023-01-24T16:00:00Z",
+ "spotPrice": 1.859880005,
+ "spotPriceCurrency": "DKK",
+ "netTariff": 1.05619,
+ "systemTariff": 0.054000,
+ "electricityTax": 0.008000,
+ "transmissionNetTariff": 0.058000
+ }
+]
+```
+
+Future spot prices for the next day are usually available around 13:00 CET and are fetched around that time.
+Historic prices older than 12 hours are removed from the JSON array each hour.
+
+## Thing Actions
+
+Thing actions can be used to perform calculations as well as import prices directly into rules without deserializing JSON from the [hourly-prices](#hourly-prices) channel.
+This is more convenient, much faster, and provides automatic summation of the price elements of interest.
+
+Actions use cached data for performing operations.
+Since data is only fetched when an item is linked to a channel, there might not be any cached data available.
+In this case the data will be fetched on demand and cached afterwards.
+The first action triggered on a given day may therefore be a bit slower, and is also prone to failing if the server call fails for any reason.
+This potential problem can be prevented by linking the individual channels to items, or by linking the `hourly-prices` channel to an item.
+
+### `calculateCheapestPeriod`
+
+This action will determine the cheapest period for using energy.
+It comes in four variants with different input parameters.
+
+The result is a `Map` with the following keys:
+
+| Key | Type | Description |
+|--------------------|--------------|-------------------------------------------------------|
+| CheapestStart | `Instant` | Start time of cheapest calculated period |
+| LowestPrice | `BigDecimal` | The total price when starting at cheapest start |
+| MostExpensiveStart | `Instant` | Start time of most expensive calculated period |
+| HighestPrice | `BigDecimal` | The total price when starting at most expensive start |
+
+#### `calculateCheapestPeriod` from Duration
+
+| Parameter | Type | Description |
+|--------------------|-----------------------------|--------------------------------------------------------|
+| earliestStart | `Instant` | Earliest start time allowed |
+| latestEnd | `Instant` | Latest end time allowed |
+| duration | `Duration` | The duration to fit within the timeslot |
+
+This is a convenience method that can be used when the power consumption is not known.
+The calculation will assume linear consumption and will find the best timeslot based on that.
+For this reason the resulting `Map` will not contain the keys `LowestPrice` and `HighestPrice`.
+
+Example:
+
+```javascript
+var Map result = actions.calculateCheapestPeriod(now.toInstant(), now.plusHours(12).toInstant(), Duration.ofMinutes(90))
+```
+
+#### `calculateCheapestPeriod` from Duration and Power
+
+| Parameter | Type | Description |
+|--------------------|-----------------------------|--------------------------------------------------------|
+| earliestStart | `Instant` | Earliest start time allowed |
+| latestEnd | `Instant` | Latest end time allowed |
+| duration | `Duration` | The duration to fit within the timeslot |
+| power | `QuantityType` | Linear power consumption |
+
+This action is identical to the variant above, but with a known linear power consumption.
+As a result the price is also included in the result.
+
+Example:
+
+```javascript
+var Map result = actions.calculateCheapestPeriod(now.toInstant(), now.plusHours(12).toInstant(), Duration.ofMinutes(90), 250 | W)
+```
+
+#### `calculateCheapestPeriod` from Power Phases
+
+| Parameter | Type | Description |
+|--------------------|-----------------------------|--------------------------------------------------------|
+| earliestStart | `Instant` | Earliest start time allowed |
+| latestEnd | `Instant` | Latest end time allowed |
+| durationPhases | `List` | List of durations for the phases |
+| powerPhases | `List>` | List of power consumption for each corresponding phase |
+
+This variant is similar to the one above, but is based on a supplied timetable.
+
+The timetable is supplied as two individual parameters, `durationPhases` and `powerPhases`, which must have the same size.
+This can be considered as different phases of using power, so each list member represents a period with a linear use of power.
+`durationPhases` should be a List populated by `Duration` objects, while `powerPhases` should be a List populated by `QuantityType` objects for that duration of time.
+
+Example:
+
+```javascript
+val ArrayList durationPhases = new ArrayList()
+durationPhases.add(Duration.ofMinutes(37))
+durationPhases.add(Duration.ofMinutes(8))
+durationPhases.add(Duration.ofMinutes(4))
+durationPhases.add(Duration.ofMinutes(2))
+durationPhases.add(Duration.ofMinutes(4))
+durationPhases.add(Duration.ofMinutes(36))
+durationPhases.add(Duration.ofMinutes(41))
+durationPhases.add(Duration.ofMinutes(104))
+
+val ArrayList> powerPhases = new ArrayList>()
+powerPhases.add(162.162 | W)
+powerPhases.add(750 | W)
+powerPhases.add(1500 | W)
+powerPhases.add(3000 | W)
+powerPhases.add(1500 | W)
+powerPhases.add(166.666 | W)
+powerPhases.add(146.341 | W)
+powerPhases.add(0 | W)
+
+var Map result = actions.calculateCheapestPeriod(now.toInstant(), now.plusHours(12).toInstant(), durationPhases, powerPhases)
+```
+
+Please note that the total duration will be calculated automatically as a sum of provided duration phases.
+Therefore, if the total duration is longer than the sum of phase durations, the remaining duration must be provided as last item with a corresponding 0 W power item.
+This is to ensure that the full program will finish before the provided `latestEnd`.
+
+#### `calculateCheapestPeriod` from Energy per Phase
+
+| Parameter | Type | Description |
+|--------------------|-----------------------------|--------------------------------------------------------|
+| earliestStart | `Instant` | Earliest start time allowed |
+| latestEnd | `Instant` | Latest end time allowed |
+| totalDuration | `Duration` | The total duration of all phases |
+| durationPhases | `List` | List of durations for the phases |
+| energyUsedPerPhase | `QuantityType` | Fixed amount of energy used per phase |
+
+This variant will assign the provided amount of energy into each phase.
+The use case for this variant is a simplification of the previous variant.
+For example, a dishwasher may provide energy consumption in 0.1 kWh steps.
+In this case it's a simple task to create a timetable accordingly without having to calculate the average power consumption per phase.
+Since a last phase may use no significant energy, the total duration must be provided also.
+
+Example:
+
+```javascript
+val ArrayList durationPhases = new ArrayList()
+durationPhases.add(Duration.ofMinutes(37))
+durationPhases.add(Duration.ofMinutes(8))
+durationPhases.add(Duration.ofMinutes(4))
+durationPhases.add(Duration.ofMinutes(2))
+durationPhases.add(Duration.ofMinutes(4))
+durationPhases.add(Duration.ofMinutes(36))
+durationPhases.add(Duration.ofMinutes(41))
+
+// 0.7 kWh is used in total (number of phases × energy used per phase)
+var Map result = actions.calculateCheapestPeriod(now.toInstant(), now.plusHours(12).toInstant(), Duration.ofMinutes(236), phases, 0.1 | kWh)
+```
+
+### `calculatePrice`
+
+| Parameter | Type | Description |
+|--------------------|-----------------------------|--------------------------------------------------------|
+| start | `Instant` | Start time |
+| end | `Instant` | End time |
+| power | `QuantityType` | Linear power consumption |
+
+**Result:** Price as `BigDecimal`.
+
+This action calculates the price for using given amount of power in the period from `start` till `end`.
+
+Example:
+
+```javascript
+var price = actions.calculatePrice(now.toInstant(), now.plusHours(4).toInstant, 200 | W)
+```
+
+### `getPrices`
+
+| Parameter | Type | Description |
+|--------------------|-----------------------------|--------------------------------------------------------|
+| priceElements | `String` | Comma-separated list of price elements to include |
+
+**Result:** `Map`
+
+The parameter `priceElements` is a case-insensitive comma-separated list of price elements to include in the returned hourly prices.
+These elements can be requested:
+
+| Price element | Description |
+|-----------------------|-------------------------|
+| SpotPrice | Spot price |
+| NetTariff | Net tariff |
+| SystemTariff | System tariff |
+| ElectricityTax | Electricity tax |
+| TransmissionNetTariff | Transmission net tariff |
+
+Using `null` as parameter returns the total prices including all price elements.
+
+Example:
+
+```javascript
+var priceMap = actions.getPrices("SpotPrice,NetTariff");
+```
+
+## Full Example
+
+### Thing Configuration
+
+```java
+Thing energidataservice:service:energidataservice "Energi Data Service" [ priceArea="DK1", currencyCode="DKK", gridCompanyGLN="5790001089030" ] {
+ Channels:
+ Number : electricity#net-tariff [ chargeTypeCodes="CD,CD R", start="StartOfYear" ]
+}
+```
+
+### Item Configuration
+
+```java
+Group:Number:SUM TotalPrice "Current Total Price"
+Number SpotPrice "Current Spot Price" (TotalPrice) { channel="energidataservice:service:energidataservice:electricity#spot-price" [profile="transform:VAT"] }
+Number NetTariff "Current Net Tariff" (TotalPrice) { channel="energidataservice:service:energidataservice:electricity#net-tariff" [profile="transform:VAT"] }
+Number SystemTariff "Current System Tariff" (TotalPrice) { channel="energidataservice:service:energidataservice:electricity#system-tariff" [profile="transform:VAT"] }
+Number ElectricityTax "Current Electricity Tax" (TotalPrice) { channel="energidataservice:service:energidataservice:electricity#electricity-tax" [profile="transform:VAT"] }
+Number TransmissionNetTariff "Current Transmission Tariff" (TotalPrice) { channel="energidataservice:service:energidataservice:electricity#transmission-net-tariff" [profile="transform:VAT"] }
+String HourlyPrices "Hourly Prices" { channel="energidataservice:service:energidataservice:electricity#hourly-prices" }
+```
+
+### Thing Actions Example
+
+:::: tabs
+
+::: tab DSL
+
+```javascript
+import java.time.Duration
+import java.util.ArrayList
+import java.util.Map
+import java.time.temporal.ChronoUnit
+
+val actions = getActions("energidataservice", "energidataservice:service:energidataservice")
+
+var priceMap = actions.getPrices(null)
+var hourStart = now.toInstant().truncatedTo(ChronoUnit.HOURS)
+logInfo("Current total price excl. VAT", priceMap.get(hourStart).toString)
+
+var priceMap = actions.getPrices("SpotPrice,NetTariff");
+logInfo("Current spot price + net tariff excl. VAT", priceMap.get(hourStart).toString)
+
+var price = actions.calculatePrice(Instant.now, now.plusHours(1).toInstant, 150 | W)
+logInfo("Total price for using 150 W for the next hour", price.toString)
+
+val ArrayList durationPhases = new ArrayList()
+durationPhases.add(Duration.ofMinutes(37))
+durationPhases.add(Duration.ofMinutes(8))
+durationPhases.add(Duration.ofMinutes(4))
+durationPhases.add(Duration.ofMinutes(2))
+durationPhases.add(Duration.ofMinutes(4))
+durationPhases.add(Duration.ofMinutes(36))
+durationPhases.add(Duration.ofMinutes(41))
+durationPhases.add(Duration.ofMinutes(104))
+
+val ArrayList> consumptionPhases = new ArrayList>()
+consumptionPhases.add(162.162 | W)
+consumptionPhases.add(750 | W)
+consumptionPhases.add(1500 | W)
+consumptionPhases.add(3000 | W)
+consumptionPhases.add(1500 | W)
+consumptionPhases.add(166.666 | W)
+consumptionPhases.add(146.341 | W)
+consumptionPhases.add(0 | W)
+
+var Map result = actions.calculateCheapestPeriod(now.toInstant, now.plusHours(24).toInstant, durationPhases, consumptionPhases)
+logInfo("Cheapest start", (result.get("CheapestStart") as Instant).toString)
+logInfo("Lowest price", (result.get("LowestPrice") as Number).doubleValue.toString)
+logInfo("Highest price", (result.get("HighestPrice") as Number).doubleValue.toString)
+logInfo("Most expensive start", (result.get("MostExpensiveStart") as Instant).toString)
+
+// This is a simpler version taking advantage of the fact that each interval here represents 0.1 kWh of consumed energy.
+// In this example we have to provide the total duration to make sure we fit the latest end. This is because there is no
+// registered consumption in the last phase.
+val ArrayList durationPhases = new ArrayList()
+durationPhases.add(Duration.ofMinutes(37))
+durationPhases.add(Duration.ofMinutes(8))
+durationPhases.add(Duration.ofMinutes(4))
+durationPhases.add(Duration.ofMinutes(2))
+durationPhases.add(Duration.ofMinutes(4))
+durationPhases.add(Duration.ofMinutes(36))
+durationPhases.add(Duration.ofMinutes(41))
+
+var Map result = actions.calculateCheapestPeriod(now.toInstant(), now.plusHours(24).toInstant(), Duration.ofMinutes(236), durationPhases, 0.1 | kWh)
+```
+
+:::
+
+::: tab JavaScript
+
+```javascript
+var edsActions = actions.get("energidataservice", "energidataservice:service:energidataservice");
+
+// Get prices and convert to JavaScript Map with Instant string representation as keys.
+var priceMap = new Map();
+utils.javaMapToJsMap(edsActions.getPrices()).forEach((value, key) => {
+ priceMap.set(key.toString(), value);
+});
+
+var hourStart = time.Instant.now().truncatedTo(time.ChronoUnit.HOURS);
+console.log("Current total price excl. VAT: " + priceMap.get(hourStart.toString()));
+
+utils.javaMapToJsMap(edsActions.getPrices("SpotPrice,NetTariff")).forEach((value, key) => {
+ priceMap.set(key.toString(), value);
+});
+console.log("Current spot price + net tariff excl. VAT: " + priceMap.get(hourStart.toString()));
+
+var price = edsActions.calculatePrice(time.Instant.now(), time.Instant.now().plusSeconds(3600), Quantity("150 W"));
+console.log("Total price for using 150 W for the next hour: " + price.toString());
+
+var durationPhases = [];
+durationPhases.push(time.Duration.ofMinutes(37));
+durationPhases.push(time.Duration.ofMinutes(8));
+durationPhases.push(time.Duration.ofMinutes(4));
+durationPhases.push(time.Duration.ofMinutes(2));
+durationPhases.push(time.Duration.ofMinutes(4));
+durationPhases.push(time.Duration.ofMinutes(36));
+durationPhases.push(time.Duration.ofMinutes(41));
+durationPhases.push(time.Duration.ofMinutes(104));
+
+var consumptionPhases = [];
+consumptionPhases.push(Quantity("162.162 W"));
+consumptionPhases.push(Quantity("750 W"));
+consumptionPhases.push(Quantity("1500 W"));
+consumptionPhases.push(Quantity("3000 W"));
+consumptionPhases.push(Quantity("1500 W"));
+consumptionPhases.push(Quantity("166.666 W"));
+consumptionPhases.push(Quantity("146.341 W"));
+consumptionPhases.push(Quantity("0 W"));
+
+var result = edsActions.calculateCheapestPeriod(time.Instant.now(), time.Instant.now().plusSeconds(24*60*60), durationPhases, consumptionPhases);
+
+console.log("Cheapest start: " + result.get("CheapestStart").toString());
+console.log("Lowest price: " + result.get("LowestPrice"));
+console.log("Highest price: " + result.get("HighestPrice"));
+console.log("Most expensive start: " + result.get("MostExpensiveStart").toString());
+
+// This is a simpler version taking advantage of the fact that each interval here represents 0.1 kWh of consumed energy.
+// In this example we have to provide the total duration to make sure we fit the latest end. This is because there is no
+// registered consumption in the last phase.
+var durationPhases = [];
+durationPhases.push(time.Duration.ofMinutes(37));
+durationPhases.push(time.Duration.ofMinutes(8));
+durationPhases.push(time.Duration.ofMinutes(4));
+durationPhases.push(time.Duration.ofMinutes(2));
+durationPhases.push(time.Duration.ofMinutes(4));
+durationPhases.push(time.Duration.ofMinutes(36));
+durationPhases.push(time.Duration.ofMinutes(41));
+
+var result = edsActions.calculateCheapestPeriod(time.Instant.now(), time.Instant.now().plusSeconds(24*60*60), time.Duration.ofMinutes(236), durationPhases, Quantity("0.1 kWh"));
+```
+
+:::
+
+::::
diff --git a/bundles/org.openhab.binding.energidataservice/pom.xml b/bundles/org.openhab.binding.energidataservice/pom.xml
new file mode 100644
index 000000000..4e4d39fdd
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/pom.xml
@@ -0,0 +1,25 @@
+
+
+
+ 4.0.0
+
+
+ org.openhab.addons.bundles
+ org.openhab.addons.reactor.bundles
+ 4.0.0-SNAPSHOT
+
+
+ org.openhab.binding.energidataservice
+
+ openHAB Add-ons :: Bundles :: Energi Data Service Binding
+
+
+
+ com.google.code.gson
+ gson
+ 2.10.1
+
+
+
+
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/feature/feature.xml b/bundles/org.openhab.binding.energidataservice/src/main/feature/feature.xml
new file mode 100644
index 000000000..69ed1af64
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/feature/feature.xml
@@ -0,0 +1,9 @@
+
+
+ mvn:org.openhab.core.features.karaf/org.openhab.core.features.karaf.openhab-core/${ohc.version}/xml/features
+
+
+ openhab-runtime-base
+ mvn:org.openhab.addons.bundles/org.openhab.binding.energidataservice/${project.version}
+
+
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/ApiController.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/ApiController.java
new file mode 100644
index 000000000..484986ff8
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/ApiController.java
@@ -0,0 +1,254 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal;
+
+import static org.openhab.binding.energidataservice.internal.EnergiDataServiceBindingConstants.*;
+
+import java.time.Instant;
+import java.time.LocalDateTime;
+import java.time.format.DateTimeFormatter;
+import java.util.Arrays;
+import java.util.Collection;
+import java.util.Currency;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+import java.util.Objects;
+import java.util.concurrent.ExecutionException;
+import java.util.concurrent.TimeoutException;
+import java.util.stream.Collectors;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.HttpClient;
+import org.eclipse.jetty.client.api.ContentResponse;
+import org.eclipse.jetty.client.api.Request;
+import org.eclipse.jetty.http.HttpFields;
+import org.eclipse.jetty.http.HttpMethod;
+import org.eclipse.jetty.http.HttpStatus;
+import org.openhab.binding.energidataservice.internal.api.ChargeType;
+import org.openhab.binding.energidataservice.internal.api.DatahubTariffFilter;
+import org.openhab.binding.energidataservice.internal.api.DateQueryParameter;
+import org.openhab.binding.energidataservice.internal.api.GlobalLocationNumber;
+import org.openhab.binding.energidataservice.internal.api.dto.DatahubPricelistRecord;
+import org.openhab.binding.energidataservice.internal.api.dto.DatahubPricelistRecords;
+import org.openhab.binding.energidataservice.internal.api.dto.ElspotpriceRecord;
+import org.openhab.binding.energidataservice.internal.api.dto.ElspotpriceRecords;
+import org.openhab.binding.energidataservice.internal.api.serialization.InstantDeserializer;
+import org.openhab.binding.energidataservice.internal.api.serialization.LocalDateTimeDeserializer;
+import org.openhab.binding.energidataservice.internal.exception.DataServiceException;
+import org.openhab.core.i18n.TimeZoneProvider;
+import org.osgi.framework.FrameworkUtil;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import com.google.gson.Gson;
+import com.google.gson.GsonBuilder;
+import com.google.gson.JsonSyntaxException;
+
+/**
+ * The {@link ApiController} is responsible for interacting with Energi Data Service.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class ApiController {
+ private static final String ENDPOINT = "https://api.energidataservice.dk/";
+ private static final String DATASET_PATH = "dataset/";
+
+ private static final String DATASET_NAME_SPOT_PRICES = "Elspotprices";
+ private static final String DATASET_NAME_DATAHUB_PRICELIST = "DatahubPricelist";
+
+ private static final String FILTER_KEY_PRICE_AREA = "PriceArea";
+ private static final String FILTER_KEY_CHARGE_TYPE = "ChargeType";
+ private static final String FILTER_KEY_CHARGE_TYPE_CODE = "ChargeTypeCode";
+ private static final String FILTER_KEY_GLN_NUMBER = "GLN_Number";
+ private static final String FILTER_KEY_NOTE = "Note";
+
+ private static final String HEADER_REMAINING_CALLS = "RemainingCalls";
+ private static final String HEADER_TOTAL_CALLS = "TotalCalls";
+
+ private final Logger logger = LoggerFactory.getLogger(ApiController.class);
+ private final Gson gson = new GsonBuilder() //
+ .registerTypeAdapter(Instant.class, new InstantDeserializer()) //
+ .registerTypeAdapter(LocalDateTime.class, new LocalDateTimeDeserializer()) //
+ .create();
+ private final HttpClient httpClient;
+ private final TimeZoneProvider timeZoneProvider;
+ private final String userAgent;
+
+ public ApiController(HttpClient httpClient, TimeZoneProvider timeZoneProvider) {
+ this.httpClient = httpClient;
+ this.timeZoneProvider = timeZoneProvider;
+ userAgent = "openHAB/" + FrameworkUtil.getBundle(this.getClass()).getVersion().toString();
+ }
+
+ /**
+ * Retrieve spot prices for requested area and in requested {@link Currency}.
+ *
+ * @param priceArea Usually DK1 or DK2
+ * @param currency DKK or EUR
+ * @param start Specifies the start point of the period for the data request
+ * @param properties Map of properties which will be updated with metadata from headers
+ * @return Records with pairs of hour start and price in requested currency.
+ * @throws InterruptedException
+ * @throws DataServiceException
+ */
+ public ElspotpriceRecord[] getSpotPrices(String priceArea, Currency currency, DateQueryParameter start,
+ Map properties) throws InterruptedException, DataServiceException {
+ if (!SUPPORTED_CURRENCIES.contains(currency)) {
+ throw new IllegalArgumentException("Invalid currency " + currency.getCurrencyCode());
+ }
+
+ Request request = httpClient.newRequest(ENDPOINT + DATASET_PATH + DATASET_NAME_SPOT_PRICES)
+ .param("start", start.toString()) //
+ .param("filter", "{\"" + FILTER_KEY_PRICE_AREA + "\":\"" + priceArea + "\"}") //
+ .param("columns", "HourUTC,SpotPrice" + currency) //
+ .agent(userAgent) //
+ .method(HttpMethod.GET);
+
+ logger.trace("GET request for {}", request.getURI());
+
+ try {
+ ContentResponse response = request.send();
+
+ updatePropertiesFromResponse(response, properties);
+
+ int status = response.getStatus();
+ if (!HttpStatus.isSuccess(status)) {
+ throw new DataServiceException("The request failed with HTTP error " + status, status);
+ }
+ String responseContent = response.getContentAsString();
+ if (responseContent.isEmpty()) {
+ throw new DataServiceException("Empty response");
+ }
+ logger.trace("Response content: '{}'", responseContent);
+
+ ElspotpriceRecords records = gson.fromJson(responseContent, ElspotpriceRecords.class);
+ if (records == null) {
+ throw new DataServiceException("Error parsing response");
+ }
+
+ if (records.total() == 0 || Objects.isNull(records.records()) || records.records().length == 0) {
+ throw new DataServiceException("No records");
+ }
+
+ return Arrays.stream(records.records()).filter(Objects::nonNull).toArray(ElspotpriceRecord[]::new);
+ } catch (JsonSyntaxException e) {
+ throw new DataServiceException("Error parsing response", e);
+ } catch (TimeoutException | ExecutionException e) {
+ throw new DataServiceException(e);
+ }
+ }
+
+ private void updatePropertiesFromResponse(ContentResponse response, Map properties) {
+ HttpFields headers = response.getHeaders();
+ String remainingCalls = headers.get(HEADER_REMAINING_CALLS);
+ if (remainingCalls != null) {
+ properties.put(PROPERTY_REMAINING_CALLS, remainingCalls);
+ }
+ String totalCalls = headers.get(HEADER_TOTAL_CALLS);
+ if (totalCalls != null) {
+ properties.put(PROPERTY_TOTAL_CALLS, totalCalls);
+ }
+ DateTimeFormatter formatter = DateTimeFormatter.ofPattern(PROPERTY_DATETIME_FORMAT);
+ properties.put(PROPERTY_LAST_CALL, LocalDateTime.now(timeZoneProvider.getTimeZone()).format(formatter));
+ }
+
+ /**
+ * Retrieve datahub pricelists for requested GLN and charge type/charge type code.
+ *
+ * @param globalLocationNumber Global Location Number of the Charge Owner
+ * @param chargeType Charge type (Subscription, Fee or Tariff).
+ * @param tariffFilter Tariff filter (charge type codes and notes).
+ * @param properties Map of properties which will be updated with metadata from headers
+ * @return Price list for requested GLN and note.
+ * @throws InterruptedException
+ * @throws DataServiceException
+ */
+ public Collection getDatahubPriceLists(GlobalLocationNumber globalLocationNumber,
+ ChargeType chargeType, DatahubTariffFilter tariffFilter, Map properties)
+ throws InterruptedException, DataServiceException {
+ String columns = "ValidFrom,ValidTo,ChargeTypeCode";
+ for (int i = 1; i < 25; i++) {
+ columns += ",Price" + i;
+ }
+
+ Map> filterMap = new HashMap<>(Map.of( //
+ FILTER_KEY_GLN_NUMBER, List.of(globalLocationNumber.toString()), //
+ FILTER_KEY_CHARGE_TYPE, List.of(chargeType.toString())));
+
+ Collection chargeTypeCodes = tariffFilter.getChargeTypeCodesAsStrings();
+ if (!chargeTypeCodes.isEmpty()) {
+ filterMap.put(FILTER_KEY_CHARGE_TYPE_CODE, chargeTypeCodes);
+ }
+
+ Collection notes = tariffFilter.getNotes();
+ if (!notes.isEmpty()) {
+ filterMap.put(FILTER_KEY_NOTE, notes);
+ }
+
+ Request request = httpClient.newRequest(ENDPOINT + DATASET_PATH + DATASET_NAME_DATAHUB_PRICELIST)
+ .param("filter", mapToFilter(filterMap)) //
+ .param("columns", columns) //
+ .agent(userAgent) //
+ .method(HttpMethod.GET);
+
+ DateQueryParameter dateQueryParameter = tariffFilter.getDateQueryParameter();
+ if (!dateQueryParameter.isEmpty()) {
+ request = request.param("start", dateQueryParameter.toString());
+ }
+
+ logger.trace("GET request for {}", request.getURI());
+
+ try {
+ ContentResponse response = request.send();
+
+ updatePropertiesFromResponse(response, properties);
+
+ int status = response.getStatus();
+ if (!HttpStatus.isSuccess(status)) {
+ throw new DataServiceException("The request failed with HTTP error " + status, status);
+ }
+ String responseContent = response.getContentAsString();
+ if (responseContent.isEmpty()) {
+ throw new DataServiceException("Empty response");
+ }
+ logger.trace("Response content: '{}'", responseContent);
+
+ DatahubPricelistRecords records = gson.fromJson(responseContent, DatahubPricelistRecords.class);
+ if (records == null) {
+ throw new DataServiceException("Error parsing response");
+ }
+
+ if (records.limit() > 0 && records.limit() < records.total()) {
+ logger.warn("{} price list records available, but only {} returned.", records.total(), records.limit());
+ }
+
+ if (Objects.isNull(records.records())) {
+ return List.of();
+ }
+
+ return Arrays.stream(records.records()).filter(Objects::nonNull).toList();
+ } catch (JsonSyntaxException e) {
+ throw new DataServiceException("Error parsing response", e);
+ } catch (TimeoutException | ExecutionException e) {
+ throw new DataServiceException(e);
+ }
+ }
+
+ private String mapToFilter(Map> map) {
+ return "{" + map.entrySet().stream().map(
+ e -> "\"" + e.getKey() + "\":[\"" + e.getValue().stream().collect(Collectors.joining("\",\"")) + "\"]")
+ .collect(Collectors.joining(",")) + "}";
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/CacheManager.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/CacheManager.java
new file mode 100644
index 000000000..d73ee12dd
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/CacheManager.java
@@ -0,0 +1,443 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal;
+
+import static org.openhab.binding.energidataservice.internal.EnergiDataServiceBindingConstants.*;
+
+import java.math.BigDecimal;
+import java.time.Clock;
+import java.time.Instant;
+import java.time.LocalDate;
+import java.time.LocalDateTime;
+import java.time.LocalTime;
+import java.time.ZonedDateTime;
+import java.time.temporal.ChronoUnit;
+import java.util.ArrayList;
+import java.util.Collection;
+import java.util.Currency;
+import java.util.HashMap;
+import java.util.Map;
+import java.util.concurrent.ConcurrentHashMap;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.energidataservice.internal.api.dto.DatahubPricelistRecord;
+import org.openhab.binding.energidataservice.internal.api.dto.ElspotpriceRecord;
+
+/**
+ * The {@link CacheManager} is responsible for maintaining a cache of received
+ * data from Energi Data Service.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class CacheManager {
+
+ public static final int NUMBER_OF_HISTORIC_HOURS = 24;
+ public static final int SPOT_PRICE_MAX_CACHE_SIZE = 24 + 11 + NUMBER_OF_HISTORIC_HOURS;
+ public static final int TARIFF_MAX_CACHE_SIZE = 24 * 2 + NUMBER_OF_HISTORIC_HOURS;
+
+ private final Clock clock;
+ private final PriceListParser priceListParser = new PriceListParser();
+
+ private Collection netTariffRecords = new ArrayList<>();
+ private Collection systemTariffRecords = new ArrayList<>();
+ private Collection electricityTaxRecords = new ArrayList<>();
+ private Collection transmissionNetTariffRecords = new ArrayList<>();
+
+ private Map spotPriceMap = new ConcurrentHashMap<>(SPOT_PRICE_MAX_CACHE_SIZE);
+ private Map netTariffMap = new ConcurrentHashMap<>(TARIFF_MAX_CACHE_SIZE);
+ private Map systemTariffMap = new ConcurrentHashMap<>(TARIFF_MAX_CACHE_SIZE);
+ private Map electricityTaxMap = new ConcurrentHashMap<>(TARIFF_MAX_CACHE_SIZE);
+ private Map transmissionNetTariffMap = new ConcurrentHashMap<>(TARIFF_MAX_CACHE_SIZE);
+
+ public CacheManager() {
+ this(Clock.systemDefaultZone());
+ }
+
+ public CacheManager(Clock clock) {
+ this.clock = clock.withZone(NORD_POOL_TIMEZONE);
+ }
+
+ /**
+ * Clear all cached data.
+ */
+ public void clear() {
+ netTariffRecords.clear();
+ systemTariffRecords.clear();
+ electricityTaxRecords.clear();
+ transmissionNetTariffRecords.clear();
+
+ spotPriceMap.clear();
+ netTariffMap.clear();
+ systemTariffMap.clear();
+ electricityTaxMap.clear();
+ transmissionNetTariffMap.clear();
+ }
+
+ /**
+ * Convert and cache the supplied {@link ElspotpriceRecord}s.
+ *
+ * @param records The records as received from Energi Data Service.
+ * @param currency The currency in which the records were requested.
+ */
+ public void putSpotPrices(ElspotpriceRecord[] records, Currency currency) {
+ boolean isDKK = EnergiDataServiceBindingConstants.CURRENCY_DKK.equals(currency);
+ for (ElspotpriceRecord record : records) {
+ spotPriceMap.put(record.hour(),
+ (isDKK ? record.spotPriceDKK() : record.spotPriceEUR()).divide(BigDecimal.valueOf(1000)));
+ }
+ cleanup();
+ }
+
+ /**
+ * Replace current "raw"/unprocessed net tariff records in cache.
+ * Map of hourly tariffs will be updated automatically.
+ *
+ * @param records to cache
+ */
+ public void putNetTariffs(Collection records) {
+ putDatahubRecords(netTariffRecords, records);
+ updateNetTariffs();
+ }
+
+ /**
+ * Replace current "raw"/unprocessed system tariff records in cache.
+ * Map of hourly tariffs will be updated automatically.
+ *
+ * @param records to cache
+ */
+ public void putSystemTariffs(Collection records) {
+ putDatahubRecords(systemTariffRecords, records);
+ updateSystemTariffs();
+ }
+
+ /**
+ * Replace current "raw"/unprocessed electricity tax records in cache.
+ * Map of hourly taxes will be updated automatically.
+ *
+ * @param records to cache
+ */
+ public void putElectricityTaxes(Collection records) {
+ putDatahubRecords(electricityTaxRecords, records);
+ updateElectricityTaxes();
+ }
+
+ /**
+ * Replace current "raw"/unprocessed transmission net tariff records in cache.
+ * Map of hourly tariffs will be updated automatically.
+ *
+ * @param records to cache
+ */
+ public void putTransmissionNetTariffs(Collection records) {
+ putDatahubRecords(transmissionNetTariffRecords, records);
+ updateTransmissionNetTariffs();
+ }
+
+ private void putDatahubRecords(Collection destination,
+ Collection source) {
+ LocalDateTime localHourStart = LocalDateTime.now(clock.withZone(DATAHUB_TIMEZONE))
+ .minus(NUMBER_OF_HISTORIC_HOURS, ChronoUnit.HOURS).truncatedTo(ChronoUnit.HOURS);
+
+ destination.clear();
+ destination.addAll(source.stream().filter(r -> !r.validTo().isBefore(localHourStart)).toList());
+ }
+
+ /**
+ * Update map of hourly net tariffs from internal cache.
+ */
+ public void updateNetTariffs() {
+ netTariffMap = priceListParser.toHourly(netTariffRecords);
+ cleanup();
+ }
+
+ /**
+ * Update map of system tariffs from internal cache.
+ */
+ public void updateSystemTariffs() {
+ systemTariffMap = priceListParser.toHourly(systemTariffRecords);
+ cleanup();
+ }
+
+ /**
+ * Update map of electricity taxes from internal cache.
+ */
+ public void updateElectricityTaxes() {
+ electricityTaxMap = priceListParser.toHourly(electricityTaxRecords);
+ cleanup();
+ }
+
+ /**
+ * Update map of hourly transmission net tariffs from internal cache.
+ */
+ public void updateTransmissionNetTariffs() {
+ transmissionNetTariffMap = priceListParser.toHourly(transmissionNetTariffRecords);
+ cleanup();
+ }
+
+ /**
+ * Get current spot price.
+ *
+ * @return spot price currently valid
+ */
+ public @Nullable BigDecimal getSpotPrice() {
+ return getSpotPrice(Instant.now(clock));
+ }
+
+ /**
+ * Get spot price valid at provided instant.
+ *
+ * @param time {@link Instant} for which to get the spot price
+ * @return spot price at given time or null if not available
+ */
+ public @Nullable BigDecimal getSpotPrice(Instant time) {
+ return spotPriceMap.get(getHourStart(time));
+ }
+
+ /**
+ * Get map of all cached spot prices.
+ *
+ * @return spot prices currently available, {@link #NUMBER_OF_HISTORIC_HOURS} back
+ */
+ public Map getSpotPrices() {
+ return new HashMap(spotPriceMap);
+ }
+
+ /**
+ * Get current net tariff.
+ *
+ * @return net tariff currently valid
+ */
+ public @Nullable BigDecimal getNetTariff() {
+ return getNetTariff(Instant.now(clock));
+ }
+
+ /**
+ * Get net tariff valid at provided instant.
+ *
+ * @param time {@link Instant} for which to get the net tariff
+ * @return net tariff at given time or null if not available
+ */
+ public @Nullable BigDecimal getNetTariff(Instant time) {
+ return netTariffMap.get(getHourStart(time));
+ }
+
+ /**
+ * Get map of all cached net tariffs.
+ *
+ * @return net tariffs currently available, {@link #NUMBER_OF_HISTORIC_HOURS} back
+ */
+ public Map getNetTariffs() {
+ return new HashMap(netTariffMap);
+ }
+
+ /**
+ * Get current system tariff.
+ *
+ * @return system tariff currently valid
+ */
+ public @Nullable BigDecimal getSystemTariff() {
+ return getSystemTariff(Instant.now(clock));
+ }
+
+ /**
+ * Get system tariff valid at provided instant.
+ *
+ * @param time {@link Instant} for which to get the system tariff
+ * @return system tariff at given time or null if not available
+ */
+ public @Nullable BigDecimal getSystemTariff(Instant time) {
+ return systemTariffMap.get(getHourStart(time));
+ }
+
+ /**
+ * Get map of all cached system tariffs.
+ *
+ * @return system tariffs currently available, {@link #NUMBER_OF_HISTORIC_HOURS} back
+ */
+ public Map getSystemTariffs() {
+ return new HashMap(systemTariffMap);
+ }
+
+ /**
+ * Get current electricity tax.
+ *
+ * @return electricity tax currently valid
+ */
+ public @Nullable BigDecimal getElectricityTax() {
+ return getElectricityTax(Instant.now(clock));
+ }
+
+ /**
+ * Get electricity tax valid at provided instant.
+ *
+ * @param time {@link Instant} for which to get the electricity tax
+ * @return electricity tax at given time or null if not available
+ */
+ public @Nullable BigDecimal getElectricityTax(Instant time) {
+ return electricityTaxMap.get(getHourStart(time));
+ }
+
+ /**
+ * Get map of all cached electricity taxes.
+ *
+ * @return electricity taxes currently available, {@link #NUMBER_OF_HISTORIC_HOURS} back
+ */
+ public Map getElectricityTaxes() {
+ return new HashMap(electricityTaxMap);
+ }
+
+ /**
+ * Get current transmission net tariff.
+ *
+ * @return transmission net tariff currently valid
+ */
+ public @Nullable BigDecimal getTransmissionNetTariff() {
+ return getTransmissionNetTariff(Instant.now(clock));
+ }
+
+ /**
+ * Get transmission net tariff valid at provided instant.
+ *
+ * @param time {@link Instant} for which to get the transmission net tariff
+ * @return transmission net tariff at given time or null if not available
+ */
+ public @Nullable BigDecimal getTransmissionNetTariff(Instant time) {
+ return transmissionNetTariffMap.get(getHourStart(time));
+ }
+
+ /**
+ * Get map of all cached transmission net tariffs.
+ *
+ * @return transmission net tariffs currently available, {@link #NUMBER_OF_HISTORIC_HOURS} back
+ */
+ public Map getTransmissionNetTariffs() {
+ return new HashMap(transmissionNetTariffMap);
+ }
+
+ /**
+ * Get number of future spot prices including current hour.
+ *
+ * @return number of future spot prices
+ */
+ public long getNumberOfFutureSpotPrices() {
+ Instant currentHourStart = getCurrentHourStart();
+
+ return spotPriceMap.entrySet().stream().filter(p -> !p.getKey().isBefore(currentHourStart)).count();
+ }
+
+ /**
+ * Check if historic spot prices ({@link #NUMBER_OF_HISTORIC_HOURS}) are cached.
+ *
+ * @return true if historic spot prices are cached
+ */
+ public boolean areHistoricSpotPricesCached() {
+ return arePricesCached(spotPriceMap, getCurrentHourStart().minus(1, ChronoUnit.HOURS));
+ }
+
+ /**
+ * Check if all current spot prices are cached taking into consideration that next day's spot prices
+ * should be available at 13:00 CET.
+ *
+ * @return true if spot prices are fully cached
+ */
+ public boolean areSpotPricesFullyCached() {
+ Instant end = ZonedDateTime.of(LocalDate.now(clock), LocalTime.of(23, 0), NORD_POOL_TIMEZONE).toInstant();
+ LocalTime now = LocalTime.now(clock);
+ if (now.isAfter(DAILY_REFRESH_TIME_CET)) {
+ end = end.plus(24, ChronoUnit.HOURS);
+ }
+
+ return arePricesCached(spotPriceMap, end);
+ }
+
+ private boolean arePricesCached(Map priceMap, Instant end) {
+ for (Instant hourStart = getFirstHourStart(); hourStart.compareTo(end) <= 0; hourStart = hourStart.plus(1,
+ ChronoUnit.HOURS)) {
+ if (priceMap.get(hourStart) == null) {
+ return false;
+ }
+ }
+
+ return true;
+ }
+
+ /**
+ * Check if we have "raw" net tariff records cached which are valid tomorrow.
+ *
+ * @return true if net tariff records for tomorrow are cached
+ */
+ public boolean areNetTariffsValidTomorrow() {
+ return isValidNextDay(netTariffRecords);
+ }
+
+ /**
+ * Check if we have "raw" system tariff records cached which are valid tomorrow.
+ *
+ * @return true if system tariff records for tomorrow are cached
+ */
+ public boolean areSystemTariffsValidTomorrow() {
+ return isValidNextDay(systemTariffRecords);
+ }
+
+ /**
+ * Check if we have "raw" electricity tax records cached which are valid tomorrow.
+ *
+ * @return true if electricity tax records for tomorrow are cached
+ */
+ public boolean areElectricityTaxesValidTomorrow() {
+ return isValidNextDay(electricityTaxRecords);
+ }
+
+ /**
+ * Check if we have "raw" transmission net tariff records cached which are valid tomorrow.
+ *
+ * @return true if transmission net tariff records for tomorrow are cached
+ */
+ public boolean areTransmissionNetTariffsValidTomorrow() {
+ return isValidNextDay(transmissionNetTariffRecords);
+ }
+
+ /**
+ * Remove historic prices.
+ */
+ public void cleanup() {
+ Instant firstHourStart = getFirstHourStart();
+
+ spotPriceMap.entrySet().removeIf(entry -> entry.getKey().isBefore(firstHourStart));
+ netTariffMap.entrySet().removeIf(entry -> entry.getKey().isBefore(firstHourStart));
+ systemTariffMap.entrySet().removeIf(entry -> entry.getKey().isBefore(firstHourStart));
+ electricityTaxMap.entrySet().removeIf(entry -> entry.getKey().isBefore(firstHourStart));
+ transmissionNetTariffMap.entrySet().removeIf(entry -> entry.getKey().isBefore(firstHourStart));
+ }
+
+ private boolean isValidNextDay(Collection records) {
+ LocalDateTime localHourStart = LocalDateTime.now(EnergiDataServiceBindingConstants.DATAHUB_TIMEZONE)
+ .truncatedTo(ChronoUnit.HOURS);
+ LocalDateTime localMidnight = localHourStart.plusDays(1).truncatedTo(ChronoUnit.DAYS);
+
+ return records.stream().anyMatch(r -> r.validTo().isAfter(localMidnight));
+ }
+
+ private Instant getCurrentHourStart() {
+ return getHourStart(Instant.now(clock));
+ }
+
+ private Instant getFirstHourStart() {
+ return getHourStart(Instant.now(clock).minus(NUMBER_OF_HISTORIC_HOURS, ChronoUnit.HOURS));
+ }
+
+ private Instant getHourStart(Instant instant) {
+ return instant.truncatedTo(ChronoUnit.HOURS);
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/EnergiDataServiceBindingConstants.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/EnergiDataServiceBindingConstants.java
new file mode 100644
index 000000000..aa828627a
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/EnergiDataServiceBindingConstants.java
@@ -0,0 +1,79 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal;
+
+import java.time.LocalDate;
+import java.time.LocalTime;
+import java.time.ZoneId;
+import java.util.Currency;
+import java.util.Set;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.core.thing.ChannelUID;
+import org.openhab.core.thing.ThingTypeUID;
+
+/**
+ * The {@link EnergiDataServiceBindingConstants} class defines common constants, which are
+ * used across the whole binding.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class EnergiDataServiceBindingConstants {
+
+ private static final String BINDING_ID = "energidataservice";
+
+ // List of all Thing Type UIDs
+ public static final ThingTypeUID THING_TYPE_SERVICE = new ThingTypeUID(BINDING_ID, "service");
+
+ // List of all Channel Group ids
+ public static final String CHANNEL_GROUP_ELECTRICITY = "electricity";
+
+ // List of all Channel ids
+ public static final String CHANNEL_SPOT_PRICE = CHANNEL_GROUP_ELECTRICITY + ChannelUID.CHANNEL_GROUP_SEPARATOR
+ + "spot-price";
+ public static final String CHANNEL_NET_TARIFF = CHANNEL_GROUP_ELECTRICITY + ChannelUID.CHANNEL_GROUP_SEPARATOR
+ + "net-tariff";
+ public static final String CHANNEL_SYSTEM_TARIFF = CHANNEL_GROUP_ELECTRICITY + ChannelUID.CHANNEL_GROUP_SEPARATOR
+ + "system-tariff";
+ public static final String CHANNEL_ELECTRICITY_TAX = CHANNEL_GROUP_ELECTRICITY + ChannelUID.CHANNEL_GROUP_SEPARATOR
+ + "electricity-tax";
+ public static final String CHANNEL_TRANSMISSION_NET_TARIFF = CHANNEL_GROUP_ELECTRICITY
+ + ChannelUID.CHANNEL_GROUP_SEPARATOR + "transmission-net-tariff";
+ public static final String CHANNEL_HOURLY_PRICES = CHANNEL_GROUP_ELECTRICITY + ChannelUID.CHANNEL_GROUP_SEPARATOR
+ + "hourly-prices";
+
+ public static final Set ELECTRICITY_CHANNELS = Set.of(CHANNEL_SPOT_PRICE, CHANNEL_NET_TARIFF,
+ CHANNEL_SYSTEM_TARIFF, CHANNEL_ELECTRICITY_TAX, CHANNEL_TRANSMISSION_NET_TARIFF, CHANNEL_HOURLY_PRICES);
+
+ // List of all properties
+ public static final String PROPERTY_REMAINING_CALLS = "remainingCalls";
+ public static final String PROPERTY_TOTAL_CALLS = "totalCalls";
+ public static final String PROPERTY_LAST_CALL = "lastCall";
+ public static final String PROPERTY_NEXT_CALL = "nextCall";
+
+ // List of supported currencies
+ public static final Currency CURRENCY_DKK = Currency.getInstance("DKK");
+ public static final Currency CURRENCY_EUR = Currency.getInstance("EUR");
+
+ public static final Set SUPPORTED_CURRENCIES = Set.of(CURRENCY_DKK, CURRENCY_EUR);
+
+ // Time-zone of Datahub
+ public static final ZoneId DATAHUB_TIMEZONE = ZoneId.of("CET");
+ public static final ZoneId NORD_POOL_TIMEZONE = ZoneId.of("CET");
+
+ // Other
+ public static final LocalTime DAILY_REFRESH_TIME_CET = LocalTime.of(13, 0);
+ public static final LocalDate ENERGINET_CUTOFF_DATE = LocalDate.of(2023, 1, 1);
+ public static final String PROPERTY_DATETIME_FORMAT = "yyyy-MM-dd HH:mm:ss";
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/PriceCalculator.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/PriceCalculator.java
new file mode 100644
index 000000000..67e988909
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/PriceCalculator.java
@@ -0,0 +1,238 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal;
+
+import java.math.BigDecimal;
+import java.math.RoundingMode;
+import java.time.Duration;
+import java.time.Instant;
+import java.time.temporal.ChronoUnit;
+import java.util.ArrayList;
+import java.util.Collection;
+import java.util.HashMap;
+import java.util.Iterator;
+import java.util.List;
+import java.util.Map;
+
+import javax.measure.quantity.Energy;
+import javax.measure.quantity.Power;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.energidataservice.internal.exception.MissingPriceException;
+import org.openhab.core.library.types.QuantityType;
+import org.openhab.core.library.unit.Units;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * Provides calculations based on price maps.
+ * This is the current stage of evolution.
+ * Ideally this binding would simply provide data in a well-defined format for
+ * openHAB core. Operations on this data could then be implemented in core.
+ * This way there would be a unified interface from rules, and the calculations
+ * could be reused between different data providers (bindings).
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class PriceCalculator {
+
+ private final Logger logger = LoggerFactory.getLogger(PriceCalculator.class);
+
+ private final Map priceMap;
+
+ public PriceCalculator(Map priceMap) {
+ this.priceMap = priceMap;
+ }
+
+ /**
+ * Calculate cheapest period from list of durations with specified amount of energy
+ * used per phase.
+ *
+ * @param earliestStart Earliest allowed start time.
+ * @param latestEnd Latest allowed end time.
+ * @param totalDuration Total duration to fit.
+ * @param durationPhases List of {@link Duration}'s representing different phases of using power.
+ * @param energyUsedPerPhase Amount of energy used per phase.
+ *
+ * @return Map containing resulting values
+ */
+ public Map calculateCheapestPeriod(Instant earliestStart, Instant latestEnd, Duration totalDuration,
+ Collection durationPhases, QuantityType energyUsedPerPhase) throws MissingPriceException {
+ QuantityType energyInWattHour = energyUsedPerPhase.toUnit(Units.WATT_HOUR);
+ if (energyInWattHour == null) {
+ throw new IllegalArgumentException(
+ "Invalid unit " + energyUsedPerPhase.getUnit() + ", expected energy unit");
+ }
+ // watts = (kWh × 1,000) ÷ hrs
+ int numerator = energyInWattHour.intValue() * 3600;
+ List> consumptionPhases = new ArrayList<>();
+ Duration remainingDuration = totalDuration;
+ for (Duration phase : durationPhases) {
+ consumptionPhases.add(QuantityType.valueOf(numerator / phase.getSeconds(), Units.WATT));
+ remainingDuration = remainingDuration.minus(phase);
+ }
+ if (remainingDuration.isNegative()) {
+ throw new IllegalArgumentException("totalDuration must be equal to or greater than sum of phases");
+ }
+ if (!remainingDuration.isZero()) {
+ List durationsWithTermination = new ArrayList<>(durationPhases);
+ durationsWithTermination.add(remainingDuration);
+ consumptionPhases.add(QuantityType.valueOf(0, Units.WATT));
+ return calculateCheapestPeriod(earliestStart, latestEnd, durationsWithTermination, consumptionPhases);
+ }
+ return calculateCheapestPeriod(earliestStart, latestEnd, durationPhases, consumptionPhases);
+ }
+
+ /**
+ * Calculate cheapest period from duration with linear power usage.
+ *
+ * @param earliestStart Earliest allowed start time.
+ * @param latestEnd Latest allowed end time.
+ * @param duration Duration to fit.
+ * @param power Power consumption for the duration of time.
+ *
+ * @return Map containing resulting values
+ */
+ public Map calculateCheapestPeriod(Instant earliestStart, Instant latestEnd, Duration duration,
+ QuantityType power) throws MissingPriceException {
+ return calculateCheapestPeriod(earliestStart, latestEnd, List.of(duration), List.of(power));
+ }
+
+ /**
+ * Calculate cheapest period from list of durations with corresponding list of consumption
+ * per duration.
+ *
+ * @param earliestStart Earliest allowed start time.
+ * @param latestEnd Latest allowed end time.
+ * @param durationPhases List of {@link Duration}'s representing different phases of using power.
+ * @param consumptionPhases Corresponding List of power consumption for the duration of time.
+ *
+ * @return Map containing resulting values
+ */
+ public Map calculateCheapestPeriod(Instant earliestStart, Instant latestEnd,
+ Collection durationPhases, Collection> consumptionPhases)
+ throws MissingPriceException {
+ if (durationPhases.size() != consumptionPhases.size()) {
+ throw new IllegalArgumentException("Number of phases do not match");
+ }
+ Map result = new HashMap<>();
+ Duration totalDuration = durationPhases.stream().reduce(Duration.ZERO, Duration::plus);
+ Instant calculationStart = earliestStart;
+ Instant calculationEnd = earliestStart.plus(totalDuration);
+ BigDecimal lowestPrice = BigDecimal.valueOf(Double.MAX_VALUE);
+ BigDecimal highestPrice = BigDecimal.ZERO;
+ Instant cheapestStart = Instant.MIN;
+ Instant mostExpensiveStart = Instant.MIN;
+
+ while (calculationEnd.compareTo(latestEnd) <= 0) {
+ BigDecimal currentPrice = BigDecimal.ZERO;
+ Duration minDurationUntilNextHour = Duration.ofHours(1);
+ Instant atomStart = calculationStart;
+
+ Iterator durationIterator = durationPhases.iterator();
+ Iterator> consumptionIterator = consumptionPhases.iterator();
+ while (durationIterator.hasNext()) {
+ Duration atomDuration = durationIterator.next();
+ QuantityType atomConsumption = consumptionIterator.next();
+
+ Instant atomEnd = atomStart.plus(atomDuration);
+ Instant hourStart = atomStart.truncatedTo(ChronoUnit.HOURS);
+ Instant hourEnd = hourStart.plus(1, ChronoUnit.HOURS);
+
+ // Get next intersection with hourly rate change.
+ Duration durationUntilNextHour = Duration.between(atomStart, hourEnd);
+ if (durationUntilNextHour.compareTo(minDurationUntilNextHour) < 0) {
+ minDurationUntilNextHour = durationUntilNextHour;
+ }
+
+ BigDecimal atomPrice = calculatePrice(atomStart, atomEnd, atomConsumption);
+ currentPrice = currentPrice.add(atomPrice);
+ atomStart = atomEnd;
+ }
+
+ if (currentPrice.compareTo(lowestPrice) < 0) {
+ lowestPrice = currentPrice;
+ cheapestStart = calculationStart;
+ }
+ if (currentPrice.compareTo(highestPrice) > 0) {
+ highestPrice = currentPrice;
+ mostExpensiveStart = calculationStart;
+ }
+
+ // Now fast forward to next hourly rate intersection.
+ calculationStart = calculationStart.plus(minDurationUntilNextHour);
+ calculationEnd = calculationStart.plus(totalDuration);
+ }
+
+ if (!cheapestStart.equals(Instant.MIN)) {
+ result.put("CheapestStart", cheapestStart);
+ result.put("LowestPrice", lowestPrice);
+ result.put("MostExpensiveStart", mostExpensiveStart);
+ result.put("HighestPrice", highestPrice);
+ }
+
+ return result;
+ }
+
+ /**
+ * Calculate total price from 'start' to 'end' given linear power consumption.
+ *
+ * @param start Start time
+ * @param end End time
+ * @param power The current power consumption.
+ */
+ public BigDecimal calculatePrice(Instant start, Instant end, QuantityType power)
+ throws MissingPriceException {
+ QuantityType quantityInWatt = power.toUnit(Units.WATT);
+ if (quantityInWatt == null) {
+ throw new IllegalArgumentException("Invalid unit " + power.getUnit() + ", expected power unit");
+ }
+ BigDecimal watt = new BigDecimal(quantityInWatt.intValue());
+ if (watt.equals(BigDecimal.ZERO)) {
+ return BigDecimal.ZERO;
+ }
+
+ Instant current = start;
+ BigDecimal result = BigDecimal.ZERO;
+ while (current.isBefore(end)) {
+ Instant hourStart = current.truncatedTo(ChronoUnit.HOURS);
+ Instant hourEnd = hourStart.plus(1, ChronoUnit.HOURS);
+
+ BigDecimal currentPrice = priceMap.get(hourStart);
+ if (currentPrice == null) {
+ throw new MissingPriceException("Price missing at " + hourStart.toString());
+ }
+
+ Instant currentStart = hourStart;
+ if (start.isAfter(hourStart)) {
+ currentStart = start;
+ }
+ Instant currentEnd = hourEnd;
+ if (end.isBefore(hourEnd)) {
+ currentEnd = end;
+ }
+
+ // E(kWh) = P(W) × t(hr) / 1000
+ Duration duration = Duration.between(currentStart, currentEnd);
+ BigDecimal contribution = currentPrice.multiply(watt).multiply(
+ new BigDecimal(duration.getSeconds()).divide(new BigDecimal(3600000), 9, RoundingMode.HALF_UP));
+ result = result.add(contribution);
+ logger.trace("Period {}-{}: {} @ {}", currentStart, currentEnd, contribution, currentPrice);
+
+ current = hourEnd;
+ }
+
+ return result;
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/PriceListParser.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/PriceListParser.java
new file mode 100644
index 000000000..5f4bedc1a
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/PriceListParser.java
@@ -0,0 +1,109 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal;
+
+import java.math.BigDecimal;
+import java.time.Clock;
+import java.time.Instant;
+import java.time.LocalDateTime;
+import java.time.LocalTime;
+import java.time.temporal.ChronoUnit;
+import java.util.Collection;
+import java.util.Map;
+import java.util.Map.Entry;
+import java.util.concurrent.ConcurrentHashMap;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.energidataservice.internal.api.dto.DatahubPricelistRecord;
+
+/**
+ * Parses results from {@link DatahubPricelistRecords} into map of hourly tariffs.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class PriceListParser {
+
+ private final Clock clock;
+
+ public PriceListParser() {
+ this(Clock.system(EnergiDataServiceBindingConstants.DATAHUB_TIMEZONE));
+ }
+
+ public PriceListParser(Clock clock) {
+ this.clock = clock;
+ }
+
+ public Map toHourly(Collection records) {
+ Map totalMap = new ConcurrentHashMap<>(CacheManager.TARIFF_MAX_CACHE_SIZE);
+ records.stream().map(record -> record.chargeTypeCode()).distinct().forEach(chargeTypeCode -> {
+ Map currentMap = toHourly(records, chargeTypeCode);
+ for (Entry current : currentMap.entrySet()) {
+ BigDecimal total = totalMap.get(current.getKey());
+ if (total == null) {
+ total = BigDecimal.ZERO;
+ }
+ totalMap.put(current.getKey(), total.add(current.getValue()));
+ }
+ });
+
+ return totalMap;
+ }
+
+ public Map toHourly(Collection records, String chargeTypeCode) {
+ Map tariffMap = new ConcurrentHashMap<>(CacheManager.TARIFF_MAX_CACHE_SIZE);
+
+ Instant firstHourStart = Instant.now(clock).minus(CacheManager.NUMBER_OF_HISTORIC_HOURS, ChronoUnit.HOURS)
+ .truncatedTo(ChronoUnit.HOURS);
+ Instant lastHourStart = Instant.now(clock).truncatedTo(ChronoUnit.HOURS).plus(2, ChronoUnit.DAYS)
+ .truncatedTo(ChronoUnit.DAYS);
+
+ LocalDateTime previousValidFrom = LocalDateTime.MAX;
+ LocalDateTime previousValidTo = LocalDateTime.MIN;
+ Map tariffs = Map.of();
+ for (Instant hourStart = firstHourStart; hourStart
+ .isBefore(lastHourStart); hourStart = hourStart.plus(1, ChronoUnit.HOURS)) {
+ LocalDateTime localDateTime = hourStart.atZone(EnergiDataServiceBindingConstants.DATAHUB_TIMEZONE)
+ .toLocalDateTime();
+ if (localDateTime.compareTo(previousValidFrom) < 0 || localDateTime.compareTo(previousValidTo) >= 0) {
+ DatahubPricelistRecord priceList = getTariffs(records, localDateTime, chargeTypeCode);
+ if (priceList != null) {
+ tariffs = priceList.getTariffMap();
+ previousValidFrom = priceList.validFrom();
+ previousValidTo = priceList.validTo();
+ } else {
+ tariffs = Map.of();
+ }
+ }
+
+ LocalTime localTime = LocalTime
+ .of(hourStart.atZone(EnergiDataServiceBindingConstants.DATAHUB_TIMEZONE).getHour(), 0);
+ BigDecimal tariff = tariffs.get(localTime);
+ if (tariff != null) {
+ tariffMap.put(hourStart, tariff);
+ }
+ }
+
+ return tariffMap;
+ }
+
+ private @Nullable DatahubPricelistRecord getTariffs(Collection records,
+ LocalDateTime localDateTime, String chargeTypeCode) {
+ return records.stream()
+ .filter(record -> localDateTime.compareTo(record.validFrom()) >= 0
+ && localDateTime.compareTo(record.validTo()) < 0
+ && record.chargeTypeCode().equals(chargeTypeCode))
+ .findFirst().orElse(null);
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/action/EnergiDataServiceActions.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/action/EnergiDataServiceActions.java
new file mode 100644
index 000000000..a39897deb
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/action/EnergiDataServiceActions.java
@@ -0,0 +1,381 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.action;
+
+import static org.openhab.binding.energidataservice.internal.EnergiDataServiceBindingConstants.*;
+
+import java.math.BigDecimal;
+import java.time.Duration;
+import java.time.Instant;
+import java.util.Arrays;
+import java.util.HashMap;
+import java.util.HashSet;
+import java.util.List;
+import java.util.Map;
+import java.util.Map.Entry;
+import java.util.Set;
+import java.util.function.Function;
+import java.util.stream.Collectors;
+import java.util.stream.Stream;
+
+import javax.measure.quantity.Energy;
+import javax.measure.quantity.Power;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.energidataservice.internal.PriceCalculator;
+import org.openhab.binding.energidataservice.internal.exception.MissingPriceException;
+import org.openhab.binding.energidataservice.internal.handler.EnergiDataServiceHandler;
+import org.openhab.core.automation.annotation.ActionInput;
+import org.openhab.core.automation.annotation.ActionOutput;
+import org.openhab.core.automation.annotation.RuleAction;
+import org.openhab.core.library.types.QuantityType;
+import org.openhab.core.library.unit.Units;
+import org.openhab.core.thing.binding.ThingActions;
+import org.openhab.core.thing.binding.ThingActionsScope;
+import org.openhab.core.thing.binding.ThingHandler;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * {@link EnergiDataServiceActions} provides actions for getting energy data into a rule context.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@ThingActionsScope(name = "energidataservice")
+@NonNullByDefault
+public class EnergiDataServiceActions implements ThingActions {
+
+ private final Logger logger = LoggerFactory.getLogger(EnergiDataServiceActions.class);
+
+ private @Nullable EnergiDataServiceHandler handler;
+
+ private enum PriceElement {
+ SPOT_PRICE("spotprice"),
+ NET_TARIFF("nettariff"),
+ SYSTEM_TARIFF("systemtariff"),
+ ELECTRICITY_TAX("electricitytax"),
+ TRANSMISSION_NET_TARIFF("transmissionnettariff");
+
+ private static final Map NAME_MAP = Stream.of(values())
+ .collect(Collectors.toMap(PriceElement::toString, Function.identity()));
+
+ private String name;
+
+ private PriceElement(String name) {
+ this.name = name;
+ }
+
+ @Override
+ public String toString() {
+ return name;
+ }
+
+ public static PriceElement fromString(final String name) {
+ PriceElement myEnum = NAME_MAP.get(name.toLowerCase());
+ if (null == myEnum) {
+ throw new IllegalArgumentException(String.format("'%s' has no corresponding value. Accepted values: %s",
+ name, Arrays.asList(values())));
+ }
+ return myEnum;
+ }
+ }
+
+ @RuleAction(label = "@text/action.get-prices.label", description = "@text/action.get-prices.description")
+ public @ActionOutput(name = "prices", type = "java.util.Map") Map getPrices() {
+ return getPrices(Arrays.stream(PriceElement.values()).collect(Collectors.toSet()));
+ }
+
+ @RuleAction(label = "@text/action.get-prices.label", description = "@text/action.get-prices.description")
+ public @ActionOutput(name = "prices", type = "java.util.Map") Map getPrices(
+ @ActionInput(name = "priceElements", label = "@text/action.get-prices.priceElements.label", description = "@text/action.get-prices.priceElements.description") @Nullable String priceElements) {
+ if (priceElements == null) {
+ logger.warn("Argument 'priceElements' is null");
+ return Map.of();
+ }
+
+ Set priceElementsSet;
+ try {
+ priceElementsSet = new HashSet(
+ Arrays.stream(priceElements.split(",")).map(PriceElement::fromString).toList());
+ } catch (IllegalArgumentException e) {
+ logger.warn("{}", e.getMessage());
+ return Map.of();
+ }
+
+ return getPrices(priceElementsSet);
+ }
+
+ @RuleAction(label = "@text/action.calculate-price.label", description = "@text/action.calculate-price.description")
+ public @ActionOutput(name = "price", type = "java.math.BigDecimal") BigDecimal calculatePrice(
+ @ActionInput(name = "start", type = "java.time.Instant") Instant start,
+ @ActionInput(name = "end", type = "java.time.Instant") Instant end,
+ @ActionInput(name = "power", type = "QuantityType") QuantityType power) {
+ PriceCalculator priceCalculator = new PriceCalculator(getPrices());
+
+ try {
+ return priceCalculator.calculatePrice(start, end, power);
+ } catch (MissingPriceException e) {
+ logger.warn("{}", e.getMessage());
+ return BigDecimal.ZERO;
+ }
+ }
+
+ @RuleAction(label = "@text/action.calculate-cheapest-period.label", description = "@text/action.calculate-cheapest-period.description")
+ public @ActionOutput(name = "result", type = "java.util.Map") Map calculateCheapestPeriod(
+ @ActionInput(name = "earliestStart", type = "java.time.Instant") Instant earliestStart,
+ @ActionInput(name = "latestEnd", type = "java.time.Instant") Instant latestEnd,
+ @ActionInput(name = "duration", type = "java.time.Duration") Duration duration) {
+ PriceCalculator priceCalculator = new PriceCalculator(getPrices());
+
+ try {
+ Map intermediateResult = priceCalculator.calculateCheapestPeriod(earliestStart, latestEnd,
+ duration, QuantityType.valueOf(1000, Units.WATT));
+
+ // Create new result with stripped price information.
+ Map result = new HashMap<>();
+ Object value = intermediateResult.get("CheapestStart");
+ if (value != null) {
+ result.put("CheapestStart", value);
+ }
+ value = intermediateResult.get("MostExpensiveStart");
+ if (value != null) {
+ result.put("MostExpensiveStart", value);
+ }
+ return result;
+ } catch (MissingPriceException | IllegalArgumentException e) {
+ logger.warn("{}", e.getMessage());
+ return Map.of();
+ }
+ }
+
+ @RuleAction(label = "@text/action.calculate-cheapest-period.label", description = "@text/action.calculate-cheapest-period.description")
+ public @ActionOutput(name = "result", type = "java.util.Map") Map calculateCheapestPeriod(
+ @ActionInput(name = "earliestStart", type = "java.time.Instant") Instant earliestStart,
+ @ActionInput(name = "latestEnd", type = "java.time.Instant") Instant latestEnd,
+ @ActionInput(name = "duration", type = "java.time.Duration") Duration duration,
+ @ActionInput(name = "power", type = "QuantityType") QuantityType power) {
+ PriceCalculator priceCalculator = new PriceCalculator(getPrices());
+
+ try {
+ return priceCalculator.calculateCheapestPeriod(earliestStart, latestEnd, duration, power);
+ } catch (MissingPriceException | IllegalArgumentException e) {
+ logger.warn("{}", e.getMessage());
+ return Map.of();
+ }
+ }
+
+ @RuleAction(label = "@text/action.calculate-cheapest-period.label", description = "@text/action.calculate-cheapest-period.description")
+ public @ActionOutput(name = "result", type = "java.util.Map") Map calculateCheapestPeriod(
+ @ActionInput(name = "earliestStart", type = "java.time.Instant") Instant earliestStart,
+ @ActionInput(name = "latestEnd", type = "java.time.Instant") Instant latestEnd,
+ @ActionInput(name = "totalDuration", type = "java.time.Duration") Duration totalDuration,
+ @ActionInput(name = "durationPhases", type = "java.util.List") List durationPhases,
+ @ActionInput(name = "energyUsedPerPhase", type = "QuantityType") QuantityType energyUsedPerPhase) {
+ PriceCalculator priceCalculator = new PriceCalculator(getPrices());
+
+ try {
+ return priceCalculator.calculateCheapestPeriod(earliestStart, latestEnd, totalDuration, durationPhases,
+ energyUsedPerPhase);
+ } catch (MissingPriceException | IllegalArgumentException e) {
+ logger.warn("{}", e.getMessage());
+ return Map.of();
+ }
+ }
+
+ @RuleAction(label = "@text/action.calculate-cheapest-period.label", description = "@text/action.calculate-cheapest-period.description")
+ public @ActionOutput(name = "result", type = "java.util.Map") Map calculateCheapestPeriod(
+ @ActionInput(name = "earliestStart", type = "java.time.Instant") Instant earliestStart,
+ @ActionInput(name = "latestEnd", type = "java.time.Instant") Instant latestEnd,
+ @ActionInput(name = "durationPhases", type = "java.util.List") List durationPhases,
+ @ActionInput(name = "powerPhases", type = "java.util.List>") List> powerPhases) {
+ if (durationPhases.size() != powerPhases.size()) {
+ logger.warn("Number of duration phases ({}) is different from number of consumption phases ({})",
+ durationPhases.size(), powerPhases.size());
+ return Map.of();
+ }
+ PriceCalculator priceCalculator = new PriceCalculator(getPrices());
+
+ try {
+ return priceCalculator.calculateCheapestPeriod(earliestStart, latestEnd, durationPhases, powerPhases);
+ } catch (MissingPriceException | IllegalArgumentException e) {
+ logger.warn("{}", e.getMessage());
+ return Map.of();
+ }
+ }
+
+ private Map getPrices(Set priceElements) {
+ EnergiDataServiceHandler handler = this.handler;
+ if (handler == null) {
+ logger.warn("EnergiDataServiceActions ThingHandler is null.");
+ return Map.of();
+ }
+
+ Map prices;
+ boolean spotPricesRequired;
+ if (priceElements.contains(PriceElement.SPOT_PRICE)) {
+ if (priceElements.size() > 1 && !handler.getCurrency().equals(CURRENCY_DKK)) {
+ logger.warn("Cannot calculate sum when spot price currency is {}", handler.getCurrency());
+ return Map.of();
+ }
+ prices = handler.getSpotPrices();
+ spotPricesRequired = true;
+ } else {
+ spotPricesRequired = false;
+ prices = new HashMap<>();
+ }
+
+ if (priceElements.contains(PriceElement.NET_TARIFF)) {
+ Map netTariffMap = handler.getNetTariffs();
+ mergeMaps(prices, netTariffMap, !spotPricesRequired);
+ }
+
+ if (priceElements.contains(PriceElement.SYSTEM_TARIFF)) {
+ Map systemTariffMap = handler.getSystemTariffs();
+ mergeMaps(prices, systemTariffMap, !spotPricesRequired);
+ }
+
+ if (priceElements.contains(PriceElement.ELECTRICITY_TAX)) {
+ Map electricityTaxMap = handler.getElectricityTaxes();
+ mergeMaps(prices, electricityTaxMap, !spotPricesRequired);
+ }
+
+ if (priceElements.contains(PriceElement.TRANSMISSION_NET_TARIFF)) {
+ Map transmissionNetTariffMap = handler.getTransmissionNetTariffs();
+ mergeMaps(prices, transmissionNetTariffMap, !spotPricesRequired);
+ }
+
+ return prices;
+ }
+
+ private void mergeMaps(Map destinationMap, Map sourceMap,
+ boolean createNew) {
+ for (Entry source : sourceMap.entrySet()) {
+ Instant key = source.getKey();
+ BigDecimal sourceValue = source.getValue();
+ BigDecimal destinationValue = destinationMap.get(key);
+ if (destinationValue != null) {
+ destinationMap.put(key, sourceValue.add(destinationValue));
+ } else if (createNew) {
+ destinationMap.put(key, sourceValue);
+ }
+ }
+ }
+
+ /**
+ * Static get prices method for DSL rule compatibility.
+ *
+ * @param actions
+ * @param priceElements Comma-separated list of price elements to include in prices.
+ * @return Map of prices
+ */
+ public static Map getPrices(@Nullable ThingActions actions, @Nullable String priceElements) {
+ if (actions instanceof EnergiDataServiceActions) {
+ if (priceElements != null && !priceElements.isBlank()) {
+ return ((EnergiDataServiceActions) actions).getPrices(priceElements);
+ } else {
+ return ((EnergiDataServiceActions) actions).getPrices();
+ }
+ } else {
+ throw new IllegalArgumentException("Instance is not an EnergiDataServiceActions class.");
+ }
+ }
+
+ /**
+ * Static get prices method for DSL rule compatibility.
+ *
+ * @param actions
+ * @param start Start time
+ * @param end End time
+ * @param power Constant power consumption
+ * @return Map of prices
+ */
+ public static BigDecimal calculatePrice(@Nullable ThingActions actions, @Nullable Instant start,
+ @Nullable Instant end, @Nullable QuantityType power) {
+ if (start == null || end == null || power == null) {
+ return BigDecimal.ZERO;
+ }
+ if (actions instanceof EnergiDataServiceActions) {
+ return ((EnergiDataServiceActions) actions).calculatePrice(start, end, power);
+ } else {
+ throw new IllegalArgumentException("Instance is not an EnergiDataServiceActions class.");
+ }
+ }
+
+ public static Map calculateCheapestPeriod(@Nullable ThingActions actions,
+ @Nullable Instant earliestStart, @Nullable Instant latestEnd, @Nullable Duration duration) {
+ if (actions instanceof EnergiDataServiceActions) {
+ if (earliestStart == null || latestEnd == null || duration == null) {
+ return Map.of();
+ }
+ return ((EnergiDataServiceActions) actions).calculateCheapestPeriod(earliestStart, latestEnd, duration);
+ } else {
+ throw new IllegalArgumentException("Instance is not an EnergiDataServiceActions class.");
+ }
+ }
+
+ public static Map calculateCheapestPeriod(@Nullable ThingActions actions,
+ @Nullable Instant earliestStart, @Nullable Instant latestEnd, @Nullable Duration duration,
+ @Nullable QuantityType power) {
+ if (actions instanceof EnergiDataServiceActions) {
+ if (earliestStart == null || latestEnd == null || duration == null || power == null) {
+ return Map.of();
+ }
+ return ((EnergiDataServiceActions) actions).calculateCheapestPeriod(earliestStart, latestEnd, duration,
+ power);
+ } else {
+ throw new IllegalArgumentException("Instance is not an EnergiDataServiceActions class.");
+ }
+ }
+
+ public static Map calculateCheapestPeriod(@Nullable ThingActions actions,
+ @Nullable Instant earliestStart, @Nullable Instant latestEnd, @Nullable Duration totalDuration,
+ @Nullable List durationPhases, @Nullable QuantityType energyUsedPerPhase) {
+ if (actions instanceof EnergiDataServiceActions) {
+ if (earliestStart == null || latestEnd == null || totalDuration == null || durationPhases == null
+ || energyUsedPerPhase == null) {
+ return Map.of();
+ }
+ return ((EnergiDataServiceActions) actions).calculateCheapestPeriod(earliestStart, latestEnd, totalDuration,
+ durationPhases, energyUsedPerPhase);
+ } else {
+ throw new IllegalArgumentException("Instance is not an EnergiDataServiceActions class.");
+ }
+ }
+
+ public static Map calculateCheapestPeriod(@Nullable ThingActions actions,
+ @Nullable Instant earliestStart, @Nullable Instant latestEnd, @Nullable List durationPhases,
+ @Nullable List> powerPhases) {
+ if (actions instanceof EnergiDataServiceActions) {
+ if (earliestStart == null || latestEnd == null || durationPhases == null || powerPhases == null) {
+ return Map.of();
+ }
+ return ((EnergiDataServiceActions) actions).calculateCheapestPeriod(earliestStart, latestEnd,
+ durationPhases, powerPhases);
+ } else {
+ throw new IllegalArgumentException("Instance is not an EnergiDataServiceActions class.");
+ }
+ }
+
+ @Override
+ public void setThingHandler(@Nullable ThingHandler handler) {
+ if (handler instanceof EnergiDataServiceHandler) {
+ this.handler = (EnergiDataServiceHandler) handler;
+ }
+ }
+
+ @Override
+ public @Nullable ThingHandler getThingHandler() {
+ return handler;
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/ChargeType.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/ChargeType.java
new file mode 100644
index 000000000..19d242b74
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/ChargeType.java
@@ -0,0 +1,39 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.api;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Charge type for DatahubPricelist dataset.
+ * See {@link https://www.energidataservice.dk/tso-electricity/DatahubPricelist#metadata-info}}
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public enum ChargeType {
+ Subscription("D01"),
+ Fee("D02"),
+ Tariff("D03");
+
+ private final String code;
+
+ ChargeType(String code) {
+ this.code = code;
+ }
+
+ @Override
+ public String toString() {
+ return code;
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/ChargeTypeCode.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/ChargeTypeCode.java
new file mode 100644
index 000000000..b9228a172
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/ChargeTypeCode.java
@@ -0,0 +1,47 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.api;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Charge type code for DatahubPricelist dataset.
+ * See {@link https://www.energidataservice.dk/tso-electricity/DatahubPricelist#metadata-info}}
+ * These codes are defined by the individual grid companies.
+ * For example, N1 uses "CD" for "Nettarif C" and "CD R" for "Rabat på nettarif N1 A/S".
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class ChargeTypeCode {
+
+ private static final int MAX_LENGTH = 20;
+
+ private final String code;
+
+ public ChargeTypeCode(String code) {
+ if (code.length() > MAX_LENGTH) {
+ throw new IllegalArgumentException("Maximum length exceeded: " + code);
+ }
+ this.code = code;
+ }
+
+ @Override
+ public String toString() {
+ return code;
+ }
+
+ public static ChargeTypeCode of(String code) {
+ return new ChargeTypeCode(code);
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/DatahubTariffFilter.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/DatahubTariffFilter.java
new file mode 100644
index 000000000..ecbec4a8b
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/DatahubTariffFilter.java
@@ -0,0 +1,58 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.api;
+
+import java.util.Collection;
+import java.util.Set;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Filter for the {@link DatahubPricelist} dataset.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class DatahubTariffFilter {
+
+ private final Set chargeTypeCodes;
+ private final Set notes;
+ private final DateQueryParameter dateQueryParameter;
+
+ public DatahubTariffFilter(DatahubTariffFilter filter, DateQueryParameter dateQueryParameter) {
+ this(filter.chargeTypeCodes, filter.notes, dateQueryParameter);
+ }
+
+ public DatahubTariffFilter(Set chargeTypeCodes, Set notes) {
+ this(chargeTypeCodes, notes, DateQueryParameter.EMPTY);
+ }
+
+ public DatahubTariffFilter(Set chargeTypeCodes, Set notes,
+ DateQueryParameter dateQueryParameter) {
+ this.chargeTypeCodes = chargeTypeCodes;
+ this.notes = notes;
+ this.dateQueryParameter = dateQueryParameter;
+ }
+
+ public Collection getChargeTypeCodesAsStrings() {
+ return chargeTypeCodes.stream().map(c -> c.toString()).toList();
+ }
+
+ public Collection getNotes() {
+ return notes;
+ }
+
+ public DateQueryParameter getDateQueryParameter() {
+ return dateQueryParameter;
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/DatahubTariffFilterFactory.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/DatahubTariffFilterFactory.java
new file mode 100644
index 000000000..2809c1592
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/DatahubTariffFilterFactory.java
@@ -0,0 +1,172 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.api;
+
+import static org.openhab.binding.energidataservice.internal.EnergiDataServiceBindingConstants.*;
+
+import java.time.Duration;
+import java.time.LocalDate;
+import java.util.Set;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Factory for creating a {@link DatahubTariffFilter} for a specific Grid Company GLN.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class DatahubTariffFilterFactory {
+
+ private static final String GLN_CERIUS = "5790000705184";
+ private static final String GLN_DINEL = "5790000610099";
+ private static final String GLN_ELEKTRUS = "5790000836239";
+ private static final String GLN_ELINORD = "5790001095277";
+ private static final String GLN_ELNET_MIDT = "5790001100520";
+ private static final String GLN_ELNET_KONGERSLEV = "5790002502699";
+ private static final String GLN_FLOW_ELNET = "5790000392551";
+ private static final String GLN_HAMMEL_ELFORSYNING_NET = "5790001090166";
+ private static final String GLN_HURUP_ELVAERK_NET = "5790000610839";
+ private static final String GLN_IKAST_E1_NET = "5790000682102";
+ private static final String GLN_KONSTANT = "5790000704842";
+ private static final String GLN_L_NET = "5790001090111";
+ private static final String GLN_MIDTFYNS_ELFORSYNING = "5790001089023";
+ private static final String GLN_N1 = "5790001089030";
+ private static final String GLN_NETSELSKABET_ELVAERK = "5790000681075";
+ private static final String GLN_NKE_ELNET = "5790001088231";
+ private static final String GLN_NORD_ENERGI_NET = "5790000610877";
+ private static final String GLN_NORDVESTJYSK_ELFORSYNING_NOE_NET = "5790000395620";
+ private static final String GLN_RADIUS = "5790000705689";
+ private static final String GLN_RAH_NET = "5790000681327";
+ private static final String GLN_RAVDEX = "5790000836727";
+ private static final String GLN_SUNDS_NET = "5790001095444";
+ private static final String GLN_TARM_ELVAERK_NET = "5790000706419";
+ private static final String GLN_TREFOR_EL_NET = "5790000392261";
+ private static final String GLN_TREFOR_EL_NET_OEST = "5790000706686";
+ private static final String GLN_VEKSEL = "5790001088217";
+ private static final String GLN_VORES_ELNET = "5790000610976";
+ private static final String GLN_ZEANET = "5790001089375";
+
+ private static final String NOTE_NET_TARIFF = "Nettarif";
+ private static final String NOTE_NET_TARIFF_C = NOTE_NET_TARIFF + " C";
+ private static final String NOTE_NET_TARIFF_C_HOUR = NOTE_NET_TARIFF_C + " time";
+ private static final String NOTE_NET_TARIFF_C_FLEX = NOTE_NET_TARIFF_C + " Flex";
+ private static final String NOTE_NET_TARIFF_C_FLEX_HOUR = NOTE_NET_TARIFF_C_FLEX + " - time";
+ private static final String NOTE_SYSTEM_TARIFF = "Systemtarif";
+ private static final String NOTE_ELECTRICITY_TAX = "Elafgift";
+ private static final String NOTE_TRANSMISSION_NET_TARIFF = "Transmissions nettarif";
+
+ public static final LocalDate N1_CUTOFF_DATE = LocalDate.of(2023, 1, 1);
+ public static final LocalDate RADIUS_CUTOFF_DATE = LocalDate.of(2023, 1, 1);
+ public static final LocalDate KONSTANT_CUTOFF_DATE = LocalDate.of(2023, 2, 1);
+
+ public static DatahubTariffFilter getNetTariffByGLN(String globalLocationNumber) {
+ switch (globalLocationNumber) {
+ case GLN_CERIUS:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("30TR_C_ET")), Set.of(NOTE_NET_TARIFF_C_HOUR));
+ case GLN_DINEL:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("TCL>100_02")), Set.of(NOTE_NET_TARIFF_C_HOUR));
+ case GLN_ELEKTRUS:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("6000091")), Set.of(NOTE_NET_TARIFF_C_HOUR),
+ DateQueryParameter.of(DateQueryParameterType.START_OF_DAY, Duration.ofDays(-1)));
+ case GLN_ELINORD:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("43300")),
+ Set.of("Transportbetaling, eget net C"));
+ case GLN_ELNET_MIDT:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("T3002")), Set.of(NOTE_NET_TARIFF_C),
+ DateQueryParameter.of(DateQueryParameterType.START_OF_DAY, Duration.ofDays(-1)));
+ case GLN_ELNET_KONGERSLEV:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("K_22100")), Set.of(NOTE_NET_TARIFF_C));
+ case GLN_FLOW_ELNET:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("FE2 NT-01")), Set.of(NOTE_NET_TARIFF_C_HOUR),
+ DateQueryParameter.of(DateQueryParameterType.START_OF_DAY, Duration.ofDays(-1)));
+ case GLN_HAMMEL_ELFORSYNING_NET:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("50001")), Set.of("Overliggende net"),
+ DateQueryParameter.of(DateQueryParameterType.START_OF_DAY, Duration.ofDays(-1)));
+ case GLN_HURUP_ELVAERK_NET:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("HEV-NT-01")), Set.of(NOTE_NET_TARIFF));
+ case GLN_IKAST_E1_NET:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("IEV-NT-01"), ChargeTypeCode.of("IEV-NT-11")),
+ Set.of(NOTE_NET_TARIFF_C_HOUR, "Transport - Overordnet net"));
+ case GLN_KONSTANT:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("151-NT01T"), ChargeTypeCode.of("151-NRA04T")),
+ Set.of(), DateQueryParameter.of(KONSTANT_CUTOFF_DATE));
+ case GLN_L_NET:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("4010")), Set.of(NOTE_NET_TARIFF_C_HOUR));
+ case GLN_MIDTFYNS_ELFORSYNING:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("TNT15000")), Set.of(NOTE_NET_TARIFF_C_FLEX),
+ DateQueryParameter.of(DateQueryParameterType.START_OF_DAY, Duration.ofDays(-1)));
+ case GLN_N1:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("CD"), ChargeTypeCode.of("CD R")), Set.of(),
+ DateQueryParameter.of(N1_CUTOFF_DATE));
+ case GLN_NETSELSKABET_ELVAERK:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("0NCFF")), Set.of(NOTE_NET_TARIFF_C + " Flex"),
+ DateQueryParameter.of(DateQueryParameterType.START_OF_DAY, Duration.ofDays(-1)));
+ case GLN_NKE_ELNET:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("94TR_C_ET")), Set.of(NOTE_NET_TARIFF_C_HOUR),
+ DateQueryParameter.of(DateQueryParameterType.START_OF_DAY, Duration.ofDays(-1)));
+ case GLN_NORD_ENERGI_NET:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("TA031U200")), Set.of(NOTE_NET_TARIFF_C));
+ case GLN_NORDVESTJYSK_ELFORSYNING_NOE_NET:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("Net C")), Set.of(NOTE_NET_TARIFF_C));
+ case GLN_RADIUS:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("DT_C_01")), Set.of(NOTE_NET_TARIFF_C_HOUR),
+ DateQueryParameter.of(RADIUS_CUTOFF_DATE));
+ case GLN_RAH_NET:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("RAH-C")), Set.of(NOTE_NET_TARIFF_C_HOUR),
+ DateQueryParameter.of(DateQueryParameterType.START_OF_DAY, Duration.ofDays(-1)));
+ case GLN_RAVDEX:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("NT-C")), Set.of(NOTE_NET_TARIFF_C_HOUR),
+ DateQueryParameter.of(DateQueryParameterType.START_OF_DAY, Duration.ofDays(-1)));
+ case GLN_SUNDS_NET:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("SEF-NT-05")),
+ Set.of(NOTE_NET_TARIFF_C_FLEX_HOUR),
+ DateQueryParameter.of(DateQueryParameterType.START_OF_DAY, Duration.ofDays(-1)));
+ case GLN_TARM_ELVAERK_NET:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("TEV-NT-01")), Set.of(NOTE_NET_TARIFF_C));
+ case GLN_TREFOR_EL_NET:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("C")), Set.of(NOTE_NET_TARIFF_C_HOUR),
+ DateQueryParameter.of(DateQueryParameterType.START_OF_DAY, Duration.ofDays(-1)));
+ case GLN_TREFOR_EL_NET_OEST:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("46")), Set.of(NOTE_NET_TARIFF_C_HOUR),
+ DateQueryParameter.of(DateQueryParameterType.START_OF_DAY, Duration.ofDays(-1)));
+ case GLN_VEKSEL:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("NT-10")),
+ Set.of(NOTE_NET_TARIFF_C_HOUR + " NT-10"));
+ case GLN_VORES_ELNET:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("TNT1009")), Set.of(NOTE_NET_TARIFF_C_HOUR),
+ DateQueryParameter.of(DateQueryParameterType.START_OF_DAY, Duration.ofDays(-1)));
+ case GLN_ZEANET:
+ return new DatahubTariffFilter(Set.of(ChargeTypeCode.of("43110")), Set.of(NOTE_NET_TARIFF_C_HOUR),
+ DateQueryParameter.of(DateQueryParameterType.START_OF_DAY, Duration.ofDays(-1)));
+ default:
+ return new DatahubTariffFilter(Set.of(), Set.of(NOTE_NET_TARIFF_C),
+ DateQueryParameter.of(DateQueryParameterType.START_OF_YEAR));
+ }
+ }
+
+ public static DatahubTariffFilter getSystemTariff() {
+ return new DatahubTariffFilter(Set.of(), Set.of(NOTE_SYSTEM_TARIFF),
+ DateQueryParameter.of(ENERGINET_CUTOFF_DATE));
+ }
+
+ public static DatahubTariffFilter getElectricityTax() {
+ return new DatahubTariffFilter(Set.of(), Set.of(NOTE_ELECTRICITY_TAX),
+ DateQueryParameter.of(ENERGINET_CUTOFF_DATE));
+ }
+
+ public static DatahubTariffFilter getTransmissionNetTariff() {
+ return new DatahubTariffFilter(Set.of(), Set.of(NOTE_TRANSMISSION_NET_TARIFF),
+ DateQueryParameter.of(ENERGINET_CUTOFF_DATE));
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/DateQueryParameter.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/DateQueryParameter.java
new file mode 100644
index 000000000..70f996a97
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/DateQueryParameter.java
@@ -0,0 +1,90 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.api;
+
+import java.time.Duration;
+import java.time.LocalDate;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * This class represents a query parameter of type {@link LocalDate} or a
+ * dynamic date defined as {@link DateQueryParameterType} with an optional offset.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class DateQueryParameter {
+
+ public static final DateQueryParameter EMPTY = new DateQueryParameter();
+
+ private @Nullable LocalDate date;
+ private @Nullable Duration offset;
+ private @Nullable DateQueryParameterType dateType;
+
+ private DateQueryParameter() {
+ }
+
+ public DateQueryParameter(LocalDate date) {
+ this.date = date;
+ }
+
+ public DateQueryParameter(DateQueryParameterType dateType, Duration offset) {
+ this.dateType = dateType;
+ this.offset = offset;
+ }
+
+ public DateQueryParameter(DateQueryParameterType dateType) {
+ this.dateType = dateType;
+ }
+
+ @Override
+ public String toString() {
+ LocalDate date = this.date;
+ if (date != null) {
+ return date.toString();
+ }
+ DateQueryParameterType dateType = this.dateType;
+ if (dateType != null) {
+ Duration offset = this.offset;
+ if (offset == null) {
+ return dateType.toString();
+ } else {
+ return dateType.toString()
+ + (offset.isNegative() ? "-" + offset.abs().toString() : "+" + offset.toString());
+ }
+ }
+ return "null";
+ }
+
+ public boolean isEmpty() {
+ return this == EMPTY;
+ }
+
+ public static DateQueryParameter of(LocalDate localDate) {
+ return new DateQueryParameter(localDate);
+ }
+
+ public static DateQueryParameter of(DateQueryParameterType dateType, Duration offset) {
+ if (offset.isZero()) {
+ return new DateQueryParameter(dateType);
+ } else {
+ return new DateQueryParameter(dateType, offset);
+ }
+ }
+
+ public static DateQueryParameter of(DateQueryParameterType dateType) {
+ return new DateQueryParameter(dateType);
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/DateQueryParameterType.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/DateQueryParameterType.java
new file mode 100644
index 000000000..3d951b27c
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/DateQueryParameterType.java
@@ -0,0 +1,40 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.api;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * This class represents a dynamic date to be used in a query.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public enum DateQueryParameterType {
+ NOW("now"),
+ UTC_NOW("utcnow"),
+ START_OF_DAY("StartOfDay"),
+ START_OF_MONTH("StartOfMonth"),
+ START_OF_YEAR("StartOfYear");
+
+ private final String name;
+
+ DateQueryParameterType(String name) {
+ this.name = name;
+ }
+
+ @Override
+ public String toString() {
+ return name;
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/GlobalLocationNumber.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/GlobalLocationNumber.java
new file mode 100644
index 000000000..4baa897b0
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/GlobalLocationNumber.java
@@ -0,0 +1,73 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.api;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Global Location Number.
+ * See {@link https://www.gs1.org/standards/id-keys/gln}}
+ * The Global Location Number (GLN) can be used by companies to identify their locations.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class GlobalLocationNumber {
+
+ public static final GlobalLocationNumber EMPTY = new GlobalLocationNumber("");
+
+ private static final int MAX_LENGTH = 13;
+
+ private final String gln;
+
+ public GlobalLocationNumber(String gln) {
+ if (gln.length() > MAX_LENGTH) {
+ throw new IllegalArgumentException("Maximum length exceeded: " + gln);
+ }
+ this.gln = gln;
+ }
+
+ @Override
+ public String toString() {
+ return gln;
+ }
+
+ public boolean isEmpty() {
+ return this == EMPTY;
+ }
+
+ public boolean isValid() {
+ if (gln.length() != 13) {
+ return false;
+ }
+
+ int checksum = 0;
+ for (int i = 13 - 2; i >= 0; i--) {
+ int digit = Character.getNumericValue(gln.charAt(i));
+ checksum += (i % 2 == 0 ? digit : digit * 3);
+ }
+ int controlDigit = 10 - (checksum % 10);
+ if (controlDigit == 10) {
+ controlDigit = 0;
+ }
+
+ return controlDigit == Character.getNumericValue(gln.charAt(13 - 1));
+ }
+
+ public static GlobalLocationNumber of(String gln) {
+ if (gln.isBlank()) {
+ return EMPTY;
+ }
+ return new GlobalLocationNumber(gln);
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/dto/DatahubPricelistRecord.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/dto/DatahubPricelistRecord.java
new file mode 100644
index 000000000..4cbae1172
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/dto/DatahubPricelistRecord.java
@@ -0,0 +1,202 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.api.dto;
+
+import java.math.BigDecimal;
+import java.time.LocalDateTime;
+import java.time.LocalTime;
+import java.util.HashMap;
+import java.util.Map;
+import java.util.Objects;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * Record as part of {@link DatahubPricelistRecords} from Energi Data Service.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public record DatahubPricelistRecord(@SerializedName("ValidFrom") LocalDateTime validFrom,
+ @SerializedName("ValidTo") LocalDateTime validTo, @SerializedName("ChargeTypeCode") String chargeTypeCode,
+ @SerializedName("Price1") BigDecimal price1, @SerializedName("Price2") BigDecimal price2,
+ @SerializedName("Price3") BigDecimal price3, @SerializedName("Price4") BigDecimal price4,
+ @SerializedName("Price5") BigDecimal price5, @SerializedName("Price6") BigDecimal price6,
+ @SerializedName("Price7") BigDecimal price7, @SerializedName("Price8") BigDecimal price8,
+ @SerializedName("Price9") BigDecimal price9, @SerializedName("Price10") BigDecimal price10,
+ @SerializedName("Price11") BigDecimal price11, @SerializedName("Price12") BigDecimal price12,
+ @SerializedName("Price13") BigDecimal price13, @SerializedName("Price14") BigDecimal price14,
+ @SerializedName("Price15") BigDecimal price15, @SerializedName("Price16") BigDecimal price16,
+ @SerializedName("Price17") BigDecimal price17, @SerializedName("Price18") BigDecimal price18,
+ @SerializedName("Price19") BigDecimal price19, @SerializedName("Price20") BigDecimal price20,
+ @SerializedName("Price21") BigDecimal price21, @SerializedName("Price22") BigDecimal price22,
+ @SerializedName("Price23") BigDecimal price23, @SerializedName("Price24") BigDecimal price24) {
+
+ @Override
+ public LocalDateTime validTo() {
+ return Objects.isNull(validTo) ? LocalDateTime.MAX : validTo;
+ }
+
+ @Override
+ public BigDecimal price2() {
+ return Objects.requireNonNullElse(price2, price1());
+ }
+
+ @Override
+ public BigDecimal price3() {
+ return Objects.requireNonNullElse(price3, price1());
+ }
+
+ @Override
+ public BigDecimal price4() {
+ return Objects.requireNonNullElse(price4, price1());
+ }
+
+ @Override
+ public BigDecimal price5() {
+ return Objects.requireNonNullElse(price5, price1());
+ }
+
+ @Override
+ public BigDecimal price6() {
+ return Objects.requireNonNullElse(price6, price1());
+ }
+
+ @Override
+ public BigDecimal price7() {
+ return Objects.requireNonNullElse(price7, price1());
+ }
+
+ @Override
+ public BigDecimal price8() {
+ return Objects.requireNonNullElse(price8, price1());
+ }
+
+ @Override
+ public BigDecimal price9() {
+ return Objects.requireNonNullElse(price9, price1());
+ }
+
+ @Override
+ public BigDecimal price10() {
+ return Objects.requireNonNullElse(price10, price1());
+ }
+
+ @Override
+ public BigDecimal price11() {
+ return Objects.requireNonNullElse(price11, price1());
+ }
+
+ @Override
+ public BigDecimal price12() {
+ return Objects.requireNonNullElse(price12, price1());
+ }
+
+ @Override
+ public BigDecimal price13() {
+ return Objects.requireNonNullElse(price13, price1());
+ }
+
+ @Override
+ public BigDecimal price14() {
+ return Objects.requireNonNullElse(price14, price1());
+ }
+
+ @Override
+ public BigDecimal price15() {
+ return Objects.requireNonNullElse(price15, price1());
+ }
+
+ @Override
+ public BigDecimal price16() {
+ return Objects.requireNonNullElse(price16, price1());
+ }
+
+ @Override
+ public BigDecimal price17() {
+ return Objects.requireNonNullElse(price17, price1());
+ }
+
+ @Override
+ public BigDecimal price18() {
+ return Objects.requireNonNullElse(price18, price1());
+ }
+
+ @Override
+ public BigDecimal price19() {
+ return Objects.requireNonNullElse(price19, price1());
+ }
+
+ @Override
+ public BigDecimal price20() {
+ return Objects.requireNonNullElse(price20, price1());
+ }
+
+ @Override
+ public BigDecimal price21() {
+ return Objects.requireNonNullElse(price21, price1());
+ }
+
+ @Override
+ public BigDecimal price22() {
+ return Objects.requireNonNullElse(price22, price1());
+ }
+
+ @Override
+ public BigDecimal price23() {
+ return Objects.requireNonNullElse(price23, price1());
+ }
+
+ @Override
+ public BigDecimal price24() {
+ return Objects.requireNonNullElse(price24, price1());
+ }
+
+ /**
+ * Get {@link Map} of tariffs with hour start as key.
+ *
+ * @return map with hourly tariffs
+ */
+ public Map getTariffMap() {
+ Map tariffMap = new HashMap<>();
+
+ tariffMap.put(LocalTime.of(0, 0), price1());
+ tariffMap.put(LocalTime.of(1, 0), price2());
+ tariffMap.put(LocalTime.of(2, 0), price3());
+ tariffMap.put(LocalTime.of(3, 0), price4());
+ tariffMap.put(LocalTime.of(4, 0), price5());
+ tariffMap.put(LocalTime.of(5, 0), price6());
+ tariffMap.put(LocalTime.of(6, 0), price7());
+ tariffMap.put(LocalTime.of(7, 0), price8());
+ tariffMap.put(LocalTime.of(8, 0), price9());
+ tariffMap.put(LocalTime.of(9, 0), price10());
+ tariffMap.put(LocalTime.of(10, 0), price11());
+ tariffMap.put(LocalTime.of(11, 0), price12());
+ tariffMap.put(LocalTime.of(12, 0), price13());
+ tariffMap.put(LocalTime.of(13, 0), price14());
+ tariffMap.put(LocalTime.of(14, 0), price15());
+ tariffMap.put(LocalTime.of(15, 0), price16());
+ tariffMap.put(LocalTime.of(16, 0), price17());
+ tariffMap.put(LocalTime.of(17, 0), price18());
+ tariffMap.put(LocalTime.of(18, 0), price19());
+ tariffMap.put(LocalTime.of(19, 0), price20());
+ tariffMap.put(LocalTime.of(20, 0), price21());
+ tariffMap.put(LocalTime.of(21, 0), price22());
+ tariffMap.put(LocalTime.of(22, 0), price23());
+ tariffMap.put(LocalTime.of(23, 0), price24());
+
+ return tariffMap;
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/dto/DatahubPricelistRecords.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/dto/DatahubPricelistRecords.java
new file mode 100644
index 000000000..4aa6c28a5
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/dto/DatahubPricelistRecords.java
@@ -0,0 +1,25 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.api.dto;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Received {@link DatahubPricelistRecords} from Energi Data Service.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public record DatahubPricelistRecords(int total, String filters, int limit, String dataset,
+ DatahubPricelistRecord[] records) {
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/dto/ElspotpriceRecord.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/dto/ElspotpriceRecord.java
new file mode 100644
index 000000000..a657471c9
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/dto/ElspotpriceRecord.java
@@ -0,0 +1,31 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.api.dto;
+
+import java.math.BigDecimal;
+import java.time.Instant;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * Record as part of {@link ElspotpriceRecords} from Energi Data Service.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public record ElspotpriceRecord(@SerializedName("HourUTC") Instant hour,
+ @SerializedName("SpotPriceDKK") BigDecimal spotPriceDKK,
+ @SerializedName("SpotPriceEUR") BigDecimal spotPriceEUR) {
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/dto/ElspotpriceRecords.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/dto/ElspotpriceRecords.java
new file mode 100644
index 000000000..87c8c2d71
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/dto/ElspotpriceRecords.java
@@ -0,0 +1,24 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.api.dto;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Received {@link ElspotpriceRecords} from Energi Data Service.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public record ElspotpriceRecords(int total, String filters, String dataset, ElspotpriceRecord[] records) {
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/serialization/InstantDeserializer.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/serialization/InstantDeserializer.java
new file mode 100644
index 000000000..0591540f4
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/serialization/InstantDeserializer.java
@@ -0,0 +1,47 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.api.serialization;
+
+import java.lang.reflect.Type;
+import java.time.Instant;
+import java.time.format.DateTimeParseException;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.JsonDeserializationContext;
+import com.google.gson.JsonDeserializer;
+import com.google.gson.JsonElement;
+import com.google.gson.JsonParseException;
+
+/**
+ * The {@link InstantDeserializer} converts a formatted UTC string to {@link Instant}.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class InstantDeserializer implements JsonDeserializer {
+
+ @Override
+ public @Nullable Instant deserialize(JsonElement element, Type arg1, JsonDeserializationContext arg2)
+ throws JsonParseException {
+ String content = element.getAsString();
+ try {
+ // When writing this, the format of the provided UTC strings lacks the trailing 'Z'.
+ // In case this would be fixed in the future, gracefully support both with and without this.
+ return Instant.parse(content.endsWith("Z") ? content : content + "Z");
+ } catch (DateTimeParseException e) {
+ throw new JsonParseException("Could not parse as Instant: " + content, e);
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/serialization/LocalDateDeserializer.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/serialization/LocalDateDeserializer.java
new file mode 100644
index 000000000..76f0b1faf
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/serialization/LocalDateDeserializer.java
@@ -0,0 +1,44 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.api.serialization;
+
+import java.lang.reflect.Type;
+import java.time.LocalDate;
+import java.time.format.DateTimeParseException;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.JsonDeserializationContext;
+import com.google.gson.JsonDeserializer;
+import com.google.gson.JsonElement;
+import com.google.gson.JsonParseException;
+
+/**
+ * The {@link LocalDateDeserializer} converts a formatted string to {@link LocalDate}.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class LocalDateDeserializer implements JsonDeserializer {
+
+ @Override
+ public @Nullable LocalDate deserialize(JsonElement element, Type arg1, JsonDeserializationContext arg2)
+ throws JsonParseException {
+ try {
+ return LocalDate.parse(element.getAsString().substring(0, 10));
+ } catch (DateTimeParseException e) {
+ throw new JsonParseException("Could not parse as LocalDate: " + element.getAsString(), e);
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/serialization/LocalDateTimeDeserializer.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/serialization/LocalDateTimeDeserializer.java
new file mode 100644
index 000000000..81b245879
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/api/serialization/LocalDateTimeDeserializer.java
@@ -0,0 +1,44 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.api.serialization;
+
+import java.lang.reflect.Type;
+import java.time.LocalDateTime;
+import java.time.format.DateTimeParseException;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.JsonDeserializationContext;
+import com.google.gson.JsonDeserializer;
+import com.google.gson.JsonElement;
+import com.google.gson.JsonParseException;
+
+/**
+ * The {@link LocalDateTimeDeserializer} converts a formatted string to {@link LocalDateTime}.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class LocalDateTimeDeserializer implements JsonDeserializer {
+
+ @Override
+ public @Nullable LocalDateTime deserialize(JsonElement element, Type arg1, JsonDeserializationContext arg2)
+ throws JsonParseException {
+ try {
+ return LocalDateTime.parse(element.getAsString());
+ } catch (DateTimeParseException e) {
+ throw new JsonParseException("Could not parse as LocalDateTime: " + element.getAsString(), e);
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/config/DatahubPriceConfiguration.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/config/DatahubPriceConfiguration.java
new file mode 100644
index 000000000..8a59da81d
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/config/DatahubPriceConfiguration.java
@@ -0,0 +1,103 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.config;
+
+import java.time.LocalDate;
+import java.time.format.DateTimeParseException;
+import java.util.Arrays;
+import java.util.HashSet;
+import java.util.Set;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.energidataservice.internal.api.ChargeTypeCode;
+import org.openhab.binding.energidataservice.internal.api.DateQueryParameter;
+import org.openhab.binding.energidataservice.internal.api.DateQueryParameterType;
+
+/**
+ * The {@link DatahubPriceConfiguration} class contains fields mapping channel configuration parameters.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class DatahubPriceConfiguration {
+
+ /**
+ * Comma-separated list of charge type codes, e.g. "CD,CD R".
+ */
+ public String chargeTypeCodes = "";
+
+ /**
+ * Comma-separated list of notes, e.g. "Nettarif C".
+ */
+ public String notes = "";
+
+ /**
+ * Query start date parameter expressed as either yyyy-mm-dd or one of StartOfDay, StartOfMonth or StartOfYear.
+ */
+ public String start = "";
+
+ /**
+ * Check if any filter values are provided.
+ *
+ * @return true if either charge type codes, notes or query start date is provided.
+ */
+ public boolean hasAnyFilterOverrides() {
+ return !chargeTypeCodes.isBlank() || !notes.isBlank() || !start.isBlank();
+ }
+
+ /**
+ * Get parsed set of charge type codes from comma-separated string.
+ *
+ * @return Set of charge type codes.
+ */
+ public Set getChargeTypeCodes() {
+ return chargeTypeCodes.isBlank() ? new HashSet<>()
+ : new HashSet(
+ Arrays.stream(chargeTypeCodes.split(",")).map(ChargeTypeCode::new).toList());
+ }
+
+ /**
+ * Get parsed set of notes from comma-separated string.
+ *
+ * @return Set of notes.
+ */
+ public Set getNotes() {
+ return notes.isBlank() ? new HashSet<>() : new HashSet(Arrays.asList(notes.split(",")));
+ }
+
+ /**
+ * Get query start parameter.
+ *
+ * @return null if invalid, otherwise an initialized {@link DateQueryParameter}.
+ */
+ public @Nullable DateQueryParameter getStart() {
+ if (start.isBlank()) {
+ return DateQueryParameter.EMPTY;
+ }
+ if (start.equals(DateQueryParameterType.START_OF_DAY.toString())) {
+ return DateQueryParameter.of(DateQueryParameterType.START_OF_DAY);
+ }
+ if (start.equals(DateQueryParameterType.START_OF_MONTH.toString())) {
+ return DateQueryParameter.of(DateQueryParameterType.START_OF_MONTH);
+ }
+ if (start.equals(DateQueryParameterType.START_OF_YEAR.toString())) {
+ return DateQueryParameter.of(DateQueryParameterType.START_OF_YEAR);
+ }
+ try {
+ return DateQueryParameter.of(LocalDate.parse(start));
+ } catch (DateTimeParseException e) {
+ return null;
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/config/EnergiDataServiceConfiguration.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/config/EnergiDataServiceConfiguration.java
new file mode 100644
index 000000000..70cf0b45d
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/config/EnergiDataServiceConfiguration.java
@@ -0,0 +1,65 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.config;
+
+import java.util.Currency;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.energidataservice.internal.EnergiDataServiceBindingConstants;
+import org.openhab.binding.energidataservice.internal.api.GlobalLocationNumber;
+
+/**
+ * The {@link EnergiDataServiceConfiguration} class contains fields mapping thing configuration parameters.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class EnergiDataServiceConfiguration {
+
+ /**
+ * Price area (DK1 = West of the Great Belt, DK2 = East of the Great Belt).
+ */
+ public String priceArea = "";
+
+ /**
+ * Currency code for the prices.
+ */
+ public String currencyCode = EnergiDataServiceBindingConstants.CURRENCY_DKK.getCurrencyCode();
+
+ /**
+ * Global Location Number of the Grid Company.
+ */
+ public String gridCompanyGLN = "";
+
+ /**
+ * Global Location Number of Energinet.
+ */
+ public String energinetGLN = "5790000432752";
+
+ /**
+ * Get {@link Currency} representing the configured currency code.
+ *
+ * @return Currency instance
+ */
+ public Currency getCurrency() {
+ return Currency.getInstance(currencyCode);
+ }
+
+ public GlobalLocationNumber getGridCompanyGLN() {
+ return GlobalLocationNumber.of(gridCompanyGLN);
+ }
+
+ public GlobalLocationNumber getEnerginetGLN() {
+ return GlobalLocationNumber.of(energinetGLN);
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/exception/DataServiceException.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/exception/DataServiceException.java
new file mode 100644
index 000000000..ab2c522e4
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/exception/DataServiceException.java
@@ -0,0 +1,50 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.exception;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * {@link DataServiceException} is a generic Energi Data Service exception thrown in case
+ * of communication failure or unexpected response. It is intended to be derived by
+ * specialized exceptions.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class DataServiceException extends Exception {
+
+ private static final long serialVersionUID = 1L;
+ private int httpStatus = 0;
+
+ public DataServiceException(String message) {
+ super(message);
+ }
+
+ public DataServiceException(Throwable cause) {
+ super(cause);
+ }
+
+ public DataServiceException(String message, Throwable cause) {
+ super(message, cause);
+ }
+
+ public DataServiceException(String message, int httpStatus) {
+ super(message);
+ this.httpStatus = httpStatus;
+ }
+
+ public int getHttpStatus() {
+ return httpStatus;
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/exception/MissingPriceException.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/exception/MissingPriceException.java
new file mode 100644
index 000000000..95814c24a
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/exception/MissingPriceException.java
@@ -0,0 +1,31 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.exception;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * {@link MissingPriceException} is thrown when there are no prices
+ * available in the requested interval, e.g. when performing a calculation.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class MissingPriceException extends Exception {
+
+ private static final long serialVersionUID = 1L;
+
+ public MissingPriceException(String message) {
+ super(message);
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/factory/EnergiDataServiceHandlerFactory.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/factory/EnergiDataServiceHandlerFactory.java
new file mode 100644
index 000000000..4f2f4ef23
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/factory/EnergiDataServiceHandlerFactory.java
@@ -0,0 +1,73 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.factory;
+
+import static org.openhab.binding.energidataservice.internal.EnergiDataServiceBindingConstants.*;
+
+import java.util.Set;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.eclipse.jetty.client.HttpClient;
+import org.openhab.binding.energidataservice.internal.handler.EnergiDataServiceHandler;
+import org.openhab.core.i18n.TimeZoneProvider;
+import org.openhab.core.io.net.http.HttpClientFactory;
+import org.openhab.core.thing.Thing;
+import org.openhab.core.thing.ThingTypeUID;
+import org.openhab.core.thing.binding.BaseThingHandlerFactory;
+import org.openhab.core.thing.binding.ThingHandler;
+import org.openhab.core.thing.binding.ThingHandlerFactory;
+import org.osgi.service.component.ComponentContext;
+import org.osgi.service.component.annotations.Activate;
+import org.osgi.service.component.annotations.Component;
+import org.osgi.service.component.annotations.Reference;
+
+/**
+ * The {@link EnergiDataServiceHandlerFactory} is responsible for creating things and thing
+ * handlers.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+@Component(configurationPid = "binding.energidataservice", service = ThingHandlerFactory.class)
+public class EnergiDataServiceHandlerFactory extends BaseThingHandlerFactory {
+
+ private static final Set SUPPORTED_THING_TYPES_UIDS = Set.of(THING_TYPE_SERVICE);
+
+ private final HttpClient httpClient;
+ private final TimeZoneProvider timeZoneProvider;
+
+ @Activate
+ public EnergiDataServiceHandlerFactory(final @Reference HttpClientFactory httpClientFactory,
+ final @Reference TimeZoneProvider timeZoneProvider, ComponentContext componentContext) {
+ super.activate(componentContext);
+ this.httpClient = httpClientFactory.getCommonHttpClient();
+ this.timeZoneProvider = timeZoneProvider;
+ }
+
+ @Override
+ public boolean supportsThingType(ThingTypeUID thingTypeUID) {
+ return SUPPORTED_THING_TYPES_UIDS.contains(thingTypeUID);
+ }
+
+ @Override
+ protected @Nullable ThingHandler createHandler(Thing thing) {
+ ThingTypeUID thingTypeUID = thing.getThingTypeUID();
+
+ if (THING_TYPE_SERVICE.equals(thingTypeUID)) {
+ return new EnergiDataServiceHandler(thing, httpClient, timeZoneProvider);
+ }
+
+ return null;
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/handler/EnergiDataServiceHandler.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/handler/EnergiDataServiceHandler.java
new file mode 100644
index 000000000..16bffcf8d
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/handler/EnergiDataServiceHandler.java
@@ -0,0 +1,541 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.handler;
+
+import static org.openhab.binding.energidataservice.internal.EnergiDataServiceBindingConstants.*;
+
+import java.math.BigDecimal;
+import java.time.Duration;
+import java.time.Instant;
+import java.time.LocalDateTime;
+import java.time.LocalTime;
+import java.time.format.DateTimeFormatter;
+import java.time.temporal.ChronoUnit;
+import java.util.Collection;
+import java.util.Currency;
+import java.util.List;
+import java.util.Map;
+import java.util.Map.Entry;
+import java.util.Set;
+import java.util.concurrent.ScheduledFuture;
+import java.util.concurrent.TimeUnit;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.eclipse.jetty.client.HttpClient;
+import org.eclipse.jetty.http.HttpStatus;
+import org.openhab.binding.energidataservice.internal.ApiController;
+import org.openhab.binding.energidataservice.internal.CacheManager;
+import org.openhab.binding.energidataservice.internal.action.EnergiDataServiceActions;
+import org.openhab.binding.energidataservice.internal.api.ChargeType;
+import org.openhab.binding.energidataservice.internal.api.ChargeTypeCode;
+import org.openhab.binding.energidataservice.internal.api.DatahubTariffFilter;
+import org.openhab.binding.energidataservice.internal.api.DatahubTariffFilterFactory;
+import org.openhab.binding.energidataservice.internal.api.DateQueryParameter;
+import org.openhab.binding.energidataservice.internal.api.DateQueryParameterType;
+import org.openhab.binding.energidataservice.internal.api.GlobalLocationNumber;
+import org.openhab.binding.energidataservice.internal.api.dto.DatahubPricelistRecord;
+import org.openhab.binding.energidataservice.internal.api.dto.ElspotpriceRecord;
+import org.openhab.binding.energidataservice.internal.config.DatahubPriceConfiguration;
+import org.openhab.binding.energidataservice.internal.config.EnergiDataServiceConfiguration;
+import org.openhab.binding.energidataservice.internal.exception.DataServiceException;
+import org.openhab.binding.energidataservice.internal.retry.RetryPolicyFactory;
+import org.openhab.binding.energidataservice.internal.retry.RetryStrategy;
+import org.openhab.core.i18n.TimeZoneProvider;
+import org.openhab.core.library.types.DecimalType;
+import org.openhab.core.library.types.StringType;
+import org.openhab.core.thing.Channel;
+import org.openhab.core.thing.ChannelUID;
+import org.openhab.core.thing.Thing;
+import org.openhab.core.thing.ThingStatus;
+import org.openhab.core.thing.ThingStatusDetail;
+import org.openhab.core.thing.binding.BaseThingHandler;
+import org.openhab.core.thing.binding.ThingHandlerService;
+import org.openhab.core.types.Command;
+import org.openhab.core.types.RefreshType;
+import org.openhab.core.types.UnDefType;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import com.google.gson.Gson;
+
+/**
+ * The {@link EnergiDataServiceHandler} is responsible for handling commands, which are
+ * sent to one of the channels.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class EnergiDataServiceHandler extends BaseThingHandler {
+
+ private final Logger logger = LoggerFactory.getLogger(EnergiDataServiceHandler.class);
+ private final TimeZoneProvider timeZoneProvider;
+ private final ApiController apiController;
+ private final CacheManager cacheManager;
+ private final Gson gson = new Gson();
+
+ private EnergiDataServiceConfiguration config;
+ private RetryStrategy retryPolicy = RetryPolicyFactory.initial();
+ private @Nullable ScheduledFuture> refreshFuture;
+ private @Nullable ScheduledFuture> priceUpdateFuture;
+
+ private record Price(String hourStart, BigDecimal spotPrice, String spotPriceCurrency,
+ @Nullable BigDecimal netTariff, @Nullable BigDecimal systemTariff, @Nullable BigDecimal electricityTax,
+ @Nullable BigDecimal transmissionNetTariff) {
+ }
+
+ public EnergiDataServiceHandler(Thing thing, HttpClient httpClient, TimeZoneProvider timeZoneProvider) {
+ super(thing);
+ this.timeZoneProvider = timeZoneProvider;
+ this.apiController = new ApiController(httpClient, timeZoneProvider);
+ this.cacheManager = new CacheManager();
+
+ // Default configuration
+ this.config = new EnergiDataServiceConfiguration();
+ }
+
+ @Override
+ public void handleCommand(ChannelUID channelUID, Command command) {
+ if (!(command instanceof RefreshType)) {
+ return;
+ }
+
+ if (ELECTRICITY_CHANNELS.contains(channelUID.getId())) {
+ refreshElectricityPrices();
+ }
+ }
+
+ @Override
+ public void initialize() {
+ config = getConfigAs(EnergiDataServiceConfiguration.class);
+
+ if (config.priceArea.isBlank()) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.OFFLINE.CONFIGURATION_ERROR,
+ "@text/offline.conf-error.no-price-area");
+ return;
+ }
+ GlobalLocationNumber gln = config.getGridCompanyGLN();
+ if (!gln.isEmpty() && !gln.isValid()) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.OFFLINE.CONFIGURATION_ERROR,
+ "@text/offline.conf-error.invalid-grid-company-gln");
+ return;
+ }
+ gln = config.getEnerginetGLN();
+ if (!gln.isEmpty() && !gln.isValid()) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.OFFLINE.CONFIGURATION_ERROR,
+ "@text/offline.conf-error.invalid-energinet-gln");
+ return;
+ }
+
+ updateStatus(ThingStatus.UNKNOWN);
+
+ refreshFuture = scheduler.schedule(this::refreshElectricityPrices, 0, TimeUnit.SECONDS);
+ }
+
+ @Override
+ public void dispose() {
+ ScheduledFuture> refreshFuture = this.refreshFuture;
+ if (refreshFuture != null) {
+ refreshFuture.cancel(true);
+ this.refreshFuture = null;
+ }
+ ScheduledFuture> priceUpdateFuture = this.priceUpdateFuture;
+ if (priceUpdateFuture != null) {
+ priceUpdateFuture.cancel(true);
+ this.priceUpdateFuture = null;
+ }
+
+ cacheManager.clear();
+ }
+
+ @Override
+ public Collection> getServices() {
+ return Set.of(EnergiDataServiceActions.class);
+ }
+
+ private void refreshElectricityPrices() {
+ RetryStrategy retryPolicy;
+ try {
+ if (isLinked(CHANNEL_SPOT_PRICE) || isLinked(CHANNEL_HOURLY_PRICES)) {
+ downloadSpotPrices();
+ }
+
+ if (isLinked(CHANNEL_NET_TARIFF) || isLinked(CHANNEL_HOURLY_PRICES)) {
+ downloadNetTariffs();
+ }
+
+ if (isLinked(CHANNEL_SYSTEM_TARIFF) || isLinked(CHANNEL_HOURLY_PRICES)) {
+ downloadSystemTariffs();
+ }
+
+ if (isLinked(CHANNEL_ELECTRICITY_TAX) || isLinked(CHANNEL_HOURLY_PRICES)) {
+ downloadElectricityTaxes();
+ }
+
+ if (isLinked(CHANNEL_TRANSMISSION_NET_TARIFF) || isLinked(CHANNEL_HOURLY_PRICES)) {
+ downloadTransmissionNetTariffs();
+ }
+
+ updateStatus(ThingStatus.ONLINE);
+ updatePrices();
+
+ if (isLinked(CHANNEL_SPOT_PRICE) || isLinked(CHANNEL_HOURLY_PRICES)) {
+ if (cacheManager.getNumberOfFutureSpotPrices() < 13) {
+ retryPolicy = RetryPolicyFactory.whenExpectedSpotPriceDataMissing(DAILY_REFRESH_TIME_CET,
+ NORD_POOL_TIMEZONE);
+ } else {
+ retryPolicy = RetryPolicyFactory.atFixedTime(DAILY_REFRESH_TIME_CET, NORD_POOL_TIMEZONE);
+ }
+ } else {
+ retryPolicy = RetryPolicyFactory.atFixedTime(LocalTime.MIDNIGHT, timeZoneProvider.getTimeZone());
+ }
+ } catch (DataServiceException e) {
+ if (e.getHttpStatus() != 0) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.OFFLINE.COMMUNICATION_ERROR,
+ HttpStatus.getCode(e.getHttpStatus()).getMessage());
+ } else {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.OFFLINE.COMMUNICATION_ERROR, e.getMessage());
+ }
+ if (e.getCause() != null) {
+ logger.debug("Error retrieving prices", e);
+ }
+ retryPolicy = RetryPolicyFactory.fromThrowable(e);
+ } catch (InterruptedException e) {
+ logger.debug("Refresh job interrupted");
+ Thread.currentThread().interrupt();
+ return;
+ }
+
+ rescheduleRefreshJob(retryPolicy);
+ }
+
+ private void downloadSpotPrices() throws InterruptedException, DataServiceException {
+ if (cacheManager.areSpotPricesFullyCached()) {
+ logger.debug("Cached spot prices still valid, skipping download.");
+ return;
+ }
+ DateQueryParameter start;
+ if (cacheManager.areHistoricSpotPricesCached()) {
+ start = DateQueryParameter.of(DateQueryParameterType.UTC_NOW);
+ } else {
+ start = DateQueryParameter.of(DateQueryParameterType.UTC_NOW,
+ Duration.ofHours(-CacheManager.NUMBER_OF_HISTORIC_HOURS));
+ }
+ Map properties = editProperties();
+ ElspotpriceRecord[] spotPriceRecords = apiController.getSpotPrices(config.priceArea, config.getCurrency(),
+ start, properties);
+ cacheManager.putSpotPrices(spotPriceRecords, config.getCurrency());
+ updateProperties(properties);
+ }
+
+ private void downloadNetTariffs() throws InterruptedException, DataServiceException {
+ if (config.getGridCompanyGLN().isEmpty()) {
+ return;
+ }
+ if (cacheManager.areNetTariffsValidTomorrow()) {
+ logger.debug("Cached net tariffs still valid, skipping download.");
+ cacheManager.updateNetTariffs();
+ } else {
+ cacheManager.putNetTariffs(downloadPriceLists(config.getGridCompanyGLN(), getNetTariffFilter()));
+ }
+ }
+
+ private void downloadSystemTariffs() throws InterruptedException, DataServiceException {
+ GlobalLocationNumber globalLocationNumber = config.getEnerginetGLN();
+ if (globalLocationNumber.isEmpty()) {
+ return;
+ }
+ if (cacheManager.areSystemTariffsValidTomorrow()) {
+ logger.debug("Cached system tariffs still valid, skipping download.");
+ cacheManager.updateSystemTariffs();
+ } else {
+ cacheManager.putSystemTariffs(
+ downloadPriceLists(globalLocationNumber, DatahubTariffFilterFactory.getSystemTariff()));
+ }
+ }
+
+ private void downloadElectricityTaxes() throws InterruptedException, DataServiceException {
+ GlobalLocationNumber globalLocationNumber = config.getEnerginetGLN();
+ if (globalLocationNumber.isEmpty()) {
+ return;
+ }
+ if (cacheManager.areElectricityTaxesValidTomorrow()) {
+ logger.debug("Cached electricity taxes still valid, skipping download.");
+ cacheManager.updateElectricityTaxes();
+ } else {
+ cacheManager.putElectricityTaxes(
+ downloadPriceLists(globalLocationNumber, DatahubTariffFilterFactory.getElectricityTax()));
+ }
+ }
+
+ private void downloadTransmissionNetTariffs() throws InterruptedException, DataServiceException {
+ GlobalLocationNumber globalLocationNumber = config.getEnerginetGLN();
+ if (globalLocationNumber.isEmpty()) {
+ return;
+ }
+ if (cacheManager.areTransmissionNetTariffsValidTomorrow()) {
+ logger.debug("Cached transmission net tariffs still valid, skipping download.");
+ cacheManager.updateTransmissionNetTariffs();
+ } else {
+ cacheManager.putTransmissionNetTariffs(
+ downloadPriceLists(globalLocationNumber, DatahubTariffFilterFactory.getTransmissionNetTariff()));
+ }
+ }
+
+ private Collection downloadPriceLists(GlobalLocationNumber globalLocationNumber,
+ DatahubTariffFilter filter) throws InterruptedException, DataServiceException {
+ Map properties = editProperties();
+ Collection records = apiController.getDatahubPriceLists(globalLocationNumber,
+ ChargeType.Tariff, filter, properties);
+ updateProperties(properties);
+
+ return records;
+ }
+
+ private DatahubTariffFilter getNetTariffFilter() {
+ Channel channel = getThing().getChannel(CHANNEL_NET_TARIFF);
+ if (channel == null) {
+ return DatahubTariffFilterFactory.getNetTariffByGLN(config.gridCompanyGLN);
+ }
+
+ DatahubPriceConfiguration datahubPriceConfiguration = channel.getConfiguration()
+ .as(DatahubPriceConfiguration.class);
+
+ if (!datahubPriceConfiguration.hasAnyFilterOverrides()) {
+ return DatahubTariffFilterFactory.getNetTariffByGLN(config.gridCompanyGLN);
+ }
+
+ DateQueryParameter start = datahubPriceConfiguration.getStart();
+ if (start == null) {
+ logger.warn("Invalid channel configuration parameter 'start': {}", datahubPriceConfiguration.start);
+ return DatahubTariffFilterFactory.getNetTariffByGLN(config.gridCompanyGLN);
+ }
+
+ Set chargeTypeCodes = datahubPriceConfiguration.getChargeTypeCodes();
+ Set notes = datahubPriceConfiguration.getNotes();
+ if (!chargeTypeCodes.isEmpty() || !notes.isEmpty()) {
+ // Completely override filter.
+ return new DatahubTariffFilter(chargeTypeCodes, notes, start);
+ } else {
+ // Only override start date in pre-configured filter.
+ return new DatahubTariffFilter(DatahubTariffFilterFactory.getNetTariffByGLN(config.gridCompanyGLN), start);
+ }
+ }
+
+ private void updatePrices() {
+ cacheManager.cleanup();
+
+ updateCurrentSpotPrice();
+ updateCurrentTariff(CHANNEL_NET_TARIFF, cacheManager.getNetTariff());
+ updateCurrentTariff(CHANNEL_SYSTEM_TARIFF, cacheManager.getSystemTariff());
+ updateCurrentTariff(CHANNEL_ELECTRICITY_TAX, cacheManager.getElectricityTax());
+ updateCurrentTariff(CHANNEL_TRANSMISSION_NET_TARIFF, cacheManager.getTransmissionNetTariff());
+ updateHourlyPrices();
+
+ reschedulePriceUpdateJob();
+ }
+
+ private void updateCurrentSpotPrice() {
+ if (!isLinked(CHANNEL_SPOT_PRICE)) {
+ return;
+ }
+ BigDecimal spotPrice = cacheManager.getSpotPrice();
+ updateState(CHANNEL_SPOT_PRICE, spotPrice != null ? new DecimalType(spotPrice) : UnDefType.UNDEF);
+ }
+
+ private void updateCurrentTariff(String channelId, @Nullable BigDecimal tariff) {
+ if (!isLinked(channelId)) {
+ return;
+ }
+ updateState(channelId, tariff != null ? new DecimalType(tariff) : UnDefType.UNDEF);
+ }
+
+ private void updateHourlyPrices() {
+ if (!isLinked(CHANNEL_HOURLY_PRICES)) {
+ return;
+ }
+ Map spotPriceMap = cacheManager.getSpotPrices();
+ Price[] targetPrices = new Price[spotPriceMap.size()];
+ List> sourcePrices = spotPriceMap.entrySet().stream()
+ .sorted(Map.Entry.comparingByKey()).toList();
+
+ int i = 0;
+ for (Entry sourcePrice : sourcePrices) {
+ Instant hourStart = sourcePrice.getKey();
+ BigDecimal netTariff = cacheManager.getNetTariff(hourStart);
+ BigDecimal systemTariff = cacheManager.getSystemTariff(hourStart);
+ BigDecimal electricityTax = cacheManager.getElectricityTax(hourStart);
+ BigDecimal transmissionNetTariff = cacheManager.getTransmissionNetTariff(hourStart);
+ targetPrices[i++] = new Price(hourStart.toString(), sourcePrice.getValue(), config.currencyCode, netTariff,
+ systemTariff, electricityTax, transmissionNetTariff);
+ }
+ updateState(CHANNEL_HOURLY_PRICES, new StringType(gson.toJson(targetPrices)));
+ }
+
+ /**
+ * Get the configured {@link Currency} for spot prices.
+ *
+ * @return Spot price currency
+ */
+ public Currency getCurrency() {
+ return config.getCurrency();
+ }
+
+ /**
+ * Get cached spot prices or try once to download them if not cached
+ * (usually if no items are linked).
+ *
+ * @return Map of future spot prices
+ */
+ public Map getSpotPrices() {
+ try {
+ downloadSpotPrices();
+ } catch (DataServiceException e) {
+ if (logger.isDebugEnabled()) {
+ logger.warn("Error retrieving spot prices", e);
+ } else {
+ logger.warn("Error retrieving spot prices: {}", e.getMessage());
+ }
+ } catch (InterruptedException e) {
+ Thread.currentThread().interrupt();
+ }
+
+ return cacheManager.getSpotPrices();
+ }
+
+ /**
+ * Get cached net tariffs or try once to download them if not cached
+ * (usually if no items are linked).
+ *
+ * @return Map of future net tariffs
+ */
+ public Map getNetTariffs() {
+ try {
+ downloadNetTariffs();
+ } catch (DataServiceException e) {
+ if (logger.isDebugEnabled()) {
+ logger.warn("Error retrieving net tariffs", e);
+ } else {
+ logger.warn("Error retrieving net tariffs: {}", e.getMessage());
+ }
+ } catch (InterruptedException e) {
+ Thread.currentThread().interrupt();
+ }
+
+ return cacheManager.getNetTariffs();
+ }
+
+ /**
+ * Get cached system tariffs or try once to download them if not cached
+ * (usually if no items are linked).
+ *
+ * @return Map of future system tariffs
+ */
+ public Map getSystemTariffs() {
+ try {
+ downloadSystemTariffs();
+ } catch (DataServiceException e) {
+ if (logger.isDebugEnabled()) {
+ logger.warn("Error retrieving system tariffs", e);
+ } else {
+ logger.warn("Error retrieving system tariffs: {}", e.getMessage());
+ }
+ } catch (InterruptedException e) {
+ Thread.currentThread().interrupt();
+ }
+
+ return cacheManager.getSystemTariffs();
+ }
+
+ /**
+ * Get cached electricity taxes or try once to download them if not cached
+ * (usually if no items are linked).
+ *
+ * @return Map of future electricity taxes
+ */
+ public Map getElectricityTaxes() {
+ try {
+ downloadElectricityTaxes();
+ } catch (DataServiceException e) {
+ if (logger.isDebugEnabled()) {
+ logger.warn("Error retrieving electricity taxes", e);
+ } else {
+ logger.warn("Error retrieving electricity taxes: {}", e.getMessage());
+ }
+ } catch (InterruptedException e) {
+ Thread.currentThread().interrupt();
+ }
+
+ return cacheManager.getElectricityTaxes();
+ }
+
+ /**
+ * Return cached transmission net tariffs or try once to download them if not cached
+ * (usually if no items are linked).
+ *
+ * @return Map of future transmissions net tariffs
+ */
+ public Map getTransmissionNetTariffs() {
+ try {
+ downloadTransmissionNetTariffs();
+ } catch (DataServiceException e) {
+ if (logger.isDebugEnabled()) {
+ logger.warn("Error retrieving transmission net tariffs", e);
+ } else {
+ logger.warn("Error retrieving transmission net tariffs: {}", e.getMessage());
+ }
+ } catch (InterruptedException e) {
+ Thread.currentThread().interrupt();
+ }
+
+ return cacheManager.getTransmissionNetTariffs();
+ }
+
+ private void reschedulePriceUpdateJob() {
+ ScheduledFuture> priceUpdateJob = this.priceUpdateFuture;
+ if (priceUpdateJob != null) {
+ // Do not interrupt ourselves.
+ priceUpdateJob.cancel(false);
+ this.priceUpdateFuture = null;
+ }
+
+ Instant now = Instant.now();
+ long millisUntilNextClockHour = Duration
+ .between(now, now.plus(1, ChronoUnit.HOURS).truncatedTo(ChronoUnit.HOURS)).toMillis() + 1;
+ this.priceUpdateFuture = scheduler.schedule(this::updatePrices, millisUntilNextClockHour,
+ TimeUnit.MILLISECONDS);
+ logger.debug("Price update job rescheduled in {} milliseconds", millisUntilNextClockHour);
+ }
+
+ private void rescheduleRefreshJob(RetryStrategy retryPolicy) {
+ // Preserve state of previous retry policy when configuration is the same.
+ if (!retryPolicy.equals(this.retryPolicy)) {
+ this.retryPolicy = retryPolicy;
+ }
+
+ ScheduledFuture> refreshJob = this.refreshFuture;
+
+ long secondsUntilNextRefresh = this.retryPolicy.getDuration().getSeconds();
+ Instant timeOfNextRefresh = Instant.now().plusSeconds(secondsUntilNextRefresh);
+ this.refreshFuture = scheduler.schedule(this::refreshElectricityPrices, secondsUntilNextRefresh,
+ TimeUnit.SECONDS);
+ logger.debug("Refresh job rescheduled in {} seconds: {}", secondsUntilNextRefresh, timeOfNextRefresh);
+ DateTimeFormatter formatter = DateTimeFormatter.ofPattern(PROPERTY_DATETIME_FORMAT);
+ updateProperty(PROPERTY_NEXT_CALL, LocalDateTime.ofInstant(timeOfNextRefresh, timeZoneProvider.getTimeZone())
+ .truncatedTo(ChronoUnit.SECONDS).format(formatter));
+
+ if (refreshJob != null) {
+ refreshJob.cancel(true);
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/retry/RetryPolicyFactory.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/retry/RetryPolicyFactory.java
new file mode 100644
index 000000000..00f0ab89c
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/retry/RetryPolicyFactory.java
@@ -0,0 +1,89 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.retry;
+
+import java.time.Clock;
+import java.time.Duration;
+import java.time.LocalTime;
+import java.time.ZoneId;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.http.HttpStatus;
+import org.openhab.binding.energidataservice.internal.exception.DataServiceException;
+import org.openhab.binding.energidataservice.internal.retry.strategy.ExponentialBackoff;
+import org.openhab.binding.energidataservice.internal.retry.strategy.FixedTime;
+import org.openhab.binding.energidataservice.internal.retry.strategy.Linear;
+
+/**
+ * This factory defines policies for determining appropriate {@link RetryStrategy} based
+ * on scenario.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class RetryPolicyFactory {
+
+ /**
+ * Determine {@link RetryStrategy} from {@link Throwable}.
+ *
+ * @param e thrown exception
+ * @return retry strategy
+ */
+ public static RetryStrategy fromThrowable(Throwable e) {
+ if (e instanceof DataServiceException dse) {
+ switch (dse.getHttpStatus()) {
+ case HttpStatus.TOO_MANY_REQUESTS_429:
+ return new ExponentialBackoff().withMinimum(Duration.ofMinutes(30));
+ default:
+ return new ExponentialBackoff().withMinimum(Duration.ofMinutes(1)).withJitter(0.2);
+ }
+ }
+
+ return new ExponentialBackoff().withMinimum(Duration.ofMinutes(1)).withJitter(0.2);
+ }
+
+ /**
+ * Default {@link RetryStrategy} with one retry per day.
+ * This is intended as a dummy strategy until replaced by a concrete one.
+ *
+ * @return retry strategy
+ */
+ public static RetryStrategy initial() {
+ return new Linear().withMinimum(Duration.ofDays(1));
+ }
+
+ /**
+ * Determine {@link RetryStrategy} for next expected data publishing.
+ *
+ * @param localTime the time of daily data request in local time-zone
+ * @param zoneId the local time-zone
+ * @return retry strategy
+ */
+ public static RetryStrategy atFixedTime(LocalTime localTime, ZoneId zoneId) {
+ return new FixedTime(localTime, Clock.system(zoneId)).withJitter(1);
+ }
+
+ /**
+ * Determine {@link RetryStrategy} when expected spot price data is missing.
+ *
+ * @param utcTime the time of daily data request in UTC time-zone
+ * @return retry strategy
+ */
+ public static RetryStrategy whenExpectedSpotPriceDataMissing(LocalTime localTime, ZoneId zoneId) {
+ LocalTime now = LocalTime.now(zoneId);
+ if (now.isAfter(localTime)) {
+ return new ExponentialBackoff().withMinimum(Duration.ofMinutes(10)).withJitter(0.2);
+ }
+ return atFixedTime(localTime, zoneId);
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/retry/RetryStrategy.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/retry/RetryStrategy.java
new file mode 100644
index 000000000..eb67ba830
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/retry/RetryStrategy.java
@@ -0,0 +1,34 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.retry;
+
+import java.time.Duration;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * This interface defines a retry strategy for failed network
+ * requests.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public interface RetryStrategy {
+ /**
+ * Get {@link Duration} until next attempt. This will auto-increment number of
+ * attempts, so should only be called once after each failed request.
+ *
+ * @return duration until next attempt according to strategy
+ */
+ Duration getDuration();
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/retry/strategy/ExponentialBackoff.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/retry/strategy/ExponentialBackoff.java
new file mode 100644
index 000000000..4b510ae22
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/retry/strategy/ExponentialBackoff.java
@@ -0,0 +1,105 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.retry.strategy;
+
+import java.time.Duration;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.energidataservice.internal.retry.RetryStrategy;
+
+/**
+ * This implements a {@link RetryStrategy} for exponential backoff with jitter.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class ExponentialBackoff implements RetryStrategy {
+
+ private int attempts = 0;
+ private int factor = 2;
+ private double jitter = 0.0;
+ private Duration minimum = Duration.ofMillis(100);
+ private Duration maximum = Duration.ofHours(6);
+
+ public ExponentialBackoff() {
+ }
+
+ @Override
+ public Duration getDuration() {
+ long minimum = this.minimum.toMillis();
+ long maximum = this.maximum.toMillis();
+ long duration = minimum * (long) Math.pow(this.factor, this.attempts++);
+ if (jitter != 0.0) {
+ double rand = Math.random();
+ if ((((int) Math.floor(rand * 10)) & 1) == 0) {
+ duration += (long) (rand * jitter * duration);
+ } else {
+ duration -= (long) (rand * jitter * duration);
+ }
+ }
+ if (duration < minimum) {
+ duration = minimum;
+ }
+ if (duration > maximum) {
+ duration = maximum;
+ }
+ return Duration.ofMillis(duration);
+ }
+
+ public ExponentialBackoff withFactor(int factor) {
+ this.factor = factor;
+ return this;
+ }
+
+ public ExponentialBackoff withJitter(double jitter) {
+ this.jitter = jitter;
+ return this;
+ }
+
+ public ExponentialBackoff withMinimum(Duration minimum) {
+ this.minimum = minimum;
+ return this;
+ }
+
+ public ExponentialBackoff withMaximum(Duration maximum) {
+ this.maximum = maximum;
+ return this;
+ }
+
+ @Override
+ public boolean equals(@Nullable Object o) {
+ if (o == this) {
+ return true;
+ }
+ if (!(o instanceof ExponentialBackoff)) {
+ return false;
+ }
+ ExponentialBackoff other = (ExponentialBackoff) o;
+
+ return this.factor == other.factor && this.jitter == other.jitter && this.minimum.equals(other.minimum)
+ && this.maximum.equals(other.maximum);
+ }
+
+ @Override
+ public final int hashCode() {
+ final int prime = 31;
+ int result = 1;
+ result = prime * result + factor;
+ result = prime * result + (int) jitter * 100;
+ result = prime * result + (int) minimum.toMillis();
+ result = prime * result + (int) maximum.toMillis();
+
+ return result;
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/retry/strategy/FixedTime.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/retry/strategy/FixedTime.java
new file mode 100644
index 000000000..98d78a048
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/retry/strategy/FixedTime.java
@@ -0,0 +1,86 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.retry.strategy;
+
+import java.time.Clock;
+import java.time.Duration;
+import java.time.LocalDate;
+import java.time.LocalDateTime;
+import java.time.LocalTime;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.energidataservice.internal.retry.RetryStrategy;
+
+/**
+ * This implements a {@link RetryStrategy} for a fixed time.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class FixedTime implements RetryStrategy {
+
+ private final Clock clock;
+
+ private LocalTime localTime;
+ private double jitter = 0.0;
+
+ public FixedTime(LocalTime localTime, Clock clock) {
+ this.localTime = localTime;
+ this.clock = clock;
+ }
+
+ @Override
+ public Duration getDuration() {
+ LocalTime now = LocalTime.now(clock);
+ LocalDateTime localDateTime = LocalDateTime.of(LocalDate.now(clock), localTime);
+ if (now.isAfter(localTime)) {
+ localDateTime = localDateTime.plusDays(1);
+ }
+
+ Duration base = Duration.between(LocalDateTime.now(clock), localDateTime);
+ if (jitter == 0.0) {
+ return base;
+ }
+
+ long duration = base.toMillis();
+ double rand = Math.random();
+ duration += (long) (rand * jitter * 1000 * 60);
+
+ return Duration.ofMillis(duration);
+ }
+
+ public FixedTime withJitter(double jitter) {
+ this.jitter = jitter;
+ return this;
+ }
+
+ @Override
+ public boolean equals(@Nullable Object o) {
+ if (o == this) {
+ return true;
+ }
+ if (!(o instanceof FixedTime)) {
+ return false;
+ }
+ FixedTime other = (FixedTime) o;
+
+ return this.jitter == other.jitter && this.localTime.equals(other.localTime);
+ }
+
+ @Override
+ public final int hashCode() {
+ final int result = 1;
+ return result;
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/retry/strategy/Linear.java b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/retry/strategy/Linear.java
new file mode 100644
index 000000000..99bd57a46
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/java/org/openhab/binding/energidataservice/internal/retry/strategy/Linear.java
@@ -0,0 +1,96 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.retry.strategy;
+
+import java.time.Duration;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.energidataservice.internal.retry.RetryStrategy;
+
+/**
+ * This implements a {@link RetryStrategy} for linear retry with jitter.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class Linear implements RetryStrategy {
+
+ private double jitter = 0.0;
+ private Duration minimum = Duration.ofMillis(100);
+ private Duration maximum = Duration.ofHours(6);
+
+ public Linear() {
+ }
+
+ @Override
+ public Duration getDuration() {
+ long minimum = this.minimum.toMillis();
+ long maximum = this.maximum.toMillis();
+ long duration = minimum;
+ if (jitter != 0.0) {
+ double rand = Math.random();
+ if ((((int) Math.floor(rand * 10)) & 1) == 0) {
+ duration += (long) (rand * jitter * duration);
+ } else {
+ duration -= (long) (rand * jitter * duration);
+ }
+ }
+ if (duration < minimum) {
+ duration = minimum;
+ }
+ if (duration > maximum) {
+ duration = maximum;
+ }
+ return Duration.ofMillis(duration);
+ }
+
+ public Linear withJitter(double jitter) {
+ this.jitter = jitter;
+ return this;
+ }
+
+ public Linear withMinimum(Duration minimum) {
+ this.minimum = minimum;
+ return this;
+ }
+
+ public Linear withMaximum(Duration maximum) {
+ this.maximum = maximum;
+ return this;
+ }
+
+ @Override
+ public boolean equals(@Nullable Object o) {
+ if (o == this) {
+ return true;
+ }
+ if (!(o instanceof Linear)) {
+ return false;
+ }
+ Linear other = (Linear) o;
+
+ return this.jitter == other.jitter && this.minimum.equals(other.minimum) && this.maximum.equals(other.maximum);
+ }
+
+ @Override
+ public final int hashCode() {
+ final int prime = 31;
+ int result = 1;
+ result = prime * result + (int) jitter * 100;
+ result = prime * result + (int) minimum.toMillis();
+ result = prime * result + (int) maximum.toMillis();
+
+ return result;
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/resources/OH-INF/addon/addon.xml b/bundles/org.openhab.binding.energidataservice/src/main/resources/OH-INF/addon/addon.xml
new file mode 100644
index 000000000..70d43d22a
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/resources/OH-INF/addon/addon.xml
@@ -0,0 +1,11 @@
+
+
+
+ binding
+ Energi Data Service Binding
+ This is the binding for Energi Data Service providing open energy data from Energinet.
+ cloud
+ dk,no,se
+
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/resources/OH-INF/config/config.xml b/bundles/org.openhab.binding.energidataservice/src/main/resources/OH-INF/config/config.xml
new file mode 100644
index 000000000..77d3dfe3c
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/resources/OH-INF/config/config.xml
@@ -0,0 +1,94 @@
+
+
+
+
+
+
+ Price area for spot prices (same as bidding zone).
+ false
+
+
+
+
+
+
+
+ Currency code in which to obtain spot prices.
+ DKK
+
+
+
+
+
+
+
+ Global Location Number of the grid company.
+ false
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Global Location Number of Energinet.
+ true
+ 5790000432752
+
+
+
+
+
+
+ Comma-separated list of charge type codes.
+ true
+
+
+
+ Comma-separated list of notes.
+ true
+
+
+
+ Query start date parameter expressed as either YYYY-MM-DD or dynamically as one of StartOfDay,
+ StartOfMonth or StartOfYear.
+ false
+
+
+
+
+
+ true
+
+
+
+
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/resources/OH-INF/i18n/energidataservice.properties b/bundles/org.openhab.binding.energidataservice/src/main/resources/OH-INF/i18n/energidataservice.properties
new file mode 100644
index 000000000..8587e9c39
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/resources/OH-INF/i18n/energidataservice.properties
@@ -0,0 +1,118 @@
+# add-on
+
+addon.energidataservice.name = Energi Data Service Binding
+addon.energidataservice.description = This is the binding for Energi Data Service providing open energy data from Energinet.
+
+# thing types
+
+thing-type.energidataservice.service.label = Energi Data Service
+thing-type.energidataservice.service.description = This Thing represents the Energi Data Service API.
+
+# thing types config
+
+thing-type.config.energidataservice.service.currencyCode.label = Currency Code
+thing-type.config.energidataservice.service.currencyCode.description = Currency code in which to obtain spot prices.
+thing-type.config.energidataservice.service.currencyCode.option.DKK = Danish Krone
+thing-type.config.energidataservice.service.currencyCode.option.EUR = Euro
+thing-type.config.energidataservice.service.energinetGLN.label = Energinet GLN
+thing-type.config.energidataservice.service.energinetGLN.description = Global Location Number of Energinet.
+thing-type.config.energidataservice.service.gridCompanyGLN.label = Grid Company GLN
+thing-type.config.energidataservice.service.gridCompanyGLN.description = Global Location Number of the grid company.
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790000705184 = Cerius
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790000610099 = Dinel
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790002502699 = El-net Kongerslev
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790000836239 = Elektrus
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790001095277 = Elinord
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790001100520 = Elnet Midt
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790000392551 = FLOW Elnet
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790001090166 = Hammel Elforsyning Net
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790000610839 = Hurup Elværk Net
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790000682102 = Ikast El Net
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790000704842 = Konstant
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790001090111 = L-Net
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790001089023 = Midtfyns Elforsyning
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790001089030 = N1
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790000681075 = Netselskabet Elværk
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790001088231 = NKE-Elnet
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790000610877 = Nord Energi Net
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790000395620 = Nordvestjysk Elforsyning (NOE Net)
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790000705689 = Radius
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790000681327 = RAH
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790000836727 = Ravdex
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790001095444 = Sunds Net
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790000706419 = Tarm Elværk Net
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790000392261 = TREFOR El-net
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790000706686 = TREFOR El-net Øst
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790001088217 = Veksel
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790000610976 = Vores Elnet
+thing-type.config.energidataservice.service.gridCompanyGLN.option.5790001089375 = Zeanet
+thing-type.config.energidataservice.service.priceArea.label = Price Area
+thing-type.config.energidataservice.service.priceArea.description = Price area for spot prices (same as bidding zone).
+thing-type.config.energidataservice.service.priceArea.option.DK1 = West of the Great Belt
+thing-type.config.energidataservice.service.priceArea.option.DK2 = East of the Great Belt
+
+# channel group types
+
+channel-group-type.energidataservice.electricity.label = Electricity
+channel-group-type.energidataservice.electricity.description = Channels related to electricity
+channel-group-type.energidataservice.electricity.channel.electricity-tax.label = Electricity Tax
+channel-group-type.energidataservice.electricity.channel.electricity-tax.description = Current electricity tax in DKK per kWh.
+channel-group-type.energidataservice.electricity.channel.net-tariff.label = Net Tariff
+channel-group-type.energidataservice.electricity.channel.net-tariff.description = Current net tariff in DKK per kWh.
+channel-group-type.energidataservice.electricity.channel.spot-price.label = Spot Price
+channel-group-type.energidataservice.electricity.channel.spot-price.description = Current spot price in DKK or EUR per kWh.
+channel-group-type.energidataservice.electricity.channel.system-tariff.label = System Tariff
+channel-group-type.energidataservice.electricity.channel.system-tariff.description = Current system tariff in DKK per kWh.
+channel-group-type.energidataservice.electricity.channel.transmission-net-tariff.label = Transmission Net Tariff
+channel-group-type.energidataservice.electricity.channel.transmission-net-tariff.description = Current transmission net tariff in DKK per kWh.
+
+# channel types
+
+channel-type.energidataservice.datahub-price.label = Datahub Price
+channel-type.energidataservice.datahub-price.description = Datahub price.
+channel-type.energidataservice.hourly-prices.label = Hourly Prices
+channel-type.energidataservice.hourly-prices.description = JSON array with hourly prices from 12 hours ago and onward.
+channel-type.energidataservice.spot-price.label = Spot Price
+channel-type.energidataservice.spot-price.description = Spot price.
+
+# channel types config
+
+channel-type.config.energidataservice.datahub-price.chargeTypeCodes.label = Charge Type Code Filters
+channel-type.config.energidataservice.datahub-price.chargeTypeCodes.description = Comma-separated list of charge type codes.
+channel-type.config.energidataservice.datahub-price.notes.label = Note Filters
+channel-type.config.energidataservice.datahub-price.notes.description = Comma-separated list of notes.
+channel-type.config.energidataservice.datahub-price.start.label = Query Start Date
+channel-type.config.energidataservice.datahub-price.start.description = Query start date parameter expressed as either YYYY-MM-DD or dynamically as one of StartOfDay, StartOfMonth or StartOfYear.
+channel-type.config.energidataservice.datahub-price.start.option.StartOfDay = Start of day
+channel-type.config.energidataservice.datahub-price.start.option.StartOfMonth = Start of month
+channel-type.config.energidataservice.datahub-price.start.option.StartOfYear = Start of year
+
+# channel group types
+
+channel-group-type.energidataservice.electricity.channel.current-electricity-tax.label = Current Electricity Tax
+channel-group-type.energidataservice.electricity.channel.current-electricity-tax.description = Electricity Tax in DKK per kWh for current hour.
+channel-group-type.energidataservice.electricity.channel.current-net-tariff.label = Current Net Tariff
+channel-group-type.energidataservice.electricity.channel.current-net-tariff.description = Net tariff in DKK per kWh for current hour.
+channel-group-type.energidataservice.electricity.channel.current-spot-price.label = Current Spot Price
+channel-group-type.energidataservice.electricity.channel.current-spot-price.description = Spot price in DKK or EUR per kWh for current hour.
+channel-group-type.energidataservice.electricity.channel.current-system-tariff.label = Current System Tariff
+channel-group-type.energidataservice.electricity.channel.current-system-tariff.description = System tariff in DKK per kWh for current hour.
+channel-group-type.energidataservice.electricity.channel.current-transmission-net-tariff.label = Current Transmission Tariff
+channel-group-type.energidataservice.electricity.channel.current-transmission-net-tariff.description = Transmission Net Tariff in DKK per kWh for current hour.
+
+# thing status descriptions
+
+offline.conf-error.no-price-area = Price area must be set
+offline.conf-error.invalid-grid-company-gln = Invalid grid company GLN
+offline.conf-error.invalid-energinet-gln = Invalid Energinet GLN
+
+# actions
+
+action.calculate-cheapest-period.label = calculate cheapest period
+action.calculate-cheapest-period.description = calculate cheapest period for using power according to a supplied timetable (excl. VAT)
+action.calculate-price.label = calculate price
+action.calculate-price.description = calculate price for power consumption in period excl. VAT
+action.get-prices.label = get prices
+action.get-prices.description = get hourly prices excl. VAT
+action.get-prices.priceElements.label = price elements
+action.get-prices.priceElements.description = comma-separated list of price elements to include in sums
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/resources/OH-INF/thing/channel-groups.xml b/bundles/org.openhab.binding.energidataservice/src/main/resources/OH-INF/thing/channel-groups.xml
new file mode 100644
index 000000000..2a2201826
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/resources/OH-INF/thing/channel-groups.xml
@@ -0,0 +1,35 @@
+
+
+
+
+
+ Channels related to electricity
+
+
+
+ Current spot price in DKK or EUR per kWh.
+
+
+
+ Current net tariff in DKK per kWh.
+
+
+
+ Current system tariff in DKK per kWh.
+
+
+
+ Current electricity tax in DKK per kWh.
+
+
+
+ Current transmission net tariff in DKK per kWh.
+
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/resources/OH-INF/thing/channel-types.xml b/bundles/org.openhab.binding.energidataservice/src/main/resources/OH-INF/thing/channel-types.xml
new file mode 100644
index 000000000..12d0273c7
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/resources/OH-INF/thing/channel-types.xml
@@ -0,0 +1,32 @@
+
+
+
+
+ Number
+
+ Spot price.
+ Price
+
+
+
+
+ Number
+
+ Datahub price.
+ Price
+
+
+
+
+
+ String
+
+ JSON array with hourly prices from 12 hours ago and onward.
+ Price
+
+
+
+
diff --git a/bundles/org.openhab.binding.energidataservice/src/main/resources/OH-INF/thing/thing-service.xml b/bundles/org.openhab.binding.energidataservice/src/main/resources/OH-INF/thing/thing-service.xml
new file mode 100644
index 000000000..c00115a67
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/main/resources/OH-INF/thing/thing-service.xml
@@ -0,0 +1,19 @@
+
+
+
+
+
+
+ This Thing represents the Energi Data Service API.
+
+
+
+
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/CacheManagerTest.java b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/CacheManagerTest.java
new file mode 100644
index 000000000..a90165211
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/CacheManagerTest.java
@@ -0,0 +1,126 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal;
+
+import static org.hamcrest.CoreMatchers.*;
+import static org.hamcrest.MatcherAssert.assertThat;
+
+import java.math.BigDecimal;
+import java.time.Clock;
+import java.time.Duration;
+import java.time.Instant;
+import java.time.ZoneId;
+import java.time.temporal.ChronoUnit;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.junit.jupiter.api.Test;
+import org.junit.jupiter.api.extension.ExtendWith;
+import org.mockito.junit.jupiter.MockitoExtension;
+import org.openhab.binding.energidataservice.internal.api.dto.ElspotpriceRecord;
+
+/**
+ * Tests for {@link CacheManager}.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+@ExtendWith(MockitoExtension.class)
+public class CacheManagerTest {
+
+ @Test
+ void areSpotPricesFullyCachedToday() {
+ Instant now = Instant.parse("2023-02-07T08:38:47Z");
+ Instant first = Instant.parse("2023-02-06T08:00:00Z");
+ Instant last = Instant.parse("2023-02-07T22:00:00Z");
+ Clock clock = Clock.fixed(now, EnergiDataServiceBindingConstants.NORD_POOL_TIMEZONE);
+ CacheManager cacheManager = new CacheManager(clock);
+ populateWithSpotPrices(cacheManager, first, last);
+ assertThat(cacheManager.areSpotPricesFullyCached(), is(true));
+ }
+
+ @Test
+ void areSpotPricesFullyCachedTodayMissingAtStart() {
+ Instant now = Instant.parse("2023-02-07T08:38:47Z");
+ Instant first = Instant.parse("2023-02-06T21:00:00Z");
+ Instant last = Instant.parse("2023-02-07T22:00:00Z");
+ Clock clock = Clock.fixed(now, EnergiDataServiceBindingConstants.NORD_POOL_TIMEZONE);
+ CacheManager cacheManager = new CacheManager(clock);
+ populateWithSpotPrices(cacheManager, first, last);
+ assertThat(cacheManager.areSpotPricesFullyCached(), is(false));
+ }
+
+ @Test
+ void areSpotPricesFullyCachedTodayMissingAtEnd() {
+ Instant now = Instant.parse("2023-02-07T08:38:47Z");
+ Instant first = Instant.parse("2023-02-06T20:00:00Z");
+ Instant last = Instant.parse("2023-02-07T21:00:00Z");
+ Clock clock = Clock.fixed(now, EnergiDataServiceBindingConstants.NORD_POOL_TIMEZONE);
+ CacheManager cacheManager = new CacheManager(clock);
+ populateWithSpotPrices(cacheManager, first, last);
+ assertThat(cacheManager.areSpotPricesFullyCached(), is(false));
+ }
+
+ @Test
+ void areSpotPricesFullyCachedTodayOtherTimezoneIsIgnored() {
+ Instant now = Instant.parse("2023-02-07T08:38:47Z");
+ Instant first = Instant.parse("2023-02-06T08:00:00Z");
+ Instant last = Instant.parse("2023-02-07T22:00:00Z");
+ Clock clock = Clock.fixed(now, ZoneId.of("Asia/Tokyo"));
+ CacheManager cacheManager = new CacheManager(clock);
+ populateWithSpotPrices(cacheManager, first, last);
+ assertThat(cacheManager.areSpotPricesFullyCached(), is(true));
+ }
+
+ @Test
+ void areSpotPricesFullyCachedTomorrow() {
+ Instant now = Instant.parse("2023-02-07T12:00:00Z");
+ Instant first = Instant.parse("2023-02-06T12:00:00Z");
+ Instant last = Instant.parse("2023-02-08T22:00:00Z");
+ Clock clock = Clock.fixed(now, EnergiDataServiceBindingConstants.NORD_POOL_TIMEZONE);
+ CacheManager cacheManager = new CacheManager(clock);
+ populateWithSpotPrices(cacheManager, first, last);
+ assertThat(cacheManager.areSpotPricesFullyCached(), is(true));
+ }
+
+ @Test
+ void areHistoricSpotPricesCached() {
+ Instant now = Instant.parse("2023-02-07T08:38:47Z");
+ Instant first = Instant.parse("2023-02-06T08:00:00Z");
+ Instant last = Instant.parse("2023-02-07T07:00:00Z");
+ Clock clock = Clock.fixed(now, EnergiDataServiceBindingConstants.NORD_POOL_TIMEZONE);
+ CacheManager cacheManager = new CacheManager(clock);
+ populateWithSpotPrices(cacheManager, first, last);
+ assertThat(cacheManager.areHistoricSpotPricesCached(), is(true));
+ }
+
+ @Test
+ void areHistoricSpotPricesCachedFirstHourMissing() {
+ Instant now = Instant.parse("2023-02-07T08:38:47Z");
+ Instant first = Instant.parse("2023-02-06T21:00:00Z");
+ Instant last = Instant.parse("2023-02-07T08:00:00Z");
+ Clock clock = Clock.fixed(now, EnergiDataServiceBindingConstants.NORD_POOL_TIMEZONE);
+ CacheManager cacheManager = new CacheManager(clock);
+ populateWithSpotPrices(cacheManager, first, last);
+ assertThat(cacheManager.areHistoricSpotPricesCached(), is(false));
+ }
+
+ private void populateWithSpotPrices(CacheManager cacheManager, Instant first, Instant last) {
+ int size = (int) Duration.between(first, last).getSeconds() / 60 / 60 + 1;
+ ElspotpriceRecord[] records = new ElspotpriceRecord[size];
+ int i = 0;
+ for (Instant hourStart = first; !hourStart.isAfter(last); hourStart = hourStart.plus(1, ChronoUnit.HOURS)) {
+ records[i++] = new ElspotpriceRecord(hourStart, BigDecimal.ONE, BigDecimal.ZERO);
+ }
+ cacheManager.putSpotPrices(records, EnergiDataServiceBindingConstants.CURRENCY_DKK);
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/PriceListParserTest.java b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/PriceListParserTest.java
new file mode 100644
index 000000000..da0b31a78
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/PriceListParserTest.java
@@ -0,0 +1,194 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal;
+
+import static org.hamcrest.MatcherAssert.assertThat;
+import static org.hamcrest.Matchers.equalTo;
+import static org.hamcrest.Matchers.is;
+
+import java.io.IOException;
+import java.io.InputStream;
+import java.math.BigDecimal;
+import java.nio.charset.StandardCharsets;
+import java.time.Clock;
+import java.time.Instant;
+import java.time.LocalDateTime;
+import java.time.temporal.ChronoUnit;
+import java.util.Arrays;
+import java.util.Map;
+import java.util.Objects;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.junit.jupiter.api.Test;
+import org.junit.jupiter.api.extension.ExtendWith;
+import org.mockito.junit.jupiter.MockitoExtension;
+import org.openhab.binding.energidataservice.internal.api.dto.DatahubPricelistRecords;
+import org.openhab.binding.energidataservice.internal.api.serialization.LocalDateTimeDeserializer;
+
+import com.google.gson.Gson;
+import com.google.gson.GsonBuilder;
+
+/**
+ * Tests for {@link PriceListParser}.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+@ExtendWith(MockitoExtension.class)
+public class PriceListParserTest {
+
+ private Gson gson = new GsonBuilder().registerTypeAdapter(LocalDateTime.class, new LocalDateTimeDeserializer())
+ .create();
+
+ private T getObjectFromJson(String filename, Class clazz) throws IOException {
+ try (InputStream inputStream = PriceListParserTest.class.getResourceAsStream(filename)) {
+ if (inputStream == null) {
+ throw new IOException("Input stream is null");
+ }
+ byte[] bytes = inputStream.readAllBytes();
+ if (bytes == null) {
+ throw new IOException("Resulting byte-array empty");
+ }
+ String json = new String(bytes, StandardCharsets.UTF_8);
+ return Objects.requireNonNull(gson.fromJson(json, clazz));
+ }
+ }
+
+ @Test
+ void toHourlyNoChanges() throws IOException {
+ PriceListParser priceListParser = new PriceListParser(
+ Clock.fixed(Instant.parse("2023-01-23T12:00:00Z"), EnergiDataServiceBindingConstants.DATAHUB_TIMEZONE));
+ DatahubPricelistRecords records = getObjectFromJson("DatahubPricelistN1.json", DatahubPricelistRecords.class);
+ Map tariffMap = priceListParser.toHourly(Arrays.stream(records.records()).toList());
+
+ assertThat(tariffMap.size(), is(60));
+ assertThat(tariffMap.get(Instant.parse("2023-01-23T15:00:00Z")), is(equalTo(new BigDecimal("0.432225"))));
+ assertThat(tariffMap.get(Instant.parse("2023-01-23T16:00:00Z")), is(equalTo(new BigDecimal("1.05619"))));
+ assertThat(tariffMap.get(Instant.parse("2023-01-24T15:00:00Z")), is(equalTo(new BigDecimal("0.432225"))));
+ assertThat(tariffMap.get(Instant.parse("2023-01-24T16:00:00Z")), is(equalTo(new BigDecimal("1.05619"))));
+ }
+
+ @Test
+ void toHourlyNewTariffTomorrowWhenSummertime() throws IOException {
+ PriceListParser priceListParser = new PriceListParser(
+ Clock.fixed(Instant.parse("2023-03-31T12:00:00Z"), EnergiDataServiceBindingConstants.DATAHUB_TIMEZONE));
+ DatahubPricelistRecords records = getObjectFromJson("DatahubPricelistN1.json", DatahubPricelistRecords.class);
+ Map tariffMap = priceListParser.toHourly(Arrays.stream(records.records()).toList());
+
+ assertThat(tariffMap.size(), is(60));
+ assertThat(tariffMap.get(Instant.parse("2023-03-31T14:00:00Z")), is(equalTo(new BigDecimal("0.432225"))));
+ assertThat(tariffMap.get(Instant.parse("2023-03-31T15:00:00Z")), is(equalTo(new BigDecimal("1.05619"))));
+ assertThat(tariffMap.get(Instant.parse("2023-04-01T14:00:00Z")), is(equalTo(new BigDecimal("0.432225"))));
+ assertThat(tariffMap.get(Instant.parse("2023-04-01T15:00:00Z")), is(equalTo(new BigDecimal("0.432225"))));
+ }
+
+ @Test
+ void toHourlyNewTariffAtMidnight() throws IOException {
+ PriceListParser priceListParser = new PriceListParser(
+ Clock.fixed(Instant.parse("2022-12-31T12:00:00Z"), EnergiDataServiceBindingConstants.DATAHUB_TIMEZONE));
+ DatahubPricelistRecords records = getObjectFromJson("DatahubPricelistN1.json", DatahubPricelistRecords.class);
+ Map tariffMap = priceListParser.toHourly(Arrays.stream(records.records()).toList(), "CD");
+
+ assertThat(tariffMap.size(), is(60));
+ assertThat(tariffMap.get(Instant.parse("2022-12-31T22:00:00Z")), is(equalTo(new BigDecimal("0.407717"))));
+ assertThat(tariffMap.get(Instant.parse("2022-12-31T23:00:00Z")), is(equalTo(new BigDecimal("0.432225"))));
+ assertThat(tariffMap.get(Instant.parse("2023-01-01T00:00:00Z")), is(equalTo(new BigDecimal("0.432225"))));
+ }
+
+ @Test
+ void toHourlyDiscount() throws IOException {
+ PriceListParser priceListParser = new PriceListParser(
+ Clock.fixed(Instant.parse("2022-12-31T12:00:00Z"), EnergiDataServiceBindingConstants.DATAHUB_TIMEZONE));
+ DatahubPricelistRecords records = getObjectFromJson("DatahubPricelistN1.json", DatahubPricelistRecords.class);
+ Map tariffMap = priceListParser.toHourly(Arrays.stream(records.records()).toList(),
+ "CD R");
+
+ assertThat(tariffMap.size(), is(60));
+ assertThat(tariffMap.get(Instant.parse("2022-12-31T22:00:00Z")), is(equalTo(new BigDecimal("-0.407717"))));
+ assertThat(tariffMap.get(Instant.parse("2022-12-31T23:00:00Z")), is(equalTo(new BigDecimal("0.0"))));
+ assertThat(tariffMap.get(Instant.parse("2023-01-01T00:00:00Z")), is(equalTo(new BigDecimal("0.0"))));
+ }
+
+ @Test
+ void toHourlyTariffAndDiscountIsSum() throws IOException {
+ PriceListParser priceListParser = new PriceListParser(
+ Clock.fixed(Instant.parse("2022-11-30T15:00:00Z"), EnergiDataServiceBindingConstants.DATAHUB_TIMEZONE));
+ DatahubPricelistRecords records = getObjectFromJson("DatahubPricelistN1.json", DatahubPricelistRecords.class);
+ Map tariffMap = priceListParser.toHourly(Arrays.stream(records.records()).toList());
+
+ assertThat(tariffMap.size(), is(57));
+ assertThat(tariffMap.get(Instant.parse("2022-11-30T15:00:00Z")), is(equalTo(new BigDecimal("0.387517"))));
+ assertThat(tariffMap.get(Instant.parse("2022-11-30T16:00:00Z")), is(equalTo(new BigDecimal("0.973404"))));
+ }
+
+ @Test
+ void toHourlyTariffAndDiscountIsFree() throws IOException {
+ PriceListParser priceListParser = new PriceListParser(
+ Clock.fixed(Instant.parse("2022-12-31T12:00:00Z"), EnergiDataServiceBindingConstants.DATAHUB_TIMEZONE));
+ DatahubPricelistRecords records = getObjectFromJson("DatahubPricelistN1.json", DatahubPricelistRecords.class);
+ Map tariffMap = priceListParser.toHourly(Arrays.stream(records.records()).toList());
+
+ assertThat(tariffMap.size(), is(60));
+ assertThat(tariffMap.get(Instant.parse("2022-12-31T16:00:00Z")), is(equalTo(new BigDecimal("0.000000"))));
+ assertThat(tariffMap.get(Instant.parse("2022-12-31T22:00:00Z")), is(equalTo(new BigDecimal("0.000000"))));
+ assertThat(tariffMap.get(Instant.parse("2022-12-31T23:00:00Z")), is(equalTo(new BigDecimal("0.432225"))));
+ assertThat(tariffMap.get(Instant.parse("2023-01-01T00:00:00Z")), is(equalTo(new BigDecimal("0.432225"))));
+ }
+
+ @Test
+ void toHourlyFixedTariff() throws IOException {
+ PriceListParser priceListParser = new PriceListParser(
+ Clock.fixed(Instant.parse("2022-12-31T23:00:00Z"), EnergiDataServiceBindingConstants.DATAHUB_TIMEZONE));
+ DatahubPricelistRecords records = getObjectFromJson("DatahubPricelistNordEnergi.json",
+ DatahubPricelistRecords.class);
+ Map tariffMap = priceListParser.toHourly(Arrays.stream(records.records()).toList());
+
+ assertThat(tariffMap.size(), is(25)); // No records in dataset before 2023-01-01
+ for (Instant i = Instant.parse("2022-12-31T23:00:00Z"); i
+ .isBefore(Instant.parse("2023-01-02T00:00:00Z")); i = i.plus(1, ChronoUnit.HOURS)) {
+ assertThat(tariffMap.get(i), is(equalTo(new BigDecimal("0.245"))));
+ }
+ }
+
+ @Test
+ void toHourlyDailyTariffs() throws IOException {
+ PriceListParser priceListParser = new PriceListParser(
+ Clock.fixed(Instant.parse("2023-01-28T04:00:00Z"), EnergiDataServiceBindingConstants.DATAHUB_TIMEZONE));
+ DatahubPricelistRecords records = getObjectFromJson("DatahubPricelistTrefor.json",
+ DatahubPricelistRecords.class);
+ Map tariffMap = priceListParser.toHourly(Arrays.stream(records.records()).toList());
+
+ assertThat(tariffMap.size(), is(68));
+ assertThat(tariffMap.get(Instant.parse("2023-01-28T04:00:00Z")), is(equalTo(new BigDecimal("0.2581"))));
+ assertThat(tariffMap.get(Instant.parse("2023-01-28T05:00:00Z")), is(equalTo(new BigDecimal("0.7742"))));
+ assertThat(tariffMap.get(Instant.parse("2023-01-28T16:00:00Z")), is(equalTo(new BigDecimal("2.3227"))));
+ assertThat(tariffMap.get(Instant.parse("2023-01-28T20:00:00Z")), is(equalTo(new BigDecimal("0.7742"))));
+ assertThat(tariffMap.get(Instant.parse("2023-01-28T23:00:00Z")), is(equalTo(new BigDecimal("0.2581"))));
+ assertThat(tariffMap.get(Instant.parse("2023-01-29T05:00:00Z")), is(equalTo(new BigDecimal("0.7742"))));
+ assertThat(tariffMap.get(Instant.parse("2023-01-29T16:00:00Z")), is(equalTo(new BigDecimal("2.3227"))));
+ assertThat(tariffMap.get(Instant.parse("2023-01-29T20:00:00Z")), is(equalTo(new BigDecimal("0.7742"))));
+ }
+
+ @Test
+ void toHourlySystemTariff() throws IOException {
+ PriceListParser priceListParser = new PriceListParser(
+ Clock.fixed(Instant.parse("2023-06-30T21:00:00Z"), EnergiDataServiceBindingConstants.DATAHUB_TIMEZONE));
+ DatahubPricelistRecords records = getObjectFromJson("DatahubPricelistElectricityTax.json",
+ DatahubPricelistRecords.class);
+ Map tariffMap = priceListParser.toHourly(Arrays.stream(records.records()).toList());
+
+ assertThat(tariffMap.size(), is(51));
+ assertThat(tariffMap.get(Instant.parse("2023-06-30T21:00:00Z")), is(equalTo(new BigDecimal("0.008"))));
+ assertThat(tariffMap.get(Instant.parse("2023-06-30T22:00:00Z")), is(equalTo(new BigDecimal("0.697"))));
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/action/EnergiDataServiceActionsTest.java b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/action/EnergiDataServiceActionsTest.java
new file mode 100644
index 000000000..3170973a0
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/action/EnergiDataServiceActionsTest.java
@@ -0,0 +1,403 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.action;
+
+import static org.hamcrest.CoreMatchers.*;
+import static org.hamcrest.MatcherAssert.assertThat;
+import static org.mockito.Mockito.when;
+
+import java.io.IOException;
+import java.io.InputStream;
+import java.math.BigDecimal;
+import java.nio.charset.StandardCharsets;
+import java.time.Clock;
+import java.time.Duration;
+import java.time.Instant;
+import java.time.LocalDateTime;
+import java.util.Arrays;
+import java.util.List;
+import java.util.Map;
+import java.util.Objects;
+import java.util.stream.Collectors;
+
+import javax.measure.quantity.Power;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.junit.jupiter.api.BeforeEach;
+import org.junit.jupiter.api.Test;
+import org.junit.jupiter.api.extension.ExtendWith;
+import org.mockito.Mock;
+import org.mockito.junit.jupiter.MockitoExtension;
+import org.mockito.junit.jupiter.MockitoSettings;
+import org.mockito.quality.Strictness;
+import org.openhab.binding.energidataservice.internal.EnergiDataServiceBindingConstants;
+import org.openhab.binding.energidataservice.internal.PriceListParser;
+import org.openhab.binding.energidataservice.internal.api.dto.DatahubPricelistRecords;
+import org.openhab.binding.energidataservice.internal.api.serialization.InstantDeserializer;
+import org.openhab.binding.energidataservice.internal.api.serialization.LocalDateTimeDeserializer;
+import org.openhab.binding.energidataservice.internal.handler.EnergiDataServiceHandler;
+import org.openhab.core.library.types.QuantityType;
+import org.openhab.core.library.unit.Units;
+import org.slf4j.LoggerFactory;
+
+import com.google.gson.Gson;
+import com.google.gson.GsonBuilder;
+
+import ch.qos.logback.classic.Level;
+import ch.qos.logback.classic.Logger;
+
+/**
+ * Tests for {@link EnergiDataServiceActions}.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+@ExtendWith(MockitoExtension.class)
+@MockitoSettings(strictness = Strictness.LENIENT)
+public class EnergiDataServiceActionsTest {
+
+ private @NonNullByDefault({}) @Mock EnergiDataServiceHandler handler;
+ private EnergiDataServiceActions actions = new EnergiDataServiceActions();
+
+ private Gson gson = new GsonBuilder().registerTypeAdapter(Instant.class, new InstantDeserializer())
+ .registerTypeAdapter(LocalDateTime.class, new LocalDateTimeDeserializer()).create();
+
+ private record SpotPrice(Instant hourStart, BigDecimal spotPrice) {
+ }
+
+ private T getObjectFromJson(String filename, Class clazz) throws IOException {
+ try (InputStream inputStream = EnergiDataServiceActionsTest.class.getResourceAsStream(filename)) {
+ if (inputStream == null) {
+ throw new IOException("Input stream is null");
+ }
+ byte[] bytes = inputStream.readAllBytes();
+ if (bytes == null) {
+ throw new IOException("Resulting byte-array empty");
+ }
+ String json = new String(bytes, StandardCharsets.UTF_8);
+ return Objects.requireNonNull(gson.fromJson(json, clazz));
+ }
+ }
+
+ @BeforeEach
+ void setUp() {
+ final Logger logger = (Logger) LoggerFactory.getLogger(EnergiDataServiceActions.class);
+ logger.setLevel(Level.OFF);
+
+ actions = new EnergiDataServiceActions();
+ }
+
+ @Test
+ void getPricesSpotPrice() throws IOException {
+ mockCommonDatasets(actions);
+
+ Map actual = actions.getPrices("SpotPrice");
+ assertThat(actual.size(), is(35));
+ assertThat(actual.get(Instant.parse("2023-02-04T12:00:00Z")), is(equalTo(new BigDecimal("0.992840027"))));
+ assertThat(actual.get(Instant.parse("2023-02-04T16:00:00Z")), is(equalTo(new BigDecimal("1.267680054"))));
+ }
+
+ @Test
+ void getPricesNetTariff() throws IOException {
+ mockCommonDatasets(actions);
+
+ Map actual = actions.getPrices("NetTariff");
+ assertThat(actual.size(), is(60));
+ assertThat(actual.get(Instant.parse("2023-02-04T12:00:00Z")), is(equalTo(new BigDecimal("0.432225"))));
+ assertThat(actual.get(Instant.parse("2023-02-04T16:00:00Z")), is(equalTo(new BigDecimal("1.05619"))));
+ }
+
+ @Test
+ void getPricesSystemTariff() throws IOException {
+ mockCommonDatasets(actions);
+
+ Map actual = actions.getPrices("SystemTariff");
+ assertThat(actual.size(), is(60));
+ assertThat(actual.get(Instant.parse("2023-02-04T12:00:00Z")), is(equalTo(new BigDecimal("0.054"))));
+ assertThat(actual.get(Instant.parse("2023-02-04T16:00:00Z")), is(equalTo(new BigDecimal("0.054"))));
+ }
+
+ @Test
+ void getPricesElectricityTax() throws IOException {
+ mockCommonDatasets(actions);
+
+ Map actual = actions.getPrices("ElectricityTax");
+ assertThat(actual.size(), is(60));
+ assertThat(actual.get(Instant.parse("2023-02-04T12:00:00Z")), is(equalTo(new BigDecimal("0.008"))));
+ assertThat(actual.get(Instant.parse("2023-02-04T16:00:00Z")), is(equalTo(new BigDecimal("0.008"))));
+ }
+
+ @Test
+ void getPricesTransmissionNetTariff() throws IOException {
+ mockCommonDatasets(actions);
+
+ Map actual = actions.getPrices("TransmissionNetTariff");
+ assertThat(actual.size(), is(60));
+ assertThat(actual.get(Instant.parse("2023-02-04T12:00:00Z")), is(equalTo(new BigDecimal("0.058"))));
+ assertThat(actual.get(Instant.parse("2023-02-04T16:00:00Z")), is(equalTo(new BigDecimal("0.058"))));
+ }
+
+ @Test
+ void getPricesSpotPriceNetTariff() throws IOException {
+ mockCommonDatasets(actions);
+
+ Map actual = actions.getPrices("SpotPrice,NetTariff");
+ assertThat(actual.size(), is(35));
+ assertThat(actual.get(Instant.parse("2023-02-04T12:00:00Z")), is(equalTo(new BigDecimal("1.425065027"))));
+ assertThat(actual.get(Instant.parse("2023-02-04T16:00:00Z")), is(equalTo(new BigDecimal("2.323870054"))));
+ }
+
+ @Test
+ void getPricesSpotPriceNetTariffElectricityTax() throws IOException {
+ mockCommonDatasets(actions);
+
+ Map actual = actions.getPrices("SpotPrice,NetTariff,ElectricityTax");
+ assertThat(actual.size(), is(35));
+ assertThat(actual.get(Instant.parse("2023-02-04T12:00:00Z")), is(equalTo(new BigDecimal("1.433065027"))));
+ assertThat(actual.get(Instant.parse("2023-02-04T16:00:00Z")), is(equalTo(new BigDecimal("2.331870054"))));
+ }
+
+ @Test
+ void getPricesTotal() throws IOException {
+ mockCommonDatasets(actions);
+
+ Map actual = actions.getPrices();
+ assertThat(actual.size(), is(35));
+ assertThat(actual.get(Instant.parse("2023-02-04T12:00:00Z")), is(equalTo(new BigDecimal("1.545065027"))));
+ assertThat(actual.get(Instant.parse("2023-02-04T16:00:00Z")), is(equalTo(new BigDecimal("2.443870054"))));
+ }
+
+ @Test
+ void getPricesTotalAllElements() throws IOException {
+ mockCommonDatasets(actions);
+
+ Map actual = actions
+ .getPrices("spotprice,nettariff,systemtariff,electricitytax,transmissionnettariff");
+ assertThat(actual.size(), is(35));
+ assertThat(actual.get(Instant.parse("2023-02-04T12:00:00Z")), is(equalTo(new BigDecimal("1.545065027"))));
+ assertThat(actual.get(Instant.parse("2023-02-04T15:00:00Z")), is(equalTo(new BigDecimal("1.708765039"))));
+ assertThat(actual.get(Instant.parse("2023-02-04T16:00:00Z")), is(equalTo(new BigDecimal("2.443870054"))));
+ }
+
+ @Test
+ void getPricesInvalidPriceElement() throws IOException {
+ mockCommonDatasets(actions);
+
+ Map actual = actions.getPrices("spotprice,nettarif");
+ assertThat(actual.size(), is(0));
+ }
+
+ @Test
+ void getPricesMixedCurrencies() throws IOException {
+ mockCommonDatasets(actions);
+ when(handler.getCurrency()).thenReturn(EnergiDataServiceBindingConstants.CURRENCY_EUR);
+
+ Map actual = actions.getPrices("spotprice,nettariff");
+ assertThat(actual.size(), is(0));
+ }
+
+ /**
+ * Calculate price in period 15:30-16:30 (UTC) with consumption 150 W and the following total prices:
+ * 15:00:00: 1.708765039
+ * 16:00:00: 2.443870054
+ *
+ * Result = (1.708765039 / 2) + (2.443870054 / 2) * 0.150
+ *
+ * @throws IOException
+ */
+ @Test
+ void calculatePriceSimple() throws IOException {
+ mockCommonDatasets(actions);
+
+ BigDecimal actual = actions.calculatePrice(Instant.parse("2023-02-04T15:30:00Z"),
+ Instant.parse("2023-02-04T16:30:00Z"), new QuantityType<>(150, Units.WATT));
+ assertThat(actual, is(equalTo(new BigDecimal("0.311447631975000000")))); // 0.3114476319750
+ }
+
+ /**
+ * Calculate price in period 15:00-17:00 (UTC) with consumption 1000 W and the following total prices:
+ * 15:00:00: 1.708765039
+ * 16:00:00: 2.443870054
+ *
+ * Result = 1.708765039 + 2.443870054
+ *
+ * @throws IOException
+ */
+ @Test
+ void calculatePriceFullHours() throws IOException {
+ mockCommonDatasets(actions);
+
+ BigDecimal actual = actions.calculatePrice(Instant.parse("2023-02-04T15:00:00Z"),
+ Instant.parse("2023-02-04T17:00:00Z"), new QuantityType<>(1, Units.KILOVAR));
+ assertThat(actual, is(equalTo(new BigDecimal("4.152635093000000000")))); // 4.152635093
+ }
+
+ @Test
+ void calculatePriceOutOfRangeStart() throws IOException {
+ mockCommonDatasets(actions);
+
+ BigDecimal actual = actions.calculatePrice(Instant.parse("2023-02-03T23:59:00Z"),
+ Instant.parse("2023-02-04T12:30:00Z"), new QuantityType<>(1000, Units.WATT));
+ assertThat(actual, is(equalTo(BigDecimal.ZERO)));
+ }
+
+ @Test
+ void calculatePriceOutOfRangeEnd() throws IOException {
+ mockCommonDatasets(actions);
+
+ BigDecimal actual = actions.calculatePrice(Instant.parse("2023-02-05T22:00:00Z"),
+ Instant.parse("2023-02-05T23:01:00Z"), new QuantityType<>(1000, Units.WATT));
+ assertThat(actual, is(equalTo(BigDecimal.ZERO)));
+ }
+
+ /**
+ * Miele G 6895 SCVi XXL K2O dishwasher, program ECO.
+ *
+ * @throws IOException
+ */
+ @Test
+ void calculateCheapestPeriodWithPowerDishwasher() throws IOException {
+ mockCommonDatasets(actions, "SpotPrices20230205.json");
+
+ List durations = List.of(Duration.ofMinutes(37), Duration.ofMinutes(8), Duration.ofMinutes(4),
+ Duration.ofMinutes(2), Duration.ofMinutes(4), Duration.ofMinutes(36), Duration.ofMinutes(41),
+ Duration.ofMinutes(104));
+ List> consumptions = List.of(QuantityType.valueOf(162.162162, Units.WATT),
+ QuantityType.valueOf(750, Units.WATT), QuantityType.valueOf(1500, Units.WATT),
+ QuantityType.valueOf(3000, Units.WATT), QuantityType.valueOf(1500, Units.WATT),
+ QuantityType.valueOf(166.666666, Units.WATT), QuantityType.valueOf(146.341463, Units.WATT),
+ QuantityType.valueOf(0, Units.WATT));
+ Map actual = actions.calculateCheapestPeriod(Instant.parse("2023-02-05T16:00:00Z"),
+ Instant.parse("2023-02-06T06:00:00Z"), durations, consumptions);
+ assertThat(actual.get("LowestPrice"), is(equalTo(new BigDecimal("1.024218147103792520"))));
+ assertThat(actual.get("CheapestStart"), is(equalTo(Instant.parse("2023-02-05T19:23:00Z"))));
+ assertThat(actual.get("HighestPrice"), is(equalTo(new BigDecimal("1.530671034828983196"))));
+ assertThat(actual.get("MostExpensiveStart"), is(equalTo(Instant.parse("2023-02-05T16:00:00Z"))));
+ }
+
+ @Test
+ void calculateCheapestPeriodWithPowerOutOfRange() throws IOException {
+ mockCommonDatasets(actions);
+
+ List durations = List.of(Duration.ofMinutes(61));
+ List> consumptions = List.of(QuantityType.valueOf(1000, Units.WATT));
+ Map actual = actions.calculateCheapestPeriod(Instant.parse("2023-02-04T12:00:00Z"),
+ Instant.parse("2023-02-06T00:01:00Z"), durations, consumptions);
+ assertThat(actual.size(), is(equalTo(0)));
+ }
+
+ /**
+ * Miele G 6895 SCVi XXL K2O dishwasher, program ECO.
+ *
+ * @throws IOException
+ */
+ @Test
+ void calculateCheapestPeriodWithEnergyDishwasher() throws IOException {
+ mockCommonDatasets(actions, "SpotPrices20230205.json");
+
+ List durations = List.of(Duration.ofMinutes(37), Duration.ofMinutes(8), Duration.ofMinutes(4),
+ Duration.ofMinutes(2), Duration.ofMinutes(4), Duration.ofMinutes(36), Duration.ofMinutes(41));
+ Map actual = actions.calculateCheapestPeriod(Instant.parse("2023-02-05T16:00:00Z"),
+ Instant.parse("2023-02-06T06:00:00Z"), Duration.ofMinutes(236), durations,
+ QuantityType.valueOf(0.1, Units.KILOWATT_HOUR));
+ assertThat(actual.get("LowestPrice"), is(equalTo(new BigDecimal("1.024218147103792520"))));
+ assertThat(actual.get("CheapestStart"), is(equalTo(Instant.parse("2023-02-05T19:23:00Z"))));
+ assertThat(actual.get("HighestPrice"), is(equalTo(new BigDecimal("1.530671034828983196"))));
+ assertThat(actual.get("MostExpensiveStart"), is(equalTo(Instant.parse("2023-02-05T16:00:00Z"))));
+ }
+
+ @Test
+ void calculateCheapestPeriodWithEnergyTotalDurationIsExactSum() throws IOException {
+ mockCommonDatasets(actions, "SpotPrices20230205.json");
+
+ List durations = List.of(Duration.ofMinutes(60), Duration.ofMinutes(60));
+ Map actual = actions.calculateCheapestPeriod(Instant.parse("2023-02-05T16:00:00Z"),
+ Instant.parse("2023-02-06T06:00:00Z"), Duration.ofMinutes(120), durations,
+ QuantityType.valueOf(100, Units.WATT_HOUR));
+ assertThat(actual.get("LowestPrice"), is(equalTo(new BigDecimal("0.293540001200000000"))));
+ assertThat(actual.get("CheapestStart"), is(equalTo(Instant.parse("2023-02-05T19:00:00Z"))));
+ }
+
+ @Test
+ void calculateCheapestPeriodWithEnergyTotalDurationInvalid() throws IOException {
+ mockCommonDatasets(actions, "SpotPrices20230205.json");
+
+ List durations = List.of(Duration.ofMinutes(60), Duration.ofMinutes(60));
+ Map actual = actions.calculateCheapestPeriod(Instant.parse("2023-02-05T16:00:00Z"),
+ Instant.parse("2023-02-06T06:00:00Z"), Duration.ofMinutes(119), durations,
+ QuantityType.valueOf(0.1, Units.KILOWATT_HOUR));
+ assertThat(actual.size(), is(equalTo(0)));
+ }
+
+ /**
+ * Like {@link #calculateCheapestPeriodWithEnergyDishwasher} but with unknown consumption/timetable map.
+ *
+ * @throws IOException
+ */
+ @Test
+ void calculateCheapestPeriodAssumingLinearUnknownConsumption() throws IOException {
+ mockCommonDatasets(actions, "SpotPrices20230205.json");
+
+ Map actual = actions.calculateCheapestPeriod(Instant.parse("2023-02-05T16:00:00Z"),
+ Instant.parse("2023-02-06T06:00:00Z"), Duration.ofMinutes(236));
+ assertThat(actual.get("LowestPrice"), is(nullValue()));
+ assertThat(actual.get("CheapestStart"), is(equalTo(Instant.parse("2023-02-05T19:00:00Z"))));
+ assertThat(actual.get("HighestPrice"), is(nullValue()));
+ assertThat(actual.get("MostExpensiveStart"), is(equalTo(Instant.parse("2023-02-05T16:00:00Z"))));
+ }
+
+ @Test
+ void calculateCheapestPeriodForLinearPowerUsage() throws IOException {
+ mockCommonDatasets(actions);
+
+ Map actual = actions.calculateCheapestPeriod(Instant.parse("2023-02-04T12:00:00Z"),
+ Instant.parse("2023-02-05T23:00:00Z"), Duration.ofMinutes(61), QuantityType.valueOf(1000, Units.WATT));
+ assertThat(actual.get("LowestPrice"), is(equalTo(new BigDecimal("1.323990859575000000"))));
+ assertThat(actual.get("CheapestStart"), is(equalTo(Instant.parse("2023-02-05T12:00:00Z"))));
+ assertThat(actual.get("HighestPrice"), is(equalTo(new BigDecimal("2.589061780353348000"))));
+ assertThat(actual.get("MostExpensiveStart"), is(equalTo(Instant.parse("2023-02-04T17:00:00Z"))));
+ }
+
+ private void mockCommonDatasets(EnergiDataServiceActions actions) throws IOException {
+ mockCommonDatasets(actions, "SpotPrices20230204.json");
+ }
+
+ private void mockCommonDatasets(EnergiDataServiceActions actions, String spotPricesFilename) throws IOException {
+ SpotPrice[] spotPriceRecords = getObjectFromJson(spotPricesFilename, SpotPrice[].class);
+ Map spotPrices = Arrays.stream(spotPriceRecords)
+ .collect(Collectors.toMap(SpotPrice::hourStart, SpotPrice::spotPrice));
+
+ PriceListParser priceListParser = new PriceListParser(
+ Clock.fixed(spotPriceRecords[0].hourStart, EnergiDataServiceBindingConstants.DATAHUB_TIMEZONE));
+ DatahubPricelistRecords datahubRecords = getObjectFromJson("NetTariffs.json", DatahubPricelistRecords.class);
+ Map netTariffs = priceListParser
+ .toHourly(Arrays.stream(datahubRecords.records()).toList());
+ datahubRecords = getObjectFromJson("SystemTariffs.json", DatahubPricelistRecords.class);
+ Map systemTariffs = priceListParser
+ .toHourly(Arrays.stream(datahubRecords.records()).toList());
+ datahubRecords = getObjectFromJson("ElectricityTaxes.json", DatahubPricelistRecords.class);
+ Map electricityTaxes = priceListParser
+ .toHourly(Arrays.stream(datahubRecords.records()).toList());
+ datahubRecords = getObjectFromJson("TransmissionNetTariffs.json", DatahubPricelistRecords.class);
+ Map transmissionNetTariffs = priceListParser
+ .toHourly(Arrays.stream(datahubRecords.records()).toList());
+
+ when(handler.getSpotPrices()).thenReturn(spotPrices);
+ when(handler.getNetTariffs()).thenReturn(netTariffs);
+ when(handler.getSystemTariffs()).thenReturn(systemTariffs);
+ when(handler.getElectricityTaxes()).thenReturn(electricityTaxes);
+ when(handler.getTransmissionNetTariffs()).thenReturn(transmissionNetTariffs);
+ when(handler.getCurrency()).thenReturn(EnergiDataServiceBindingConstants.CURRENCY_DKK);
+ actions.setThingHandler(handler);
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/api/DateQueryParameterTest.java b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/api/DateQueryParameterTest.java
new file mode 100644
index 000000000..9334b3d90
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/api/DateQueryParameterTest.java
@@ -0,0 +1,65 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.api;
+
+import static org.hamcrest.MatcherAssert.assertThat;
+import static org.hamcrest.Matchers.equalTo;
+import static org.hamcrest.Matchers.is;
+
+import java.time.Duration;
+import java.time.LocalDate;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.junit.jupiter.api.Test;
+import org.junit.jupiter.api.extension.ExtendWith;
+import org.mockito.junit.jupiter.MockitoExtension;
+
+/**
+ * Tests for {@link DateQueryParameter}.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+@ExtendWith(MockitoExtension.class)
+public class DateQueryParameterTest {
+
+ @Test
+ void dateQueryParameterTypeWithNegativeOffset() {
+ DateQueryParameter parameter = DateQueryParameter.of(DateQueryParameterType.UTC_NOW, Duration.ofHours(-12));
+ assertThat(parameter.toString(), is(equalTo("utcnow-PT12H")));
+ }
+
+ @Test
+ void dateQueryParameterTypeWithPositiveOffset() {
+ DateQueryParameter parameter = DateQueryParameter.of(DateQueryParameterType.UTC_NOW, Duration.ofHours(12));
+ assertThat(parameter.toString(), is(equalTo("utcnow+PT12H")));
+ }
+
+ @Test
+ void dateQueryParameterTypeWithZeroOffset() {
+ DateQueryParameter parameter = DateQueryParameter.of(DateQueryParameterType.UTC_NOW, Duration.ZERO);
+ assertThat(parameter.toString(), is(equalTo("utcnow")));
+ }
+
+ @Test
+ void dateQueryParameterTypeWithoutOffset() {
+ DateQueryParameter parameter = DateQueryParameter.of(DateQueryParameterType.NOW);
+ assertThat(parameter.toString(), is(equalTo("now")));
+ }
+
+ @Test
+ void localDate() {
+ DateQueryParameter parameter = DateQueryParameter.of(LocalDate.of(2023, 2, 28));
+ assertThat(parameter.toString(), is(equalTo("2023-02-28")));
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/api/GlobalLocationNumberTest.java b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/api/GlobalLocationNumberTest.java
new file mode 100644
index 000000000..15e135e2c
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/api/GlobalLocationNumberTest.java
@@ -0,0 +1,46 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.api;
+
+import static org.hamcrest.MatcherAssert.assertThat;
+import static org.hamcrest.Matchers.is;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.junit.jupiter.api.Test;
+import org.junit.jupiter.api.extension.ExtendWith;
+import org.mockito.junit.jupiter.MockitoExtension;
+
+/**
+ * Tests for {@link GlobalLocationNumber}.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+@ExtendWith(MockitoExtension.class)
+public class GlobalLocationNumberTest {
+
+ @Test
+ void isValid() {
+ assertThat(GlobalLocationNumber.of("5790000682102").isValid(), is(true));
+ }
+
+ @Test
+ void isInvalid() {
+ assertThat(GlobalLocationNumber.of("5790000682103").isValid(), is(false));
+ }
+
+ @Test
+ void emptyIsInvalid() {
+ assertThat(GlobalLocationNumber.EMPTY.isValid(), is(false));
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/api/serialization/InstantDeserializerTest.java b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/api/serialization/InstantDeserializerTest.java
new file mode 100644
index 000000000..92a9c23c5
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/api/serialization/InstantDeserializerTest.java
@@ -0,0 +1,58 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.api.serialization;
+
+import static org.hamcrest.MatcherAssert.assertThat;
+import static org.hamcrest.Matchers.equalTo;
+import static org.hamcrest.Matchers.is;
+import static org.junit.jupiter.api.Assertions.assertThrows;
+
+import java.time.Instant;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.junit.jupiter.api.Test;
+import org.junit.jupiter.api.extension.ExtendWith;
+import org.junit.jupiter.params.ParameterizedTest;
+import org.junit.jupiter.params.provider.ValueSource;
+import org.mockito.junit.jupiter.MockitoExtension;
+
+import com.google.gson.Gson;
+import com.google.gson.GsonBuilder;
+import com.google.gson.JsonParseException;
+
+/**
+ * Tests for {@link InstantDeserializer}.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+@ExtendWith(MockitoExtension.class)
+public class InstantDeserializerTest {
+
+ private final Gson gson = new GsonBuilder().registerTypeAdapter(Instant.class, new InstantDeserializer()).create();
+
+ @Test
+ void instantWhenInvalidShouldThrowJsonParseException() {
+ assertThrows(JsonParseException.class, () -> {
+ gson.fromJson("\"invalid\"", Instant.class);
+ });
+ }
+
+ @ParameterizedTest
+ @ValueSource(strings = { "\"2023-04-17T20:38:01Z\"", "\"2023-04-17T20:38:01\"" })
+ void instantWhenValidShouldParse(String input) {
+ assertThat((@Nullable Instant) gson.fromJson(input, Instant.class),
+ is(equalTo(Instant.ofEpochSecond(1681763881))));
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/api/serialization/LocalDateDeserializerTest.java b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/api/serialization/LocalDateDeserializerTest.java
new file mode 100644
index 000000000..116a8a87a
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/api/serialization/LocalDateDeserializerTest.java
@@ -0,0 +1,56 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.api.serialization;
+
+import static org.hamcrest.MatcherAssert.assertThat;
+import static org.hamcrest.Matchers.equalTo;
+import static org.hamcrest.Matchers.is;
+import static org.junit.jupiter.api.Assertions.assertThrows;
+
+import java.time.LocalDateTime;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.junit.jupiter.api.Test;
+import org.junit.jupiter.api.extension.ExtendWith;
+import org.mockito.junit.jupiter.MockitoExtension;
+
+import com.google.gson.Gson;
+import com.google.gson.GsonBuilder;
+import com.google.gson.JsonParseException;
+
+/**
+ * Tests for {@link LocalDateDeserializer}.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+@ExtendWith(MockitoExtension.class)
+public class LocalDateDeserializerTest {
+
+ private final Gson gson = new GsonBuilder()
+ .registerTypeAdapter(LocalDateTime.class, new LocalDateTimeDeserializer()).create();
+
+ @Test
+ void localDateTimeWhenInvalidShouldThrowJsonParseException() {
+ assertThrows(JsonParseException.class, () -> {
+ gson.fromJson("\"invalid\"", LocalDateTime.class);
+ });
+ }
+
+ @Test
+ void instantWhenValidShouldParse() {
+ assertThat((@Nullable LocalDateTime) gson.fromJson("\"2023-04-17T20:38:01\"", LocalDateTime.class),
+ is(equalTo(LocalDateTime.of(2023, 4, 17, 20, 38, 1, 0))));
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/retry/strategy/ExponentialBackoffTest.java b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/retry/strategy/ExponentialBackoffTest.java
new file mode 100644
index 000000000..90a6c4317
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/retry/strategy/ExponentialBackoffTest.java
@@ -0,0 +1,39 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.retry.strategy;
+
+import static org.hamcrest.CoreMatchers.*;
+import static org.hamcrest.MatcherAssert.assertThat;
+
+import java.time.Duration;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.junit.jupiter.api.Test;
+import org.openhab.binding.energidataservice.internal.retry.RetryStrategy;
+
+/**
+ * Tests for {@link ExponentialBackoff}.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class ExponentialBackoffTest {
+
+ @Test
+ void exponential() {
+ RetryStrategy retryPolicy = new ExponentialBackoff().withMinimum(Duration.ofSeconds(2)).withJitter(0.0);
+ for (long i = 2; i <= 256; i *= 2) {
+ assertThat(retryPolicy.getDuration().toSeconds(), is(i));
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/retry/strategy/FixedTimeTest.java b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/retry/strategy/FixedTimeTest.java
new file mode 100644
index 000000000..22f4b21a7
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/retry/strategy/FixedTimeTest.java
@@ -0,0 +1,57 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.retry.strategy;
+
+import static org.hamcrest.CoreMatchers.*;
+import static org.hamcrest.MatcherAssert.assertThat;
+import static org.hamcrest.Matchers.is;
+
+import java.time.Clock;
+import java.time.Duration;
+import java.time.Instant;
+import java.time.LocalTime;
+import java.time.ZoneId;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.junit.jupiter.api.Test;
+import org.openhab.binding.energidataservice.internal.retry.RetryStrategy;
+
+/**
+ * Tests for {@link FixedTime}.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class FixedTimeTest {
+
+ @Test
+ void beforeNoon() {
+ RetryStrategy retryPolicy = new FixedTime(LocalTime.of(12, 0),
+ Clock.fixed(Instant.parse("2023-01-24T10:00:00Z"), ZoneId.of("UTC")));
+ assertThat(retryPolicy.getDuration(), is(Duration.ofHours(2)));
+ }
+
+ @Test
+ void atNoon() {
+ RetryStrategy retryPolicy = new FixedTime(LocalTime.of(12, 0),
+ Clock.fixed(Instant.parse("2023-01-24T12:00:00Z"), ZoneId.of("UTC")));
+ assertThat(retryPolicy.getDuration(), is(Duration.ZERO));
+ }
+
+ @Test
+ void afterNoon() {
+ RetryStrategy retryPolicy = new FixedTime(LocalTime.of(12, 0),
+ Clock.fixed(Instant.parse("2023-01-24T13:00:00Z"), ZoneId.of("UTC")));
+ assertThat(retryPolicy.getDuration(), is(Duration.ofHours(23)));
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/retry/strategy/LinearTest.java b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/retry/strategy/LinearTest.java
new file mode 100644
index 000000000..4d8b6abe2
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/java/org/openhab/binding/energidataservice/internal/retry/strategy/LinearTest.java
@@ -0,0 +1,39 @@
+/**
+ * Copyright (c) 2010-2023 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.energidataservice.internal.retry.strategy;
+
+import static org.hamcrest.CoreMatchers.*;
+import static org.hamcrest.MatcherAssert.assertThat;
+
+import java.time.Duration;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.junit.jupiter.api.Test;
+import org.openhab.binding.energidataservice.internal.retry.RetryStrategy;
+
+/**
+ * Tests for {@link Linear}.
+ *
+ * @author Jacob Laursen - Initial contribution
+ */
+@NonNullByDefault
+public class LinearTest {
+
+ @Test
+ void linear() {
+ RetryStrategy retryPolicy = new Linear().withMinimum(Duration.ofMinutes(1)).withJitter(0.0);
+ for (int i = 0; i <= 10; i++) {
+ assertThat(retryPolicy.getDuration(), is(Duration.ofMinutes(1)));
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/DatahubPricelistElectricityTax.json b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/DatahubPricelistElectricityTax.json
new file mode 100644
index 000000000..fce1f184d
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/DatahubPricelistElectricityTax.json
@@ -0,0 +1,83 @@
+{
+ "total": 2,
+ "filters": "{\"GLN_Number\":[\"5790000432752\"],\"ChargeType\":[\"D03\"],\"Note\":[\"Elafgift\"]}",
+ "dataset": "DatahubPricelist",
+ "records": [
+ {
+ "ChargeOwner": "Energinet Systemansvar A/S (SYO)",
+ "GLN_Number": "5790000432752",
+ "ChargeType": "D03",
+ "ChargeTypeCode": "EA-001",
+ "Note": "Elafgift",
+ "Description": "Elafgiften",
+ "ValidFrom": "2023-07-01T00:00:00",
+ "ValidTo": null,
+ "VATClass": "D02",
+ "Price1": 0.697,
+ "Price2": null,
+ "Price3": null,
+ "Price4": null,
+ "Price5": null,
+ "Price6": null,
+ "Price7": null,
+ "Price8": null,
+ "Price9": null,
+ "Price10": null,
+ "Price11": null,
+ "Price12": null,
+ "Price13": null,
+ "Price14": null,
+ "Price15": null,
+ "Price16": null,
+ "Price17": null,
+ "Price18": null,
+ "Price19": null,
+ "Price20": null,
+ "Price21": null,
+ "Price22": null,
+ "Price23": null,
+ "Price24": null,
+ "TransparentInvoicing": 1,
+ "TaxIndicator": 1,
+ "ResolutionDuration": "P1D"
+ },
+ {
+ "ChargeOwner": "Energinet Systemansvar A/S (SYO)",
+ "GLN_Number": "5790000432752",
+ "ChargeType": "D03",
+ "ChargeTypeCode": "EA-001",
+ "Note": "Elafgift",
+ "Description": "Elafgiften",
+ "ValidFrom": "2023-01-01T00:00:00",
+ "ValidTo": "2023-07-01T00:00:00",
+ "VATClass": "D02",
+ "Price1": 0.008,
+ "Price2": null,
+ "Price3": null,
+ "Price4": null,
+ "Price5": null,
+ "Price6": null,
+ "Price7": null,
+ "Price8": null,
+ "Price9": null,
+ "Price10": null,
+ "Price11": null,
+ "Price12": null,
+ "Price13": null,
+ "Price14": null,
+ "Price15": null,
+ "Price16": null,
+ "Price17": null,
+ "Price18": null,
+ "Price19": null,
+ "Price20": null,
+ "Price21": null,
+ "Price22": null,
+ "Price23": null,
+ "Price24": null,
+ "TransparentInvoicing": 1,
+ "TaxIndicator": 1,
+ "ResolutionDuration": "P1D"
+ }
+ ]
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/DatahubPricelistN1.json b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/DatahubPricelistN1.json
new file mode 100644
index 000000000..8d265147a
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/DatahubPricelistN1.json
@@ -0,0 +1,588 @@
+{
+ "total": 20,
+ "filters": "{\"ChargeTypeCode\":[\"CD\",\"CD R\"],\"ChargeType\":[\"D03\"],\"GLN_Number\":[\"5790001089030\"]}",
+ "limit": 100,
+ "dataset": "DatahubPricelist",
+ "records": [
+ {
+ "ValidFrom": "2023-04-01T00:00:00",
+ "ValidTo": null,
+ "ChargeTypeCode": "CD",
+ "Price1": 0.432225,
+ "Price2": 0.432225,
+ "Price3": 0.432225,
+ "Price4": 0.432225,
+ "Price5": 0.432225,
+ "Price6": 0.432225,
+ "Price7": 0.432225,
+ "Price8": 0.432225,
+ "Price9": 0.432225,
+ "Price10": 0.432225,
+ "Price11": 0.432225,
+ "Price12": 0.432225,
+ "Price13": 0.432225,
+ "Price14": 0.432225,
+ "Price15": 0.432225,
+ "Price16": 0.432225,
+ "Price17": 0.432225,
+ "Price18": 0.432225,
+ "Price19": 0.432225,
+ "Price20": 0.432225,
+ "Price21": 0.432225,
+ "Price22": 0.432225,
+ "Price23": 0.432225,
+ "Price24": 0.432225
+ },
+ {
+ "ValidFrom": "2023-01-01T00:00:00",
+ "ValidTo": "2023-04-01T00:00:00",
+ "ChargeTypeCode": "CD",
+ "Price1": 0.432225,
+ "Price2": 0.432225,
+ "Price3": 0.432225,
+ "Price4": 0.432225,
+ "Price5": 0.432225,
+ "Price6": 0.432225,
+ "Price7": 0.432225,
+ "Price8": 0.432225,
+ "Price9": 0.432225,
+ "Price10": 0.432225,
+ "Price11": 0.432225,
+ "Price12": 0.432225,
+ "Price13": 0.432225,
+ "Price14": 0.432225,
+ "Price15": 0.432225,
+ "Price16": 0.432225,
+ "Price17": 0.432225,
+ "Price18": 1.05619,
+ "Price19": 1.05619,
+ "Price20": 1.05619,
+ "Price21": 0.432225,
+ "Price22": 0.432225,
+ "Price23": 0.432225,
+ "Price24": 0.432225
+ },
+ {
+ "ValidFrom": "2022-11-01T00:00:00",
+ "ValidTo": "2023-01-01T00:00:00",
+ "ChargeTypeCode": "CD",
+ "Price1": 0.407717,
+ "Price2": 0.407717,
+ "Price3": 0.407717,
+ "Price4": 0.407717,
+ "Price5": 0.407717,
+ "Price6": 0.407717,
+ "Price7": 0.407717,
+ "Price8": 0.407717,
+ "Price9": 0.407717,
+ "Price10": 0.407717,
+ "Price11": 0.407717,
+ "Price12": 0.407717,
+ "Price13": 0.407717,
+ "Price14": 0.407717,
+ "Price15": 0.407717,
+ "Price16": 0.407717,
+ "Price17": 0.407717,
+ "Price18": 1.015888,
+ "Price19": 1.015888,
+ "Price20": 1.015888,
+ "Price21": 0.407717,
+ "Price22": 0.407717,
+ "Price23": 0.407717,
+ "Price24": 0.407717
+ },
+ {
+ "ValidFrom": "2022-10-01T00:00:00",
+ "ValidTo": "2022-11-01T00:00:00",
+ "ChargeTypeCode": "CD",
+ "Price1": 0.31535,
+ "Price2": 0.31535,
+ "Price3": 0.31535,
+ "Price4": 0.31535,
+ "Price5": 0.31535,
+ "Price6": 0.31535,
+ "Price7": 0.31535,
+ "Price8": 0.31535,
+ "Price9": 0.31535,
+ "Price10": 0.31535,
+ "Price11": 0.31535,
+ "Price12": 0.31535,
+ "Price13": 0.31535,
+ "Price14": 0.31535,
+ "Price15": 0.31535,
+ "Price16": 0.31535,
+ "Price17": 0.31535,
+ "Price18": 0.821619,
+ "Price19": 0.821619,
+ "Price20": 0.821619,
+ "Price21": 0.31535,
+ "Price22": 0.31535,
+ "Price23": 0.31535,
+ "Price24": 0.31535
+ },
+ {
+ "ValidFrom": "2022-08-01T00:00:00",
+ "ValidTo": "2022-10-01T00:00:00",
+ "ChargeTypeCode": "CD",
+ "Price1": 0.31535,
+ "Price2": 0.31535,
+ "Price3": 0.31535,
+ "Price4": 0.31535,
+ "Price5": 0.31535,
+ "Price6": 0.31535,
+ "Price7": 0.31535,
+ "Price8": 0.31535,
+ "Price9": 0.31535,
+ "Price10": 0.31535,
+ "Price11": 0.31535,
+ "Price12": 0.31535,
+ "Price13": 0.31535,
+ "Price14": 0.31535,
+ "Price15": 0.31535,
+ "Price16": 0.31535,
+ "Price17": 0.31535,
+ "Price18": 0.31535,
+ "Price19": 0.31535,
+ "Price20": 0.31535,
+ "Price21": 0.31535,
+ "Price22": 0.31535,
+ "Price23": 0.31535,
+ "Price24": 0.31535
+ },
+ {
+ "ValidFrom": "2022-04-01T00:00:00",
+ "ValidTo": "2022-08-01T00:00:00",
+ "ChargeTypeCode": "CD",
+ "Price1": 0.227969,
+ "Price2": 0.227969,
+ "Price3": 0.227969,
+ "Price4": 0.227969,
+ "Price5": 0.227969,
+ "Price6": 0.227969,
+ "Price7": 0.227969,
+ "Price8": 0.227969,
+ "Price9": 0.227969,
+ "Price10": 0.227969,
+ "Price11": 0.227969,
+ "Price12": 0.227969,
+ "Price13": 0.227969,
+ "Price14": 0.227969,
+ "Price15": 0.227969,
+ "Price16": 0.227969,
+ "Price17": 0.227969,
+ "Price18": 0.227969,
+ "Price19": 0.227969,
+ "Price20": 0.227969,
+ "Price21": 0.227969,
+ "Price22": 0.227969,
+ "Price23": 0.227969,
+ "Price24": 0.227969
+ },
+ {
+ "ValidFrom": "2022-01-01T00:00:00",
+ "ValidTo": "2022-04-01T00:00:00",
+ "ChargeTypeCode": "CD",
+ "Price1": 0.183226,
+ "Price2": 0.183226,
+ "Price3": 0.183226,
+ "Price4": 0.183226,
+ "Price5": 0.183226,
+ "Price6": 0.183226,
+ "Price7": 0.183226,
+ "Price8": 0.183226,
+ "Price9": 0.183226,
+ "Price10": 0.183226,
+ "Price11": 0.183226,
+ "Price12": 0.183226,
+ "Price13": 0.183226,
+ "Price14": 0.183226,
+ "Price15": 0.183226,
+ "Price16": 0.183226,
+ "Price17": 0.183226,
+ "Price18": 0.543732,
+ "Price19": 0.543732,
+ "Price20": 0.543732,
+ "Price21": 0.183226,
+ "Price22": 0.183226,
+ "Price23": 0.183226,
+ "Price24": 0.183226
+ },
+ {
+ "ValidFrom": "2021-10-01T00:00:00",
+ "ValidTo": "2022-01-01T00:00:00",
+ "ChargeTypeCode": "CD",
+ "Price1": 0.1717,
+ "Price2": 0.1717,
+ "Price3": 0.1717,
+ "Price4": 0.1717,
+ "Price5": 0.1717,
+ "Price6": 0.1717,
+ "Price7": 0.1717,
+ "Price8": 0.1717,
+ "Price9": 0.1717,
+ "Price10": 0.1717,
+ "Price11": 0.1717,
+ "Price12": 0.1717,
+ "Price13": 0.1717,
+ "Price14": 0.1717,
+ "Price15": 0.1717,
+ "Price16": 0.1717,
+ "Price17": 0.1717,
+ "Price18": 0.5448,
+ "Price19": 0.5448,
+ "Price20": 0.5448,
+ "Price21": 0.1717,
+ "Price22": 0.1717,
+ "Price23": 0.1717,
+ "Price24": 0.1717
+ },
+ {
+ "ValidFrom": "2021-04-01T00:00:00",
+ "ValidTo": "2021-10-01T00:00:00",
+ "ChargeTypeCode": "CD",
+ "Price1": 0.1717,
+ "Price2": 0.1717,
+ "Price3": 0.1717,
+ "Price4": 0.1717,
+ "Price5": 0.1717,
+ "Price6": 0.1717,
+ "Price7": 0.1717,
+ "Price8": 0.1717,
+ "Price9": 0.1717,
+ "Price10": 0.1717,
+ "Price11": 0.1717,
+ "Price12": 0.1717,
+ "Price13": 0.1717,
+ "Price14": 0.1717,
+ "Price15": 0.1717,
+ "Price16": 0.1717,
+ "Price17": 0.1717,
+ "Price18": 0.1717,
+ "Price19": 0.1717,
+ "Price20": 0.1717,
+ "Price21": 0.1717,
+ "Price22": 0.1717,
+ "Price23": 0.1717,
+ "Price24": 0.1717
+ },
+ {
+ "ValidFrom": "2021-01-01T00:00:00",
+ "ValidTo": "2021-04-01T00:00:00",
+ "ChargeTypeCode": "CD",
+ "Price1": 0.1717,
+ "Price2": 0.1717,
+ "Price3": 0.1717,
+ "Price4": 0.1717,
+ "Price5": 0.1717,
+ "Price6": 0.1717,
+ "Price7": 0.1717,
+ "Price8": 0.1717,
+ "Price9": 0.1717,
+ "Price10": 0.1717,
+ "Price11": 0.1717,
+ "Price12": 0.1717,
+ "Price13": 0.1717,
+ "Price14": 0.1717,
+ "Price15": 0.1717,
+ "Price16": 0.1717,
+ "Price17": 0.1717,
+ "Price18": 0.5448,
+ "Price19": 0.5448,
+ "Price20": 0.5448,
+ "Price21": 0.1717,
+ "Price22": 0.1717,
+ "Price23": 0.1717,
+ "Price24": 0.1717
+ },
+ {
+ "ValidFrom": "2023-01-01T00:00:00",
+ "ValidTo": null,
+ "ChargeTypeCode": "CD R",
+ "Price1": 0.0,
+ "Price2": 0.0,
+ "Price3": 0.0,
+ "Price4": 0.0,
+ "Price5": 0.0,
+ "Price6": 0.0,
+ "Price7": 0.0,
+ "Price8": 0.0,
+ "Price9": 0.0,
+ "Price10": 0.0,
+ "Price11": 0.0,
+ "Price12": 0.0,
+ "Price13": 0.0,
+ "Price14": 0.0,
+ "Price15": 0.0,
+ "Price16": 0.0,
+ "Price17": 0.0,
+ "Price18": 0.0,
+ "Price19": 0.0,
+ "Price20": 0.0,
+ "Price21": 0.0,
+ "Price22": 0.0,
+ "Price23": 0.0,
+ "Price24": 0.0
+ },
+ {
+ "ValidFrom": "2022-12-01T00:00:00",
+ "ValidTo": "2023-01-01T00:00:00",
+ "ChargeTypeCode": "CD R",
+ "Price1": -0.407717,
+ "Price2": -0.407717,
+ "Price3": -0.407717,
+ "Price4": -0.407717,
+ "Price5": -0.407717,
+ "Price6": -0.407717,
+ "Price7": -0.407717,
+ "Price8": -0.407717,
+ "Price9": -0.407717,
+ "Price10": -0.407717,
+ "Price11": -0.407717,
+ "Price12": -0.407717,
+ "Price13": -0.407717,
+ "Price14": -0.407717,
+ "Price15": -0.407717,
+ "Price16": -0.407717,
+ "Price17": -0.407717,
+ "Price18": -1.015888,
+ "Price19": -1.015888,
+ "Price20": -1.015888,
+ "Price21": -0.407717,
+ "Price22": -0.407717,
+ "Price23": -0.407717,
+ "Price24": -0.407717
+ },
+ {
+ "ValidFrom": "2022-10-01T00:00:00",
+ "ValidTo": "2022-12-01T00:00:00",
+ "ChargeTypeCode": "CD R",
+ "Price1": -0.0202,
+ "Price2": -0.0202,
+ "Price3": -0.0202,
+ "Price4": -0.0202,
+ "Price5": -0.0202,
+ "Price6": -0.0202,
+ "Price7": -0.0202,
+ "Price8": -0.0202,
+ "Price9": -0.0202,
+ "Price10": -0.0202,
+ "Price11": -0.0202,
+ "Price12": -0.0202,
+ "Price13": -0.0202,
+ "Price14": -0.0202,
+ "Price15": -0.0202,
+ "Price16": -0.0202,
+ "Price17": -0.0202,
+ "Price18": -0.042484,
+ "Price19": -0.042484,
+ "Price20": -0.042484,
+ "Price21": -0.0202,
+ "Price22": -0.0202,
+ "Price23": -0.0202,
+ "Price24": -0.0202
+ },
+ {
+ "ValidFrom": "2022-08-01T00:00:00",
+ "ValidTo": "2022-10-01T00:00:00",
+ "ChargeTypeCode": "CD R",
+ "Price1": -0.0202,
+ "Price2": -0.0202,
+ "Price3": -0.0202,
+ "Price4": -0.0202,
+ "Price5": -0.0202,
+ "Price6": -0.0202,
+ "Price7": -0.0202,
+ "Price8": -0.0202,
+ "Price9": -0.0202,
+ "Price10": -0.0202,
+ "Price11": -0.0202,
+ "Price12": -0.0202,
+ "Price13": -0.0202,
+ "Price14": -0.0202,
+ "Price15": -0.0202,
+ "Price16": -0.0202,
+ "Price17": -0.0202,
+ "Price18": -0.0202,
+ "Price19": -0.0202,
+ "Price20": -0.0202,
+ "Price21": -0.0202,
+ "Price22": -0.0202,
+ "Price23": -0.0202,
+ "Price24": -0.0202
+ },
+ {
+ "ValidFrom": "2022-04-01T00:00:00",
+ "ValidTo": "2022-08-01T00:00:00",
+ "ChargeTypeCode": "CD R",
+ "Price1": -0.0202,
+ "Price2": -0.0202,
+ "Price3": -0.0202,
+ "Price4": -0.0202,
+ "Price5": -0.0202,
+ "Price6": -0.0202,
+ "Price7": -0.0202,
+ "Price8": -0.0202,
+ "Price9": -0.0202,
+ "Price10": -0.0202,
+ "Price11": -0.0202,
+ "Price12": -0.0202,
+ "Price13": -0.0202,
+ "Price14": -0.0202,
+ "Price15": -0.0202,
+ "Price16": -0.0202,
+ "Price17": -0.0202,
+ "Price18": -0.0202,
+ "Price19": -0.0202,
+ "Price20": -0.0202,
+ "Price21": -0.0202,
+ "Price22": -0.0202,
+ "Price23": -0.0202,
+ "Price24": -0.0202
+ },
+ {
+ "ValidFrom": "2022-01-01T00:00:00",
+ "ValidTo": "2022-04-01T00:00:00",
+ "ChargeTypeCode": "CD R",
+ "Price1": -0.0202,
+ "Price2": -0.0202,
+ "Price3": -0.0202,
+ "Price4": -0.0202,
+ "Price5": -0.0202,
+ "Price6": -0.0202,
+ "Price7": -0.0202,
+ "Price8": -0.0202,
+ "Price9": -0.0202,
+ "Price10": -0.0202,
+ "Price11": -0.0202,
+ "Price12": -0.0202,
+ "Price13": -0.0202,
+ "Price14": -0.0202,
+ "Price15": -0.0202,
+ "Price16": -0.0202,
+ "Price17": -0.0202,
+ "Price18": -0.042484,
+ "Price19": -0.042484,
+ "Price20": -0.042484,
+ "Price21": -0.0202,
+ "Price22": -0.0202,
+ "Price23": -0.0202,
+ "Price24": -0.0202
+ },
+ {
+ "ValidFrom": "2021-10-01T00:00:00",
+ "ValidTo": "2022-01-01T00:00:00",
+ "ChargeTypeCode": "CD R",
+ "Price1": -0.0224,
+ "Price2": -0.0224,
+ "Price3": -0.0224,
+ "Price4": -0.0224,
+ "Price5": -0.0224,
+ "Price6": -0.0224,
+ "Price7": -0.0224,
+ "Price8": -0.0224,
+ "Price9": -0.0224,
+ "Price10": -0.0224,
+ "Price11": -0.0224,
+ "Price12": -0.0224,
+ "Price13": -0.0224,
+ "Price14": -0.0224,
+ "Price15": -0.0224,
+ "Price16": -0.0224,
+ "Price17": -0.0224,
+ "Price18": -0.0471,
+ "Price19": -0.0471,
+ "Price20": -0.0471,
+ "Price21": -0.0224,
+ "Price22": -0.0224,
+ "Price23": -0.0224,
+ "Price24": -0.0224
+ },
+ {
+ "ValidFrom": "2021-04-01T00:00:00",
+ "ValidTo": "2021-10-01T00:00:00",
+ "ChargeTypeCode": "CD R",
+ "Price1": -0.0224,
+ "Price2": -0.0224,
+ "Price3": -0.0224,
+ "Price4": -0.0224,
+ "Price5": -0.0224,
+ "Price6": -0.0224,
+ "Price7": -0.0224,
+ "Price8": -0.0224,
+ "Price9": -0.0224,
+ "Price10": -0.0224,
+ "Price11": -0.0224,
+ "Price12": -0.0224,
+ "Price13": -0.0224,
+ "Price14": -0.0224,
+ "Price15": -0.0224,
+ "Price16": -0.0224,
+ "Price17": -0.0224,
+ "Price18": -0.0224,
+ "Price19": -0.0224,
+ "Price20": -0.0224,
+ "Price21": -0.0224,
+ "Price22": -0.0224,
+ "Price23": -0.0224,
+ "Price24": -0.0224
+ },
+ {
+ "ValidFrom": "2021-03-01T00:00:00",
+ "ValidTo": "2021-04-01T00:00:00",
+ "ChargeTypeCode": "CD R",
+ "Price1": -0.0224,
+ "Price2": -0.0224,
+ "Price3": -0.0224,
+ "Price4": -0.0224,
+ "Price5": -0.0224,
+ "Price6": -0.0224,
+ "Price7": -0.0224,
+ "Price8": -0.0224,
+ "Price9": -0.0224,
+ "Price10": -0.0224,
+ "Price11": -0.0224,
+ "Price12": -0.0224,
+ "Price13": -0.0224,
+ "Price14": -0.0224,
+ "Price15": -0.0224,
+ "Price16": -0.0224,
+ "Price17": -0.0224,
+ "Price18": -0.0471,
+ "Price19": -0.0471,
+ "Price20": -0.0471,
+ "Price21": -0.0224,
+ "Price22": -0.0224,
+ "Price23": -0.0224,
+ "Price24": -0.0224
+ },
+ {
+ "ValidFrom": "2021-01-01T00:00:00",
+ "ValidTo": "2021-03-01T00:00:00",
+ "ChargeTypeCode": "CD R",
+ "Price1": -0.0224,
+ "Price2": -0.0224,
+ "Price3": -0.0224,
+ "Price4": -0.0224,
+ "Price5": -0.0224,
+ "Price6": -0.0224,
+ "Price7": -0.0224,
+ "Price8": -0.0224,
+ "Price9": -0.0224,
+ "Price10": -0.0224,
+ "Price11": -0.0224,
+ "Price12": -0.0224,
+ "Price13": -0.0224,
+ "Price14": -0.0224,
+ "Price15": -0.0224,
+ "Price16": -0.0224,
+ "Price17": -0.0471,
+ "Price18": -0.0471,
+ "Price19": -0.0471,
+ "Price20": -0.0471,
+ "Price21": -0.0224,
+ "Price22": -0.0224,
+ "Price23": -0.0224,
+ "Price24": -0.0224
+ }
+ ]
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/DatahubPricelistNordEnergi.json b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/DatahubPricelistNordEnergi.json
new file mode 100644
index 000000000..7641d136c
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/DatahubPricelistNordEnergi.json
@@ -0,0 +1,37 @@
+{
+ "total": 1,
+ "filters": "{\"Note\":[\"Nettarif C\"],\"ChargeType\":[\"D03\"],\"GLN_Number\":[\"5790000610877\"]}",
+ "limit": 100,
+ "dataset": "DatahubPricelist",
+ "records": [
+ {
+ "ValidFrom": "2023-01-01T00:00:00",
+ "ValidTo": null,
+ "ChargeTypeCode": "TA031U200",
+ "Price1": 0.245,
+ "Price2": null,
+ "Price3": null,
+ "Price4": null,
+ "Price5": null,
+ "Price6": null,
+ "Price7": null,
+ "Price8": null,
+ "Price9": null,
+ "Price10": null,
+ "Price11": null,
+ "Price12": null,
+ "Price13": null,
+ "Price14": null,
+ "Price15": null,
+ "Price16": null,
+ "Price17": null,
+ "Price18": null,
+ "Price19": null,
+ "Price20": null,
+ "Price21": null,
+ "Price22": null,
+ "Price23": null,
+ "Price24": null
+ }
+ ]
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/DatahubPricelistTrefor.json b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/DatahubPricelistTrefor.json
new file mode 100644
index 000000000..e0610a8ca
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/DatahubPricelistTrefor.json
@@ -0,0 +1,2908 @@
+{
+ "total": 850,
+ "filters": "{\"ChargeType\":[\"D03\"],\"GLN_Number\":[\"5790000706686\"],\"ChargeTypeCode\":[\"46\"],\"Note\":[\"Nettarif C time\"]}",
+ "limit": 100,
+ "dataset": "DatahubPricelist",
+ "records": [
+ {
+ "ValidFrom": "2023-04-30T00:00:00",
+ "ValidTo": null,
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-29T00:00:00",
+ "ValidTo": "2023-04-30T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-28T00:00:00",
+ "ValidTo": "2023-04-29T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-27T00:00:00",
+ "ValidTo": "2023-04-28T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-26T00:00:00",
+ "ValidTo": "2023-04-27T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-25T00:00:00",
+ "ValidTo": "2023-04-26T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-24T00:00:00",
+ "ValidTo": "2023-04-25T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-23T00:00:00",
+ "ValidTo": "2023-04-24T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-22T00:00:00",
+ "ValidTo": "2023-04-23T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-21T00:00:00",
+ "ValidTo": "2023-04-22T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-20T00:00:00",
+ "ValidTo": "2023-04-21T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-19T00:00:00",
+ "ValidTo": "2023-04-20T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-18T00:00:00",
+ "ValidTo": "2023-04-19T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-17T00:00:00",
+ "ValidTo": "2023-04-18T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-16T00:00:00",
+ "ValidTo": "2023-04-17T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-15T00:00:00",
+ "ValidTo": "2023-04-16T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-14T00:00:00",
+ "ValidTo": "2023-04-15T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-13T00:00:00",
+ "ValidTo": "2023-04-14T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-12T00:00:00",
+ "ValidTo": "2023-04-13T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-11T00:00:00",
+ "ValidTo": "2023-04-12T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-10T00:00:00",
+ "ValidTo": "2023-04-11T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-09T00:00:00",
+ "ValidTo": "2023-04-10T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-08T00:00:00",
+ "ValidTo": "2023-04-09T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-07T00:00:00",
+ "ValidTo": "2023-04-08T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-06T00:00:00",
+ "ValidTo": "2023-04-07T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-05T00:00:00",
+ "ValidTo": "2023-04-06T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-04T00:00:00",
+ "ValidTo": "2023-04-05T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-03T00:00:00",
+ "ValidTo": "2023-04-04T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-02T00:00:00",
+ "ValidTo": "2023-04-03T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-04-01T00:00:00",
+ "ValidTo": "2023-04-02T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.3871,
+ "Price8": 0.3871,
+ "Price9": 0.3871,
+ "Price10": 0.3871,
+ "Price11": 0.3871,
+ "Price12": 0.3871,
+ "Price13": 0.3871,
+ "Price14": 0.3871,
+ "Price15": 0.3871,
+ "Price16": 0.3871,
+ "Price17": 0.3871,
+ "Price18": 1.0065,
+ "Price19": 1.0065,
+ "Price20": 1.0065,
+ "Price21": 1.0065,
+ "Price22": 0.3871,
+ "Price23": 0.3871,
+ "Price24": 0.3871
+ },
+ {
+ "ValidFrom": "2023-03-31T00:00:00",
+ "ValidTo": "2023-04-01T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-30T00:00:00",
+ "ValidTo": "2023-03-31T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-29T00:00:00",
+ "ValidTo": "2023-03-30T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-28T00:00:00",
+ "ValidTo": "2023-03-29T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-27T00:00:00",
+ "ValidTo": "2023-03-28T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-26T00:00:00",
+ "ValidTo": "2023-03-27T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-25T00:00:00",
+ "ValidTo": "2023-03-26T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-24T00:00:00",
+ "ValidTo": "2023-03-25T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-23T00:00:00",
+ "ValidTo": "2023-03-24T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-22T00:00:00",
+ "ValidTo": "2023-03-23T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-21T00:00:00",
+ "ValidTo": "2023-03-22T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-20T00:00:00",
+ "ValidTo": "2023-03-21T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-19T00:00:00",
+ "ValidTo": "2023-03-20T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-18T00:00:00",
+ "ValidTo": "2023-03-19T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-17T00:00:00",
+ "ValidTo": "2023-03-18T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-16T00:00:00",
+ "ValidTo": "2023-03-17T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-15T00:00:00",
+ "ValidTo": "2023-03-16T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-14T00:00:00",
+ "ValidTo": "2023-03-15T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-13T00:00:00",
+ "ValidTo": "2023-03-14T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-12T00:00:00",
+ "ValidTo": "2023-03-13T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-11T00:00:00",
+ "ValidTo": "2023-03-12T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-10T00:00:00",
+ "ValidTo": "2023-03-11T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-09T00:00:00",
+ "ValidTo": "2023-03-10T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-08T00:00:00",
+ "ValidTo": "2023-03-09T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-07T00:00:00",
+ "ValidTo": "2023-03-08T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-06T00:00:00",
+ "ValidTo": "2023-03-07T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-05T00:00:00",
+ "ValidTo": "2023-03-06T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-04T00:00:00",
+ "ValidTo": "2023-03-05T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-03T00:00:00",
+ "ValidTo": "2023-03-04T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-02T00:00:00",
+ "ValidTo": "2023-03-03T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-03-01T00:00:00",
+ "ValidTo": "2023-03-02T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-28T00:00:00",
+ "ValidTo": "2023-03-01T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-27T00:00:00",
+ "ValidTo": "2023-02-28T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-26T00:00:00",
+ "ValidTo": "2023-02-27T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-25T00:00:00",
+ "ValidTo": "2023-02-26T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-24T00:00:00",
+ "ValidTo": "2023-02-25T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-23T00:00:00",
+ "ValidTo": "2023-02-24T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-22T00:00:00",
+ "ValidTo": "2023-02-23T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-21T00:00:00",
+ "ValidTo": "2023-02-22T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-20T00:00:00",
+ "ValidTo": "2023-02-21T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-19T00:00:00",
+ "ValidTo": "2023-02-20T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-18T00:00:00",
+ "ValidTo": "2023-02-19T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-17T00:00:00",
+ "ValidTo": "2023-02-18T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-16T00:00:00",
+ "ValidTo": "2023-02-17T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-15T00:00:00",
+ "ValidTo": "2023-02-16T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-14T00:00:00",
+ "ValidTo": "2023-02-15T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-13T00:00:00",
+ "ValidTo": "2023-02-14T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-12T00:00:00",
+ "ValidTo": "2023-02-13T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-11T00:00:00",
+ "ValidTo": "2023-02-12T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-10T00:00:00",
+ "ValidTo": "2023-02-11T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-09T00:00:00",
+ "ValidTo": "2023-02-10T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-08T00:00:00",
+ "ValidTo": "2023-02-09T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-07T00:00:00",
+ "ValidTo": "2023-02-08T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-06T00:00:00",
+ "ValidTo": "2023-02-07T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-05T00:00:00",
+ "ValidTo": "2023-02-06T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-04T00:00:00",
+ "ValidTo": "2023-02-05T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-03T00:00:00",
+ "ValidTo": "2023-02-04T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-02T00:00:00",
+ "ValidTo": "2023-02-03T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-02-01T00:00:00",
+ "ValidTo": "2023-02-02T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-01-31T00:00:00",
+ "ValidTo": "2023-02-01T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-01-30T00:00:00",
+ "ValidTo": "2023-01-31T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-01-29T00:00:00",
+ "ValidTo": "2023-01-30T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-01-28T00:00:00",
+ "ValidTo": "2023-01-29T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-01-27T00:00:00",
+ "ValidTo": "2023-01-28T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-01-26T00:00:00",
+ "ValidTo": "2023-01-27T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-01-25T00:00:00",
+ "ValidTo": "2023-01-26T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-01-24T00:00:00",
+ "ValidTo": "2023-01-25T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-01-23T00:00:00",
+ "ValidTo": "2023-01-24T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-01-22T00:00:00",
+ "ValidTo": "2023-01-23T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ },
+ {
+ "ValidFrom": "2023-01-21T00:00:00",
+ "ValidTo": "2023-01-22T00:00:00",
+ "ChargeTypeCode": "46",
+ "Price1": 0.2581,
+ "Price2": 0.2581,
+ "Price3": 0.2581,
+ "Price4": 0.2581,
+ "Price5": 0.2581,
+ "Price6": 0.2581,
+ "Price7": 0.7742,
+ "Price8": 0.7742,
+ "Price9": 0.7742,
+ "Price10": 0.7742,
+ "Price11": 0.7742,
+ "Price12": 0.7742,
+ "Price13": 0.7742,
+ "Price14": 0.7742,
+ "Price15": 0.7742,
+ "Price16": 0.7742,
+ "Price17": 0.7742,
+ "Price18": 2.3227,
+ "Price19": 2.3227,
+ "Price20": 2.3227,
+ "Price21": 2.3227,
+ "Price22": 0.7742,
+ "Price23": 0.7742,
+ "Price24": 0.7742
+ }
+ ]
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/action/ElectricityTaxes.json b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/action/ElectricityTaxes.json
new file mode 100644
index 000000000..5ee45ea67
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/action/ElectricityTaxes.json
@@ -0,0 +1,45 @@
+{
+ "total": 1,
+ "filters": "{\"GLN_Number\":[\"5790000432752\"],\"ChargeType\":[\"D03\"],\"Note\":[\"Elafgift\"]}",
+ "dataset": "DatahubPricelist",
+ "records": [
+ {
+ "ChargeOwner": "Energinet Systemansvar A/S (SYO)",
+ "GLN_Number": "5790000432752",
+ "ChargeType": "D03",
+ "ChargeTypeCode": "EA-001",
+ "Note": "Elafgift",
+ "Description": "Elafgiften",
+ "ValidFrom": "2023-01-01T00:00:00",
+ "ValidTo": "2023-07-01T00:00:00",
+ "VATClass": "D02",
+ "Price1": 0.008,
+ "Price2": null,
+ "Price3": null,
+ "Price4": null,
+ "Price5": null,
+ "Price6": null,
+ "Price7": null,
+ "Price8": null,
+ "Price9": null,
+ "Price10": null,
+ "Price11": null,
+ "Price12": null,
+ "Price13": null,
+ "Price14": null,
+ "Price15": null,
+ "Price16": null,
+ "Price17": null,
+ "Price18": null,
+ "Price19": null,
+ "Price20": null,
+ "Price21": null,
+ "Price22": null,
+ "Price23": null,
+ "Price24": null,
+ "TransparentInvoicing": 1,
+ "TaxIndicator": 1,
+ "ResolutionDuration": "P1D"
+ }
+ ]
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/action/NetTariffs.json b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/action/NetTariffs.json
new file mode 100644
index 000000000..1af9754e1
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/action/NetTariffs.json
@@ -0,0 +1,37 @@
+{
+ "total": 1,
+ "filters": "{\"ChargeTypeCode\":[\"CD\",\"CD R\"],\"ChargeType\":[\"D03\"],\"GLN_Number\":[\"5790001089030\"]}",
+ "limit": 100,
+ "dataset": "DatahubPricelist",
+ "records": [
+ {
+ "ValidFrom": "2023-01-01T00:00:00",
+ "ValidTo": "2023-04-01T00:00:00",
+ "ChargeTypeCode": "CD",
+ "Price1": 0.432225,
+ "Price2": 0.432225,
+ "Price3": 0.432225,
+ "Price4": 0.432225,
+ "Price5": 0.432225,
+ "Price6": 0.432225,
+ "Price7": 0.432225,
+ "Price8": 0.432225,
+ "Price9": 0.432225,
+ "Price10": 0.432225,
+ "Price11": 0.432225,
+ "Price12": 0.432225,
+ "Price13": 0.432225,
+ "Price14": 0.432225,
+ "Price15": 0.432225,
+ "Price16": 0.432225,
+ "Price17": 0.432225,
+ "Price18": 1.05619,
+ "Price19": 1.05619,
+ "Price20": 1.05619,
+ "Price21": 0.432225,
+ "Price22": 0.432225,
+ "Price23": 0.432225,
+ "Price24": 0.432225
+ }
+ ]
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/action/SpotPrices20230204.json b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/action/SpotPrices20230204.json
new file mode 100644
index 000000000..76423da41
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/action/SpotPrices20230204.json
@@ -0,0 +1,142 @@
+[
+ {
+ "hourStart": "2023-02-04T12:00:00Z",
+ "spotPrice": 0.992840027
+ },
+ {
+ "hourStart": "2023-02-04T13:00:00Z",
+ "spotPrice": 0.998200012
+ },
+ {
+ "hourStart": "2023-02-04T14:00:00Z",
+ "spotPrice": 1.054180054
+ },
+ {
+ "hourStart": "2023-02-04T15:00:00Z",
+ "spotPrice": 1.156540039
+ },
+ {
+ "hourStart": "2023-02-04T16:00:00Z",
+ "spotPrice": 1.267680054
+ },
+ {
+ "hourStart": "2023-02-04T17:00:00Z",
+ "spotPrice": 1.370939941
+ },
+ {
+ "hourStart": "2023-02-04T18:00:00Z",
+ "spotPrice": 1.339670044
+ },
+ {
+ "hourStart": "2023-02-04T19:00:00Z",
+ "spotPrice": 1.24973999
+ },
+ {
+ "hourStart": "2023-02-04T20:00:00Z",
+ "spotPrice": 1.177160034
+ },
+ {
+ "hourStart": "2023-02-04T21:00:00Z",
+ "spotPrice": 0.979809998
+ },
+ {
+ "hourStart": "2023-02-04T22:00:00Z",
+ "spotPrice": 0.804200012
+ },
+ {
+ "hourStart": "2023-02-04T23:00:00Z",
+ "spotPrice": 0.82826001
+ },
+ {
+ "hourStart": "2023-02-05T00:00:00Z",
+ "spotPrice": 0.777280029
+ },
+ {
+ "hourStart": "2023-02-05T01:00:00Z",
+ "spotPrice": 0.771549988
+ },
+ {
+ "hourStart": "2023-02-05T02:00:00Z",
+ "spotPrice": 0.757559998
+ },
+ {
+ "hourStart": "2023-02-05T03:00:00Z",
+ "spotPrice": 0.751599976
+ },
+ {
+ "hourStart": "2023-02-05T04:00:00Z",
+ "spotPrice": 0.76373999
+ },
+ {
+ "hourStart": "2023-02-05T05:00:00Z",
+ "spotPrice": 0.764700012
+ },
+ {
+ "hourStart": "2023-02-05T06:00:00Z",
+ "spotPrice": 0.784650024
+ },
+ {
+ "hourStart": "2023-02-05T07:00:00Z",
+ "spotPrice": 0.79551001
+ },
+ {
+ "hourStart": "2023-02-05T08:00:00Z",
+ "spotPrice": 0.805789978
+ },
+ {
+ "hourStart": "2023-02-05T09:00:00Z",
+ "spotPrice": 0.807789978
+ },
+ {
+ "hourStart": "2023-02-05T10:00:00Z",
+ "spotPrice": 0.796849976
+ },
+ {
+ "hourStart": "2023-02-05T11:00:00Z",
+ "spotPrice": 0.756289978
+ },
+ {
+ "hourStart": "2023-02-05T12:00:00Z",
+ "spotPrice": 0.749369995
+ },
+ {
+ "hourStart": "2023-02-05T13:00:00Z",
+ "spotPrice": 0.7915
+ },
+ {
+ "hourStart": "2023-02-05T14:00:00Z",
+ "spotPrice": 0.838830017
+ },
+ {
+ "hourStart": "2023-02-05T15:00:00Z",
+ "spotPrice": 0.892859985
+ },
+ {
+ "hourStart": "2023-02-05T16:00:00Z",
+ "spotPrice": 1.01997998
+ },
+ {
+ "hourStart": "2023-02-05T17:00:00Z",
+ "spotPrice": 0.99452002
+ },
+ {
+ "hourStart": "2023-02-05T18:00:00Z",
+ "spotPrice": 0.976140015
+ },
+ {
+ "hourStart": "2023-02-05T19:00:00Z",
+ "spotPrice": 0.923669983
+ },
+ {
+ "hourStart": "2023-02-05T20:00:00Z",
+ "spotPrice": 0.906700012
+ },
+ {
+ "hourStart": "2023-02-05T21:00:00Z",
+ "spotPrice": 0.931859985
+ },
+ {
+ "hourStart": "2023-02-05T22:00:00Z",
+ "spotPrice": 0.941159973
+ }
+]
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/action/SpotPrices20230205.json b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/action/SpotPrices20230205.json
new file mode 100644
index 000000000..ce12925c9
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/action/SpotPrices20230205.json
@@ -0,0 +1,142 @@
+[
+ {
+ "hourStart": "2023-02-05T12:00:00Z",
+ "spotPrice": 0.749609985
+ },
+ {
+ "hourStart": "2023-02-05T13:00:00Z",
+ "spotPrice": 0.79173999
+ },
+ {
+ "hourStart": "2023-02-05T14:00:00Z",
+ "spotPrice": 0.839090027
+ },
+ {
+ "hourStart": "2023-02-05T15:00:00Z",
+ "spotPrice": 0.893140015
+ },
+ {
+ "hourStart": "2023-02-05T16:00:00Z",
+ "spotPrice": 1.020299988
+ },
+ {
+ "hourStart": "2023-02-05T17:00:00Z",
+ "spotPrice": 0.994840027
+ },
+ {
+ "hourStart": "2023-02-05T18:00:00Z",
+ "spotPrice": 0.976450012
+ },
+ {
+ "hourStart": "2023-02-05T19:00:00Z",
+ "spotPrice": 0.923960022
+ },
+ {
+ "hourStart": "2023-02-05T20:00:00Z",
+ "spotPrice": 0.90698999
+ },
+ {
+ "hourStart": "2023-02-05T21:00:00Z",
+ "spotPrice": 0.932150024
+ },
+ {
+ "hourStart": "2023-02-05T22:00:00Z",
+ "spotPrice": 0.941460022
+ },
+ {
+ "hourStart": "2023-02-05T23:00:00Z",
+ "spotPrice": 1.07947998
+ },
+ {
+ "hourStart": "2023-02-06T00:00:00Z",
+ "spotPrice": 1.070030029
+ },
+ {
+ "hourStart": "2023-02-06T01:00:00Z",
+ "spotPrice": 1.082540039
+ },
+ {
+ "hourStart": "2023-02-06T02:00:00Z",
+ "spotPrice": 1.057819946
+ },
+ {
+ "hourStart": "2023-02-06T03:00:00Z",
+ "spotPrice": 1.0430
+ },
+ {
+ "hourStart": "2023-02-06T04:00:00Z",
+ "spotPrice": 1.10873999
+ },
+ {
+ "hourStart": "2023-02-06T05:00:00Z",
+ "spotPrice": 1.307810059
+ },
+ {
+ "hourStart": "2023-02-06T06:00:00Z",
+ "spotPrice": 1.493780029
+ },
+ {
+ "hourStart": "2023-02-06T07:00:00Z",
+ "spotPrice": 1.588630005
+ },
+ {
+ "hourStart": "2023-02-06T08:00:00Z",
+ "spotPrice": 1.493780029
+ },
+ {
+ "hourStart": "2023-02-06T09:00:00Z",
+ "spotPrice": 1.377869995
+ },
+ {
+ "hourStart": "2023-02-06T10:00:00Z",
+ "spotPrice": 1.338859985
+ },
+ {
+ "hourStart": "2023-02-06T11:00:00Z",
+ "spotPrice": 1.256069946
+ },
+ {
+ "hourStart": "2023-02-06T12:00:00Z",
+ "spotPrice": 1.199790039
+ },
+ {
+ "hourStart": "2023-02-06T13:00:00Z",
+ "spotPrice": 1.220189941
+ },
+ {
+ "hourStart": "2023-02-06T14:00:00Z",
+ "spotPrice": 1.270589966
+ },
+ {
+ "hourStart": "2023-02-06T15:00:00Z",
+ "spotPrice": 1.353449951
+ },
+ {
+ "hourStart": "2023-02-06T16:00:00Z",
+ "spotPrice": 1.481050049
+ },
+ {
+ "hourStart": "2023-02-06T17:00:00Z",
+ "spotPrice": 1.589449951
+ },
+ {
+ "hourStart": "2023-02-06T18:00:00Z",
+ "spotPrice": 1.52898999
+ },
+ {
+ "hourStart": "2023-02-06T19:00:00Z",
+ "spotPrice": 1.386280029
+ },
+ {
+ "hourStart": "2023-02-06T20:00:00Z",
+ "spotPrice": 1.239400024
+ },
+ {
+ "hourStart": "2023-02-06T21:00:00Z",
+ "spotPrice": 1.135319946
+ },
+ {
+ "hourStart": "2023-02-06T22:00:00Z",
+ "spotPrice": 1.14648999
+ }
+]
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/action/SystemTariffs.json b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/action/SystemTariffs.json
new file mode 100644
index 000000000..8b1d42b98
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/action/SystemTariffs.json
@@ -0,0 +1,36 @@
+{
+ "total": 1,
+ "filters": "{\"ChargeType\":[\"D03\"],\"GLN_Number\":[\"5790000432752\"],\"Note\":[\"Systemtarif\"]}",
+ "dataset": "DatahubPricelist",
+ "records": [
+ {
+ "ValidFrom": "2023-01-01T00:00:00",
+ "ValidTo": null,
+ "ChargeTypeCode": "41000",
+ "Price1": 0.054,
+ "Price2": null,
+ "Price3": null,
+ "Price4": null,
+ "Price5": null,
+ "Price6": null,
+ "Price7": null,
+ "Price8": null,
+ "Price9": null,
+ "Price10": null,
+ "Price11": null,
+ "Price12": null,
+ "Price13": null,
+ "Price14": null,
+ "Price15": null,
+ "Price16": null,
+ "Price17": null,
+ "Price18": null,
+ "Price19": null,
+ "Price20": null,
+ "Price21": null,
+ "Price22": null,
+ "Price23": null,
+ "Price24": null
+ }
+ ]
+}
diff --git a/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/action/TransmissionNetTariffs.json b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/action/TransmissionNetTariffs.json
new file mode 100644
index 000000000..c2ed99340
--- /dev/null
+++ b/bundles/org.openhab.binding.energidataservice/src/test/resources/org/openhab/binding/energidataservice/internal/action/TransmissionNetTariffs.json
@@ -0,0 +1,36 @@
+{
+ "total": 1,
+ "filters": "{\"ChargeType\":[\"D03\"],\"GLN_Number\":[\"5790000432752\"],\"Note\":[\"Transmissions nettarif\"]}",
+ "dataset": "DatahubPricelist",
+ "records": [
+ {
+ "ValidFrom": "2023-01-01T00:00:00",
+ "ValidTo": null,
+ "ChargeTypeCode": "40000",
+ "Price1": 0.058,
+ "Price2": null,
+ "Price3": null,
+ "Price4": null,
+ "Price5": null,
+ "Price6": null,
+ "Price7": null,
+ "Price8": null,
+ "Price9": null,
+ "Price10": null,
+ "Price11": null,
+ "Price12": null,
+ "Price13": null,
+ "Price14": null,
+ "Price15": null,
+ "Price16": null,
+ "Price17": null,
+ "Price18": null,
+ "Price19": null,
+ "Price20": null,
+ "Price21": null,
+ "Price22": null,
+ "Price23": null,
+ "Price24": null
+ }
+ ]
+}
diff --git a/bundles/pom.xml b/bundles/pom.xml
index cf42108c5..4a2b07e05 100644
--- a/bundles/pom.xml
+++ b/bundles/pom.xml
@@ -125,6 +125,7 @@
org.openhab.binding.elerotransmitterstick
org.openhab.binding.elroconnects
org.openhab.binding.energenie
+ org.openhab.binding.energidataservice
org.openhab.binding.enigma2
org.openhab.binding.enocean
org.openhab.binding.enphase