[amberelectric] Initial contribution (#16850)

Signed-off-by: Paul Smedley <paul@smedley.id.au>
Signed-off-by: Ciprian Pascu <contact@ciprianpascu.ro>
This commit is contained in:
Paul Smedley 2024-06-15 19:10:01 +09:30 committed by Ciprian Pascu
parent 7658c1cef1
commit 44d749c379
18 changed files with 792 additions and 0 deletions

View File

@ -22,6 +22,7 @@
/bundles/org.openhab.binding.allplay/ @dominicdesu
/bundles/org.openhab.binding.amazondashbutton/ @openhab/add-ons-maintainers
/bundles/org.openhab.binding.amazonechocontrol/ @mgeramb
/bundles/org.openhab.binding.amberelectric/ @psmedley
/bundles/org.openhab.binding.ambientweather/ @mhilbush
/bundles/org.openhab.binding.amplipi/ @kaikreuzer
/bundles/org.openhab.binding.androiddebugbridge/ @GiviMAD

View File

@ -101,6 +101,11 @@
<artifactId>org.openhab.binding.amazonechocontrol</artifactId>
<version>${project.version}</version>
</dependency>
<dependency>
<groupId>org.openhab.addons.bundles</groupId>
<artifactId>org.openhab.binding.amberelectric</artifactId>
<version>${project.version}</version>
</dependency>
<dependency>
<groupId>org.openhab.addons.bundles</groupId>
<artifactId>org.openhab.binding.ambientweather</artifactId>

View File

@ -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

View File

@ -0,0 +1,69 @@
# Amber Electric Binding
A binding that supports the Australian energy retailer Amber's API (<https://www.amber.com.au/>) and provides data on the current pricing for buying and selling power, as well as the current level of renewables in the NEM.
## Supported Things
- `service` Amber Electric API
## Discovery
The binding does not support auto discovery.
## Thing Configuration
As a minimum, the IP address is needed:
- `apiKey` - The API key from the 'Developer' section of <https://apps.amber.com.au>
- 'nmi' optional - the NMI for your property. Required if you have multiple properties with Amber
- 'refresh' the refresh rate for querying the API.
## Channels
| channel id | type | description |
|------------------------|----------------------|---------------------------------------------------------------------------------|
| electricity-price | Number:EnergyPrice | Current price to import power from the grid
| controlled-load-price | Number:EnergyPrice | Current price to import power for Controlled Load
| feed-in-price | Number:EnergyPrice | Current price to export power to the grid
| electricity-status | String | Current price status of grid import
| controlled-load-status | String | Current price status of controlled load import
| feed-in-status | String | Current price status of Feed-In
| nem-time | String | NEM time of last pricing update
| renewables | Number:Dimensionless | Current level of renewables in the grid
| spike | Switch | Report if the grid has a current price spike
## Full Example
### `amberelectric.things`:
```java
amberelectric:service:AmberElectric [ apiKey="psk_xxxxxxxxxxxxxxxxxxxx" ]
```
### `amberelectric.items`:
```java
Number:EnergyPrice AmberElectric_ElectricityPrice { channel="amberelectric:service:AmberElectric:electricity-price" }
Number:EnergyPrice AmberElectric_ControlledLoadPrice { channel="amberelectric:service:AmberElectric:controlled-load-price" }
Number:EnergyPrice AmberElectric_FeedInPrice { channel="amberelectric:service:AmberElectric:feed-in-price" }
String AmberElectric_ElectricityStatus { channel="amberelectric:service:AmberElectric:electricity-status" }
String AmberElectric_ControlledLoadStatus { channel="amberelectric:service:AmberElectric:controlled-load-status" }
String AmberElectric_FeedInStatus { channel="amberelectric:service:AmberElectric:feed-in-status" }
String AmberElectric_nemtime { channel="amberelectric:service:AmberElectric:nem-time" }
Number AmberElectric_Renewables { channel="amberelectric:service:AmberElectric:renewables" }
Switch AmberElectric_Spike { channel="amberelectric:service:AmberElectric:spike" }
```
### `amberelectric.sitemap`:
```perl
Text item=AmberElectric_ElectricityPrice label="Electricity Price"
Text item=AmberElectric_ControlledLoadPrice label="Controlled Load Price"
Text item=AmberElectric_FeedInPrice label="Feed-In Price"
Text item=AmberElectric_ElectricityStatus label="Electricity Price Status"
Text item=AmberElectric_ControlledLoadStatus label="Controlled Load Price Status"
Text item=AmberElectric_FeedInStatus label="Feed-In Price Status"
Text item=AmberElectric_nemtime label="Current time of NEM pricing"
Text item=AmberElectric_Renewables label="Renewables Level"
Switch item=AmberElectric_Spike label="Spike Status"
```

View File

@ -0,0 +1,17 @@
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>org.openhab.addons.bundles</groupId>
<artifactId>org.openhab.addons.reactor.bundles</artifactId>
<version>4.2.0-SNAPSHOT</version>
</parent>
<artifactId>org.openhab.binding.amberelectric</artifactId>
<name>openHAB Add-ons :: Bundles :: Amber Electric Binding</name>
</project>

View File

@ -0,0 +1,9 @@
<?xml version="1.0" encoding="UTF-8"?>
<features name="org.openhab.binding.amberelectric-${project.version}" xmlns="http://karaf.apache.org/xmlns/features/v1.4.0">
<repository>mvn:org.openhab.core.features.karaf/org.openhab.core.features.karaf.openhab-core/${ohc.version}/xml/features</repository>
<feature name="openhab-binding-amberelectric" description="Amber Electric Binding" version="${project.version}">
<feature>openhab-runtime-base</feature>
<bundle start-level="80">mvn:org.openhab.addons.bundles/org.openhab.binding.amberelectric/${project.version}</bundle>
</feature>
</features>

View File

@ -0,0 +1,46 @@
/**
* Copyright (c) 2010-2024 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.amberelectric.internal;
import java.util.Set;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.openhab.core.thing.ThingTypeUID;
/**
* The {@link AmberElectricBindingConstants} class defines common constants, which are
* used across the whole binding.
*
* @author Paul Smedley - Initial contribution
*/
@NonNullByDefault
public class AmberElectricBindingConstants {
private static final String BINDING_ID = "amberelectric";
// List of all Thing Type UIDs
public static final ThingTypeUID AMBERELECTRIC_THING = new ThingTypeUID(BINDING_ID, "service");
// List of all Channel ids
public static final String CHANNEL_ELECTRICITY_PRICE = "electricity-price";
public static final String CHANNEL_CONTROLLED_LOAD_PRICE = "controlled-load-price";
public static final String CHANNEL_FEED_IN_PRICE = "feed-in-price";
public static final String CHANNEL_ELECTRICITY_STATUS = "electricity-status";
public static final String CHANNEL_CONTROLLED_LOAD_STATUS = "controlled-load-status";
public static final String CHANNEL_FEED_IN_STATUS = "feed-in-status";
public static final String CHANNEL_NEM_TIME = "nem-time";
public static final String CHANNEL_RENEWABLES = "renewables";
public static final String CHANNEL_SPIKE = "spike";
public static final Set<ThingTypeUID> SUPPORTED_THING_TYPES_UIDS = Set.of(AMBERELECTRIC_THING);
}

View File

@ -0,0 +1,38 @@
/**
* Copyright (c) 2010-2024 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.amberelectric.internal;
import org.eclipse.jdt.annotation.NonNullByDefault;
/**
* Exception for when an unexpected response is received from the AmberAPI.
*
* @author Paul Smedley - Initial contribution
*
*/
@NonNullByDefault
public class AmberElectricCommunicationException extends Exception {
private static final long serialVersionUID = 529232811860854017L;
public AmberElectricCommunicationException(String message) {
super(message);
}
public AmberElectricCommunicationException(Throwable ex) {
super(ex);
}
public AmberElectricCommunicationException(String message, Throwable cause) {
super(message, cause);
}
}

View File

@ -0,0 +1,27 @@
/**
* Copyright (c) 2010-2024 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.amberelectric.internal;
import org.eclipse.jdt.annotation.NonNullByDefault;
/**
* The AmberElectricConfiguration class contains fields mapping thing configuration parameters.
*
* @author Paul Smedley - Initial contribution
*/
@NonNullByDefault
public class AmberElectricConfiguration {
public String apiKey = "";
public String nmi = "";
public long refresh = 60;
}

View File

@ -0,0 +1,168 @@
/**
* Copyright (c) 2010-2024 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.amberelectric.internal;
import java.io.IOException;
import java.util.concurrent.Future;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;
import javax.measure.Unit;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
import org.openhab.binding.amberelectric.internal.api.CurrentPrices;
import org.openhab.binding.amberelectric.internal.api.Sites;
import org.openhab.core.config.core.Configuration;
import org.openhab.core.library.types.DecimalType;
import org.openhab.core.library.types.OnOffType;
import org.openhab.core.library.types.QuantityType;
import org.openhab.core.library.types.StringType;
import org.openhab.core.library.unit.CurrencyUnits;
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.types.Command;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* The {@link AmberElectricHandler} is responsible for handling commands, which are
* sent to one of the channels.
*
* @author Paul Smedley - Initial contribution
*/
@NonNullByDefault
public class AmberElectricHandler extends BaseThingHandler {
private final Logger logger = LoggerFactory.getLogger(AmberElectricHandler.class);
private long refreshInterval;
private String apiKey = "";
private String nmi = "";
private String siteID = "";
private @NonNullByDefault({}) AmberElectricConfiguration config;
private @NonNullByDefault({}) AmberElectricWebTargets webTargets;
private @Nullable ScheduledFuture<?> pollFuture;
public AmberElectricHandler(Thing thing) {
super(thing);
}
@Override
public void handleCommand(ChannelUID channelUID, Command command) {
logger.warn("This binding is read only");
}
@Override
public void initialize() {
config = getConfigAs(AmberElectricConfiguration.class);
if (config.apiKey.isBlank()) {
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
"@text/offline.conf-error.no-api-key");
return;
}
webTargets = new AmberElectricWebTargets();
updateStatus(ThingStatus.UNKNOWN);
refreshInterval = config.refresh;
nmi = config.nmi;
apiKey = config.apiKey;
schedulePoll();
}
@Override
public void dispose() {
super.dispose();
stopPoll();
}
private void schedulePoll() {
logger.debug("Scheduling poll every {} s", refreshInterval);
this.pollFuture = scheduler.scheduleWithFixedDelay(this::poll, 0, refreshInterval, TimeUnit.SECONDS);
}
private void poll() {
try {
logger.debug("Polling for state");
pollStatus();
} catch (IOException e) {
logger.debug("Could not connect to AmberAPI", e);
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR, e.getMessage());
} catch (RuntimeException e) {
logger.warn("Unexpected error connecting to AmberAPI", e);
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR, e.getMessage());
}
}
private void stopPoll() {
final Future<?> future = pollFuture;
if (future != null) {
future.cancel(true);
pollFuture = null;
}
}
private void pollStatus() throws IOException {
try {
if (siteID.isEmpty()) {
Sites sites = webTargets.getSites(apiKey, nmi);
// add error handling
siteID = sites.siteid;
Configuration configuration = editConfiguration();
configuration.put("nmi", sites.nmi);
updateConfiguration(configuration);
logger.debug("Detected amber siteid is {}, for nmi {}", sites.siteid, sites.nmi);
}
CurrentPrices currentPrices = webTargets.getCurrentPrices(siteID, apiKey);
final String electricityUnit = " AUD/kWh";
updateStatus(ThingStatus.ONLINE);
Unit<?> unit = CurrencyUnits.getInstance().getUnit("AUD");
if (unit == null) {
logger.trace("Currency AUD is unknown, falling back to DecimalType");
updateState(AmberElectricBindingConstants.CHANNEL_ELECTRICITY_PRICE,
new DecimalType(currentPrices.elecPerKwh / 100));
updateState(AmberElectricBindingConstants.CHANNEL_CONTROLLED_LOAD_PRICE,
new DecimalType(currentPrices.clPerKwh / 100));
updateState(AmberElectricBindingConstants.CHANNEL_FEED_IN_PRICE,
new DecimalType(currentPrices.feedInPerKwh / 100));
} else {
updateState(AmberElectricBindingConstants.CHANNEL_ELECTRICITY_PRICE,
new QuantityType<>(currentPrices.elecPerKwh / 100 + " " + electricityUnit));
updateState(AmberElectricBindingConstants.CHANNEL_CONTROLLED_LOAD_PRICE,
new QuantityType<>(currentPrices.clPerKwh / 100 + " " + electricityUnit));
updateState(AmberElectricBindingConstants.CHANNEL_FEED_IN_PRICE,
new QuantityType<>(currentPrices.feedInPerKwh / 100 + " " + electricityUnit));
}
updateState(AmberElectricBindingConstants.CHANNEL_CONTROLLED_LOAD_STATUS,
new StringType(currentPrices.clStatus));
updateState(AmberElectricBindingConstants.CHANNEL_ELECTRICITY_STATUS,
new StringType(currentPrices.elecStatus));
updateState(AmberElectricBindingConstants.CHANNEL_FEED_IN_STATUS,
new StringType(currentPrices.feedInStatus));
updateState(AmberElectricBindingConstants.CHANNEL_NEM_TIME, new StringType(currentPrices.nemTime));
updateState(AmberElectricBindingConstants.CHANNEL_RENEWABLES, new DecimalType(currentPrices.renewables));
updateState(AmberElectricBindingConstants.CHANNEL_SPIKE,
OnOffType.from(!"none".equals(currentPrices.spikeStatus)));
} catch (AmberElectricCommunicationException e) {
logger.debug("Unexpected error connecting to Amber Electric API", e);
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR, e.getMessage());
}
}
}

View File

@ -0,0 +1,49 @@
/**
* Copyright (c) 2010-2024 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.amberelectric.internal;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
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.annotations.Component;
/**
* The {@link AmberElectricHandlerFactory} is responsible for creating things and thing
* handlers.
*
* @author Paul Smedley - Initial contribution
*/
@Component(service = ThingHandlerFactory.class, configurationPid = "binding.amberelectric")
@NonNullByDefault
public class AmberElectricHandlerFactory extends BaseThingHandlerFactory {
@Override
public boolean supportsThingType(ThingTypeUID thingTypeUID) {
return AmberElectricBindingConstants.SUPPORTED_THING_TYPES_UIDS.contains(thingTypeUID);
}
@Override
protected @Nullable ThingHandler createHandler(Thing thing) {
ThingTypeUID thingTypeUID = thing.getThingTypeUID();
if (thingTypeUID.equals(AmberElectricBindingConstants.AMBERELECTRIC_THING)) {
return new AmberElectricHandler(thing);
}
return null;
}
}

View File

@ -0,0 +1,88 @@
/**
* Copyright (c) 2010-2024 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.amberelectric.internal;
import java.io.IOException;
import java.io.InputStream;
import java.util.Properties;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
import org.openhab.binding.amberelectric.internal.api.CurrentPrices;
import org.openhab.binding.amberelectric.internal.api.Sites;
import org.openhab.core.io.net.http.HttpUtil;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Handles performing the actual HTTP requests for communicating with the AmberAPI.
*
* @author Paul Smedley - Initial Contribution
*
*/
@NonNullByDefault
public class AmberElectricWebTargets {
private static final int TIMEOUT_MS = 30000;
private static final String BASE_URI = "https://api.amber.com.au/v1/";
private final Logger logger = LoggerFactory.getLogger(AmberElectricWebTargets.class);
public AmberElectricWebTargets() {
}
public Sites getSites(String apiKey, String nmi) throws AmberElectricCommunicationException {
String getSitesUri = BASE_URI + "sites";
String response = invoke("GET", getSitesUri, apiKey);
logger.trace("Received response: \"{}\"", response);
return Sites.parse(response, nmi);
}
public CurrentPrices getCurrentPrices(String siteid, String apiKey) throws AmberElectricCommunicationException {
String getCurrentPricesUri = BASE_URI + "sites/" + siteid + "/prices/current";
String response = invoke("GET", getCurrentPricesUri, apiKey);
logger.trace("Received response: \"{}\"", response);
return CurrentPrices.parse(response);
}
protected Properties getHttpHeaders(String accessToken) {
Properties httpHeaders = new Properties();
httpHeaders.put("Authorization", "Bearer " + accessToken);
httpHeaders.put("Content-Type", "application/json");
return httpHeaders;
}
private String invoke(String httpMethod, String uri, String accessToken)
throws AmberElectricCommunicationException {
return invoke(httpMethod, uri, accessToken, null, null);
}
private String invoke(String httpMethod, String uri, String apiKey, @Nullable InputStream content,
@Nullable String contentType) throws AmberElectricCommunicationException {
logger.debug("Calling url: {}", uri);
@Nullable
String response;
try {
response = HttpUtil.executeUrl(httpMethod, uri, getHttpHeaders(apiKey), content, contentType, TIMEOUT_MS);
} catch (IOException ex) {
logger.debug("{}", ex.getLocalizedMessage(), ex);
// Response will also be set to null if parsing in executeUrl fails so we use null here to make the
// error check below consistent.
response = null;
}
if (response == null) {
throw new AmberElectricCommunicationException(
String.format("AmberElectric returned no response while invoking %s", uri));
}
return response;
}
}

View File

@ -0,0 +1,68 @@
/**
* Copyright (c) 2010-2024 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.amberelectric.internal.api;
import org.eclipse.jdt.annotation.NonNullByDefault;
import com.google.gson.JsonArray;
import com.google.gson.JsonObject;
import com.google.gson.JsonParser;
/**
* Container class for Current Pricing, related to amberelectric
*
* @author Paul Smedley <paul@smedley.id.au> - Initial contribution
*
*/
@NonNullByDefault
public class CurrentPrices {
public double elecPerKwh;
public double clPerKwh;
public double feedInPerKwh;
public String elecStatus = "";
public String clStatus = "";
public String feedInStatus = "";
public double renewables;
public String spikeStatus = "";
public String nemTime = "";
private CurrentPrices() {
}
public static CurrentPrices parse(String response) {
/* parse json string */
JsonArray jsonArray = JsonParser.parseString(response).getAsJsonArray();
JsonObject jsonObject = jsonArray.get(0).getAsJsonObject();
CurrentPrices currentprices = new CurrentPrices();
currentprices.nemTime = jsonObject.get("nemTime").getAsString();
currentprices.renewables = jsonObject.get("renewables").getAsDouble();
currentprices.spikeStatus = jsonObject.get("spikeStatus").getAsString();
for (int i = 0; i < jsonArray.size(); i++) {
jsonObject = jsonArray.get(i).getAsJsonObject();
if ("general".equals(jsonObject.get("channelType").getAsString())) {
currentprices.elecPerKwh = jsonObject.get("perKwh").getAsDouble();
currentprices.elecStatus = jsonObject.get("descriptor").getAsString();
}
if ("feedIn".equals(jsonObject.get("channelType").getAsString())) {
// Multiple value from API by -1 to make the value match the app
currentprices.feedInPerKwh = -1 * jsonObject.get("perKwh").getAsDouble();
currentprices.feedInStatus = jsonObject.get("descriptor").getAsString();
}
if ("controlledLoad".equals(jsonObject.get("channelType").getAsString())) {
currentprices.clPerKwh = jsonObject.get("perKwh").getAsDouble();
currentprices.clStatus = jsonObject.get("descriptor").getAsString();
}
}
return currentprices;
}
}

View File

@ -0,0 +1,55 @@
/**
* Copyright (c) 2010-2024 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.amberelectric.internal.api;
import org.eclipse.jdt.annotation.NonNullByDefault;
import com.google.gson.JsonArray;
import com.google.gson.JsonObject;
import com.google.gson.JsonParser;
/**
* Class for holding the set of parameters used to read the controller variables.
*
* @author Paul Smedley - Initial Contribution
*
*/
@NonNullByDefault
public class Sites {
public String siteid = "";
public String nmi = "";
private Sites() {
}
public static Sites parse(String response, String nem) {
/* parse json string */
JsonArray jsonArray = JsonParser.parseString(response).getAsJsonArray();
Sites sites = new Sites();
for (int i = 0; i < jsonArray.size(); i++) {
JsonObject jsonObject = jsonArray.get(i).getAsJsonObject();
if (nem.equals(jsonObject.get("nmi").getAsString())) {
sites.siteid = jsonObject.get("id").getAsString();
sites.nmi = jsonObject.get("nmi").getAsString();
}
}
if ((nem.isEmpty()) || (sites.siteid.isEmpty())) { // nem not specified, or not found so we take the first
// siteid
// found
JsonObject jsonObject = jsonArray.get(0).getAsJsonObject();
sites.siteid = jsonObject.get("id").getAsString();
sites.nmi = jsonObject.get("nmi").getAsString();
}
return sites;
}
}

View File

@ -0,0 +1,11 @@
<?xml version="1.0" encoding="UTF-8"?>
<addon:addon id="amberelectric" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:addon="https://openhab.org/schemas/addon/v1.0.0"
xsi:schemaLocation="https://openhab.org/schemas/addon/v1.0.0 https://openhab.org/schemas/addon-1.0.0.xsd">
<type>binding</type>
<name>Amber Electric Binding</name>
<description>This is the binding for Amber Electric.</description>
<connection>cloud</connection>
</addon:addon>

View File

@ -0,0 +1,43 @@
# add-on
addon.amberelectric.name = Amber Electric Binding
addon.amberelectric.description = This is the binding for Amber Electric.
# thing types
thing-type.amberelectric.service.label = Amber Electric
thing-type.amberelectric.service.description = Amber Electric - wholesale access to power prices
thing-type.amberelectric.service.channel.controlled-load-price.label = Current Controlled Load Price
thing-type.amberelectric.service.channel.controlled-load-price.description = Current price to import power for Controlled Load
thing-type.amberelectric.service.channel.controlled-load-status.label = Current Controlled Load Status
thing-type.amberelectric.service.channel.controlled-load-status.description = Current price status of Controlled Load
thing-type.amberelectric.service.channel.feed-in-price.label = Current Feed-In Price
thing-type.amberelectric.service.channel.feed-in-price.description = Current price to export power to the grid
thing-type.amberelectric.service.channel.feed-in-status.label = Current Feed-In Status
thing-type.amberelectric.service.channel.feed-in-status.description = Current price status of Feed-In
# thing types config
thing-type.config.amberelectric.service.apiKey.label = API Key
thing-type.config.amberelectric.service.apiKey.description = API key from the Amber website
thing-type.config.amberelectric.service.nmi.label = NMI
thing-type.config.amberelectric.service.nmi.description = NMI for your address (Optional)
thing-type.config.amberelectric.service.refresh.label = Refresh Interval
thing-type.config.amberelectric.service.refresh.description = Specifies the refresh interval in seconds
# channel types
channel-type.amberelectric.electricity-price.label = Current Electricity Price
channel-type.amberelectric.electricity-price.description = Current price to import power from the grid
channel-type.amberelectric.electricity-status.label = Current Electricity Status
channel-type.amberelectric.electricity-status.description = Current price status of grid import
channel-type.amberelectric.nem-time.label = NEM Time
channel-type.amberelectric.nem-time.description = NEM time of last pricing update
channel-type.amberelectric.renewables.label = Current Renewables
channel-type.amberelectric.renewables.description = Current level of renewables in the grid
channel-type.amberelectric.spike.label = Energy Price Spike
channel-type.amberelectric.spike.description = Report if the grid has a current price spike
# thing status descriptions
offline.conf-error.no-api-key = API key must be set

View File

@ -0,0 +1,84 @@
<?xml version="1.0" encoding="UTF-8"?>
<thing:thing-descriptions bindingId="amberelectric"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:thing="https://openhab.org/schemas/thing-description/v1.0.0"
xsi:schemaLocation="https://openhab.org/schemas/thing-description/v1.0.0 https://openhab.org/schemas/thing-description-1.0.0.xsd">
<thing-type id="service">
<label>Amber Electric</label>
<description>Amber Electric - wholesale access to power prices</description>
<channels>
<channel id="electricity-price" typeId="electricity-price"/>
<channel id="controlled-load-price" typeId="electricity-price">
<label>Current Controlled Load Price</label>
<description>Current price to import power for Controlled Load</description>
</channel>
<channel id="feed-in-price" typeId="electricity-price">
<label>Current Feed-In Price</label>
<description>Current price to export power to the grid</description>
</channel>
<channel id="electricity-status" typeId="electricity-status"/>
<channel id="controlled-load-status" typeId="electricity-status">
<label>Current Controlled Load Status</label>
<description>Current price status of Controlled Load</description>
</channel>
<channel id="feed-in-status" typeId="electricity-status">
<label>Current Feed-In Status</label>
<description>Current price status of Feed-In</description>
</channel>
<channel id="nem-time" typeId="nemtime"/>
<channel id="spike" typeId="spike"/>
<channel id="renewables" typeId="renewables"/>
</channels>
<config-description>
<parameter name="apiKey" type="text" required="true">
<label>API Key</label>
<description>API key from the Amber website</description>
</parameter>
<parameter name="nmi" type="text">
<label>NMI</label>
<description>NMI for your address (Optional)</description>
</parameter>
<parameter name="refresh" type="integer" min="60" unit="s">
<label>Refresh Interval</label>
<description>Specifies the refresh interval in seconds</description>
<default>60</default>
</parameter>
</config-description>
</thing-type>
<channel-type id="electricity-price">
<item-type>Number:EnergyPrice</item-type>
<label>Current Electricity Price</label>
<description>Current price to import power from the grid</description>
<category>Price</category>
<state readOnly="true" pattern="%.3f %unit%"/>
</channel-type>
<channel-type id="electricity-status">
<item-type>String</item-type>
<label>Current Electricity Status</label>
<description>Current price status of grid import</description>
<state readOnly="true" pattern="%s"/>
</channel-type>
<channel-type id="nem-time">
<item-type>String</item-type>
<label>NEM Time</label>
<description>NEM time of last pricing update</description>
<state readOnly="true" pattern="%s"/>
</channel-type>
<channel-type id="renewables">
<item-type unitHint="%">Number:Dimensionless</item-type>
<label>Current Renewables</label>
<description>Current level of renewables in the grid</description>
<state readOnly="true" pattern="%d %unit%"/>
</channel-type>
<channel-type id="spike">
<item-type>Switch</item-type>
<label>Energy Price Spike</label>
<description>Report if the grid has a current price spike</description>
<state readOnly="true"/>
</channel-type>
</thing:thing-descriptions>

View File

@ -55,6 +55,7 @@
<module>org.openhab.binding.allplay</module>
<module>org.openhab.binding.amazondashbutton</module>
<module>org.openhab.binding.amazonechocontrol</module>
<module>org.openhab.binding.amberelectric</module>
<module>org.openhab.binding.ambientweather</module>
<module>org.openhab.binding.amplipi</module>
<module>org.openhab.binding.androiddebugbridge</module>