diff --git a/bom/openhab-addons/pom.xml b/bom/openhab-addons/pom.xml
index 2104eaff37d..fbadb8d5142 100644
--- a/bom/openhab-addons/pom.xml
+++ b/bom/openhab-addons/pom.xml
@@ -376,6 +376,11 @@
org.openhab.binding.dwdunwetter
${project.version}
+
+ org.openhab.addons.bundles
+ org.openhab.binding.easee
+ ${project.version}
+
org.openhab.addons.bundles
org.openhab.binding.ecobee
diff --git a/bundles/org.openhab.binding.easee/NOTICE b/bundles/org.openhab.binding.easee/NOTICE
new file mode 100644
index 00000000000..38d625e3492
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/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.easee/README.md b/bundles/org.openhab.binding.easee/README.md
new file mode 100644
index 00000000000..20e29d538c6
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/README.md
@@ -0,0 +1,178 @@
+# Easee Binding
+
+The Easee binding can be used to retrieve data from the Easee Cloud API and also to control your wallbox via the Cloud API.
+This allows you to dynamically adjust the charge current for your car depending on production of your solar plant.
+
+## Supported Things
+
+This binding provides three thing types:
+
+| Thing/Bridge | Thing Type | Description |
+|---------------------|---------------------|-----------------------------------------------------------------------------------------------|
+| bridge | site | cloud connection to a site within an Easee account |
+| thing | charger | the physical charger which is connected to a circuit within the given site |
+| thing | mastercharger | like the "normal" charger but with additional capability to control the circuit |
+
+
+Basically any Easee wallbox that supports the Cloud API should automatically be supported by this binding.
+
+## Discovery
+
+Auto-discovery is supported and will discover all circuits and chargers assigned to a given site.
+
+## Bridge Configuration
+
+The following configuration parameters are available for the binding/bridge:
+
+| Configuration Parameter | Required | Description |
+|-------------------------|----------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
+| username | yes | The username to login at Easee Cloud service. This should be an e-mail address or phone number. |
+| passord | yes | Your password to login at Easee Cloud service. |
+| siteId | yes | The ID of the site containing the wallbox(es) and circuit(s) that should be integrated into openHAB. The ID of your site can be found via the sites overview (https://easee.cloud/sites). You just need to click one of the sites listed there, the id will be part of the URL which is then opened. It will be a number with typically 6 digits. |
+| dataPollingInterval | no | Interval (seconds) in which live data values are retrieved from the Easee Cloud API. (default = 120) |
+
+## Thing configuration
+
+It is recommended to use auto discovery which does not require further configuration.
+If manual configuration is preferred you need to specify configuration as below.
+
+### Charger
+
+| Configuration Parameter | Required | Description |
+|-------------------------|----------|------------------------------------------------------------------------------------------------------------------------|
+| id | yes | The id of the charger that will be represented by this thing. |
+
+### Mastercharger
+
+| Configuration Parameter | Required | Description |
+|-------------------------|----------|------------------------------------------------------------------------------------------------------------------------|
+| id | yes | The id of the charger that will be represented by this thing. |
+| circuitId | yes | The id of the circuit that is controlled by this charger. |
+
+## Channels
+
+The binding only supports a subset of the available endpoints provided by the Easee Cloud API.
+The tables below show all available channels and which of them are writable.
+The settings that start with "dynamic" can be changed frequently, the others are written to flash and thus should not be changed too often as this could result in damage of your flash.
+
+### Charger Channels
+
+| Channel | Item Type | Writable | Description | Allowed Values (write access) |
+|---------------------------------------------|--------------------------|----------|------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------|
+| state#smartCharging | Switch | no | | |
+| state#cableLocked | Switch | no | | |
+| state#chargerOpMode | Number | no | | |
+| state#totalPower | Number:Power | no | current session total power (all phases) | |
+| state#sessionEnergy | Number:Energy | no | current session | |
+| state#dynamicCircuitCurrentP1 | Number:ElectricCurrent | no | | |
+| state#dynamicCircuitCurrentP2 | Number:ElectricCurrent | no | | |
+| state#dynamicCircuitCurrentP3 | Number:ElectricCurrent | no | | |
+| state#latestPulse | DateTime | no | | |
+| state#chargerFirmware | Number | no | | |
+| state#latestFirmware | Number | no | | |
+| state#voltage | Number:ElectricPotential | no | | |
+| state#outputCurrent | Number:ElectricCurrent | no | | |
+| state#isOnline | Switch | no | | |
+| state#dynamicChargerCurrent | Number:ElectricCurrent | yes | | 0, 6-32 |
+| state#reasonForNoCurrent | Number | no | | |
+| state#lifetimeEnergy | Number:Energy | no | | |
+| state#errorCode | Number | no | | |
+| state#fatalErrorCode | Number | no | | |
+| config#lockCablePermanently | Switch | yes | | true/false |
+| config#authorizationRequired | Switch | yes | | true/false |
+| config#limitToSinglePhaseCharging | Switch | yes | | true/false |
+| config#phaseMode | Number | yes | 1=1phase, 2=auto, 3=3phase | 1-3 |
+| config#maxChargerCurrent | Number:ElectricCurrent | no | write access not yet implemented | |
+| commands#genericCommand | String | yes | Generic Endpoint to send commands | reboot, update_firmware, poll_all, smart_charging, start_charging, stop_charging, pause_charging, resume_charging, toggle_charging, override_schedule |
+| commands#startStop | Switch | yes | Start/Stop Charing, only works with authorization | |
+| commands#pauseResume | Switch | yes | Pause/Resume Charing | |
+| latestSession#sessionEnergy | Number:Energy | no | latest (already ended) session | |
+| latestSession#sessionStart | DateTime | no | | |
+| latestSession#sessionEnd | DateTime | no | | |
+
+### Master Charger Channels
+
+The Master Charger is like the "normal" charger but has some extra channels to control the circuit. These additional channels are listed in the table below.
+
+| Channel | Item Type | Writable | Description | Allowed Values (write access) |
+|---------------------------------------------|--------------------------|----------|------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------|
+| dynamicCurrent#phase1 | Number:ElectricCurrent | no | | |
+| dynamicCurrent#phase2 | Number:ElectricCurrent | no | | |
+| dynamicCurrent#phase3 | Number:ElectricCurrent | no | | |
+| dynamicCurrent#dynamicCurrents | String | yes | read/write only for all phases. | ;; valid values for each phase are 0, 6-32 |
+| settings#maxCircuitCurrentP1 | Number:ElectricCurrent | no | | |
+| settings#maxCircuitCurrentP2 | Number:ElectricCurrent | no | | |
+| settings#maxCircuitCurrentP3 | Number:ElectricCurrent | no | | |
+| settings#maxCurrents | String | yes | read/write only for all phases. | ;; valid values for each phase are 0, 6-32 |
+| settings#offlineMaxCircuitCurrentP1 | Number:ElectricCurrent | no | | |
+| settings#offlineMaxCircuitCurrentP2 | Number:ElectricCurrent | no | | |
+| settings#offlineMaxCircuitCurrentP3 | Number:ElectricCurrent | no | | |
+| settings#offlineMaxCurrents | String | yes | read/write only for all phases. | ;; valid values for each phase are 0, 6-32 |
+| settings#enableIdleCurrent | Switch | yes | | true/false |
+| settings#allowOfflineMaxCircuitCurrent | Switch | no | | |
+
+## Full Example
+
+### Thing
+
+#### Minimum configuration
+
+```
+Bridge easee:site:mysite1 [ username="abc@def.net", password="secret", siteId="123456" ]
+```
+
+#### Manual configuration with two chargers, pollingInterval set to 60 seconds.
+
+```
+Bridge easee:site:mysite1 [ username="abc@def.net", password="secret", siteId="471111", dataPollingInterval=60 ] {
+ Thing mastercharger myCharger1 [ id="EHXXXXX1", circuitId="1234567" ]
+ Thing charger myCharger2 [ id="EHXXXXX2" ]
+}
+```
+
+
+### Items
+
+```
+Number:ElectricCurrent Easee_Circuit_Phase1 "Phase 1" { channel="easee:mastercharger:mysite1:myCharger1:dynamicCurrent#phase1" }
+Number:ElectricCurrent Easee_Circuit_Phase2 "Phase 2" { channel="easee:mastercharger:mysite1:myCharger1:dynamicCurrent#phase2" }
+Number:ElectricCurrent Easee_Circuit_Phase3 "Phase 3" { channel="easee:mastercharger:mysite1:myCharger1:dynamicCurrent#phase3" }
+String Easee_Circuit_Dynamic_Phases "Dynamic Power [MAP(easeePhases.map):%s]" { channel="easee:mastercharger:mysite1:myCharger1:dynamicCurrent#setDynamicCurrents" }
+Switch Easee_Charger_Start_Stop "Start / Stop" { channel="easee:mastercharger:mysite1:myCharger1:commands#startStop" }
+```
+
+### Sitemap
+
+```
+ Switch item=Easee_Circuit_Dynamic_Phases mappings=["0;0;0"="0.00 kW", "6;0;0"="1.44 kW", "7;0;0"="1.68 kW", "8;0;0"="1.92 kW", "9;0;0"="2.16 kW", "10;0;0"="2.40 kW", "16;0;0"="3.72 kW", "16;16;16"="11.1 kW"] icon="energy"
+```
+
+### Mapping
+
+easeePhases.map will make the phase setting more readable.
+
+```
+0;0;0=0.00 kW
+6;0;0=1.44 kW
+7;0;0=1.68 kW
+8;0;0=1.92 kW
+9;0;0=2.16 kW
+10;0;0=2.40 kW
+11;0;0=2.64 kW
+12;0;0=2.88 kW
+13;0;0=3.12 kW
+14;0;0=3.36 kW
+15;0;0=3.60 kW
+16;0;0=3.72 kW
+6;6;6=4.32 kW
+7;7;7=5.04 kW
+8;8;8=5.76 kW
+9;9;9=6.48 kW
+10;10;10=7.20 kW
+11;11;11=7.92 kW
+12;12;12=8.64 kW
+13;13;13=9.36 kW
+14;14;14=10.1 kW
+15;15;15=10.8 kW
+16;16;16=11.1 kW
+```
diff --git a/bundles/org.openhab.binding.easee/pom.xml b/bundles/org.openhab.binding.easee/pom.xml
new file mode 100644
index 00000000000..1a4a24a84b7
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/pom.xml
@@ -0,0 +1,17 @@
+
+
+
+ 4.0.0
+
+
+ org.openhab.addons.bundles
+ org.openhab.addons.reactor.bundles
+ 3.4.0-SNAPSHOT
+
+
+ org.openhab.binding.easee
+
+ openHAB Add-ons :: Bundles :: Easee Binding
+
+
diff --git a/bundles/org.openhab.binding.easee/src/main/feature/feature.xml b/bundles/org.openhab.binding.easee/src/main/feature/feature.xml
new file mode 100644
index 00000000000..743e37d7c9c
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/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.easee/${project.version}
+
+
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/AtomicReferenceTrait.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/AtomicReferenceTrait.java
new file mode 100644
index 00000000000..571e0c36526
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/AtomicReferenceTrait.java
@@ -0,0 +1,59 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal;
+
+import java.util.concurrent.Future;
+import java.util.concurrent.atomic.AtomicReference;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * trait class which contains useful helper methods. Thus, the interface can be implemented and methods are available
+ * within the class.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public interface AtomicReferenceTrait {
+
+ /**
+ * this should usually not called directly. use updateJobReference or cancelJobReference instead
+ *
+ * @param job job to cancel.
+ */
+ default void cancelJob(@Nullable Future> job) {
+ if (job != null) {
+ job.cancel(true);
+ }
+ }
+
+ /**
+ * updates a job reference with a new job. the old job will be cancelled if there is one.
+ *
+ * @param jobReference reference to be updated
+ * @param newJob job to be assigned
+ */
+ default void updateJobReference(AtomicReference<@Nullable Future>> jobReference, Future> newJob) {
+ cancelJob(jobReference.getAndSet(newJob));
+ }
+
+ /**
+ * updates a job reference to null and cancels any existing job which might be assigned to the reference.
+ *
+ * @param jobReference to be updated to null.
+ */
+ default void cancelJobReference(AtomicReference<@Nullable Future>> jobReference) {
+ cancelJob(jobReference.getAndSet(null));
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/EaseeBindingConstants.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/EaseeBindingConstants.java
new file mode 100644
index 00000000000..7d4ebae0142
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/EaseeBindingConstants.java
@@ -0,0 +1,172 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal;
+
+import java.time.Instant;
+import java.util.Set;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.core.thing.ThingTypeUID;
+
+/**
+ * The {@link EaseeBindingConstants} class defines common constants, which are
+ * used across the whole binding.
+ *
+ * @author Alexander Friese - Initial contribution
+ */
+@NonNullByDefault
+public class EaseeBindingConstants {
+
+ public static final String BINDING_ID = "easee";
+
+ // List of main device types
+ public static final String DEVICE_SITE = "site";
+ public static final String DEVICE_MASTER_CHARGER = "mastercharger";
+ public static final String DEVICE_CHARGER = "charger";
+
+ // List of all Thing Type UIDs
+ public static final ThingTypeUID THING_TYPE_SITE = new ThingTypeUID(BINDING_ID, DEVICE_SITE);
+ public static final ThingTypeUID THING_TYPE_MASTER_CHARGER = new ThingTypeUID(BINDING_ID, DEVICE_MASTER_CHARGER);
+ public static final ThingTypeUID THING_TYPE_CHARGER = new ThingTypeUID(BINDING_ID, DEVICE_CHARGER);
+
+ // List of all channel groups
+ public static final String CHANNEL_GROUP_NONE = "";
+ public static final String CHANNEL_GROUP_SITE_INFO = "info";
+ public static final String CHANNEL_GROUP_CHARGER = "charger";
+ public static final String CHANNEL_GROUP_CHARGER_STATE = "state";
+ public static final String CHANNEL_GROUP_CHARGER_CONFIG = "config";
+ public static final String CHANNEL_GROUP_CHARGER_COMMANDS = "commands";
+ public static final String CHANNEL_GROUP_CHARGER_LATEST_SESSION = "latestSession";
+ public static final String CHANNEL_GROUP_CIRCUIT_DYNAMIC_CURRENT = "dynamicCurrent";
+ public static final String CHANNEL_GROUP_CIRCUIT_SETTINGS = "settings";
+
+ // Channel types
+ public static final String CHANNEL_TYPE_SWITCH = "Switch";
+ public static final String CHANNEL_TYPE_VOLT = "Number:ElectricPotential";
+ public static final String CHANNEL_TYPE_AMPERE = "Number:ElectricCurrent";
+ public static final String CHANNEL_TYPE_KWH = "Number:Energy";
+ public static final String CHANNEL_TYPE_KW = "Number:Power";
+ public static final String CHANNEL_TYPE_DATE = "DateTime";
+ public static final String CHANNEL_TYPE_STRING = "String";
+ public static final String CHANNEL_TYPE_NUMBER = "Number";
+
+ public static final String CHANNEL_TYPEPREFIX_RW = "rw";
+
+ public static final String CHANNEL_TYPENAME_INTEGER = "type-integer";
+
+ // Channels with specific handling
+ public static final String CHANNEL_CHARGER_OP_MODE = "chargerOpMode";
+ public static final String CHANNEL_CHARGER_DYNAMIC_CURRENT = "dynamicChargerCurrent";
+ public static final String CHANNEL_CHARGER_REASON_FOR_NO_CURRENT = "reasonForNoCurrent";
+ public static final String CHANNEL_CHARGER_START_STOP = "startStop";
+ public static final String CHANNEL_CHARGER_PAUSE_RESUME = "pauseResume";
+ public static final String CHANNEL_CIRCUIT_DYNAMIC_CURRENTS = "dynamicCurrents";
+ public static final String CHANNEL_CIRCUIT_DYNAMIC_CURRENT_PHASE1 = "phase1";
+ public static final String CHANNEL_CIRCUIT_DYNAMIC_CURRENT_PHASE2 = "phase2";
+ public static final String CHANNEL_CIRCUIT_DYNAMIC_CURRENT_PHASE3 = "phase3";
+ public static final String CHANNEL_CIRCUIT_MAX_CURRENTS = "maxCurrents";
+ public static final String CHANNEL_CIRCUIT_MAX_CURRENT_PHASE1 = "maxCircuitCurrentP1";
+ public static final String CHANNEL_CIRCUIT_MAX_CURRENT_PHASE2 = "maxCircuitCurrentP2";
+ public static final String CHANNEL_CIRCUIT_MAX_CURRENT_PHASE3 = "maxCircuitCurrentP3";
+ public static final String CHANNEL_CIRCUIT_OFFLINE_MAX_CURRENTS = "offlineMaxCurrents";
+ public static final String CHANNEL_CIRCUIT_OFFLINE_MAX_CURRENT_PHASE1 = "offlineMaxCircuitCurrentP1";
+ public static final String CHANNEL_CIRCUIT_OFFLINE_MAX_CURRENT_PHASE2 = "offlineMaxCircuitCurrentP2";
+ public static final String CHANNEL_CIRCUIT_OFFLINE_MAX_CURRENT_PHASE3 = "offlineMaxCircuitCurrentP3";
+
+ // JSON Keys
+ public static final String JSON_KEY_GENERIC_ID = "id";
+ public static final String JSON_KEY_GENERIC_NAME = "name";
+ public static final String JSON_KEY_CIRCUIT_NAME = "panelName";
+ public static final String JSON_KEY_CIRCUIT_ID = "circuitId";
+ public static final String JSON_KEY_CIRCUITS = "circuits";
+ public static final String JSON_KEY_CHARGERS = "chargers";
+ public static final String JSON_KEY_BACK_PLATE = "backPlate";
+ public static final String JSON_KEY_MASTER_BACK_PLATE = "masterBackPlate";
+ public static final String JSON_KEY_MASTER_BACK_PLATE_ID = "masterBackPlateId";
+ public static final String JSON_KEY_ONLINE = "isOnline";
+ public static final String JSON_KEY_SITE_KEY = "siteKey";
+ public static final String JSON_KEY_ERROR_TITLE = "title";
+ public static final String JSON_KEY_AUTH_ACCESS_TOKEN = "accessToken";
+ public static final String JSON_KEY_AUTH_REFRESH_TOKEN = "refreshToken";
+ public static final String JSON_KEY_AUTH_EXPIRES_IN = "expiresIn";
+
+ // Write Commands
+ public static final String COMMAND_CHANGE_CONFIGURATION = "ChangeConfiguration";
+ public static final String COMMAND_SEND_COMMAND = "SendCommand";
+ public static final String COMMAND_SEND_COMMAND_START_STOP = "SendCommandStartStop";
+ public static final String COMMAND_SET_CIRCUIT_SETTINGS = "SetCircuitSettings";
+ public static final String COMMAND_SET_DYNAMIC_CIRCUIT_CURRENTS = "SetDynamicCircuitCurrents";
+ public static final String COMMAND_SET_MAX_CIRCUIT_CURRENTS = "SetMaxCircuitCurrents";
+ public static final String COMMAND_SET_OFFLINE_MAX_CIRCUIT_CURRENTS = "SetOfflineMaxCircuitCurrents";
+
+ // Command Values
+ public static final String CMD_VAL_START_CHARGING = "start_charging";
+ public static final String CMD_VAL_STOP_CHARGING = "stop_charging";
+ public static final String CMD_VAL_PAUSE_CHARGING = "pause_charging";
+ public static final String CMD_VAL_RESUME_CHARGING = "resume_charging";
+
+ // web request constants
+ public static final long WEB_REQUEST_INITIAL_DELAY = 30;
+ public static final long WEB_REQUEST_INTERVAL = 5;
+ public static final int WEB_REQUEST_QUEUE_MAX_SIZE = 20;
+ public static final int WEB_REQUEST_TOKEN_EXPIRY_BUFFER_MINUTES = 5;
+ public static final int WEB_REQUEST_TOKEN_MAX_AGE_MINUTES = 60;
+ public static final String WEB_REQUEST_BEARER_TOKEN_PREFIX = "Bearer ";
+
+ // URLs
+ public static final String LOGIN_URL = "https://api.easee.cloud/api/accounts/login";
+ public static final String REFRESH_TOKEN_URL = "https://api.easee.cloud/api/accounts/refresh_token";
+ public static final String GET_SITE_URL = "https://api.easee.cloud/api/sites/{siteId}";
+ public static final String CHARGER_URL = "https://api.easee.cloud/api/chargers/{id}";
+ public static final String STATE_URL = "https://api.easee.cloud/api/chargers/{id}/state";
+ public static final String GET_CONFIGURATION_URL = "https://api.easee.cloud/api/chargers/{id}/config";
+ public static final String CHANGE_CONFIGURATION_URL = "https://api.easee.cloud/api/chargers/{id}/settings";
+ public static final String COMMANDS_URL = "https://api.easee.cloud/api/chargers/{id}/commands/{command}";
+ public static final String LATEST_CHARGING_SESSION_URL = "https://api.easee.cloud/api/chargers/{id}/sessions/latest";
+ public static final String DYNAMIC_CIRCUIT_CURRENT_URL = "https://api.easee.cloud/api/sites/{siteId}/circuits/{circuitId}/dynamicCurrent";
+ public static final String CIRCUIT_SETTINGS_URL = "https://api.easee.cloud/api/sites/{siteId}/circuits/{circuitId}/settings";
+
+ // Status Keys
+ public static final String STATUS_TOKEN_VALIDATED = "@text/status.token.validated";
+ public static final String STATUS_WAITING_FOR_BRIDGE = "@text/status.waiting.for.bridge";
+ public static final String STATUS_WAITING_FOR_LOGIN = "@text/status.waiting.for.login";
+ public static final String STATUS_NO_VALID_DATA = "@text/status.no.valid.data";
+ public static final String STATUS_NO_CONNECTION = "@text/status.no.connection";
+
+ // other
+ public static final long POLLING_INITIAL_DELAY = 1;
+
+ public static final String GENERIC_YES = "Yes";
+ public static final String GENERIC_NO = "No";
+ public static final int CHARGER_OP_STATE_WAITING = 2;
+ public static final int CHARGER_OP_STATE_CHARGING = 3;
+ public static final int CHARGER_DYNAMIC_CURRENT_PAUSE = 0;
+ public static final int CHARGER_REASON_FOR_NO_CURRENT_PAUSED = 52;
+
+ public static final String THING_CONFIG_ID = "id";
+ public static final String THING_CONFIG_SITE_ID = "siteId";
+ public static final String THING_CONFIG_CIRCUIT_ID = "circuitId";
+ public static final String THING_CONFIG_CIRCUIT_NAME = "circuitName";
+ public static final String THING_CONFIG_IS_MASTER = "isMaster";
+ public static final String THING_CONFIG_BACK_PLATE_ID = "backPlateId";
+ public static final String THING_CONFIG_MASTER_BACK_PLATE_ID = "masterBackPlateId";
+
+ public static final Instant OUTDATED_DATE = Instant.MIN;
+ public static final String DATE_FORMAT = "yyyy-MM-dd HH:mm:ss";
+
+ public static final String PARAMETER_NAME_WRITE_COMMAND = "writeCommand";
+ public static final String PARAMETER_NAME_VALIDATION_REGEXP = "validationExpression";
+
+ public static final Set SUPPORTED_THING_TYPES_UIDS = Set.of(THING_TYPE_SITE,
+ THING_TYPE_MASTER_CHARGER, THING_TYPE_CHARGER);
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/EaseeHandlerFactory.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/EaseeHandlerFactory.java
new file mode 100644
index 00000000000..ec56bf982c3
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/EaseeHandlerFactory.java
@@ -0,0 +1,79 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.eclipse.jetty.client.HttpClient;
+import org.openhab.binding.easee.internal.handler.EaseeChargerHandler;
+import org.openhab.binding.easee.internal.handler.EaseeMasterChargerHandler;
+import org.openhab.binding.easee.internal.handler.EaseeSiteHandler;
+import org.openhab.core.io.net.http.HttpClientFactory;
+import org.openhab.core.thing.Bridge;
+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.Activate;
+import org.osgi.service.component.annotations.Component;
+import org.osgi.service.component.annotations.Reference;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * The {@link EaseeHandlerFactory} is responsible for creating things and thing
+ * handlers.
+ *
+ * @author Alexander Friese - Initial contribution
+ */
+@NonNullByDefault
+@Component(configurationPid = "binding.easee", service = ThingHandlerFactory.class)
+public class EaseeHandlerFactory extends BaseThingHandlerFactory {
+
+ private final Logger logger = LoggerFactory.getLogger(EaseeHandlerFactory.class);
+
+ /**
+ * the shared http client
+ */
+ private final HttpClient httpClient;
+
+ @Activate
+ public EaseeHandlerFactory(final @Reference HttpClientFactory httpClientFactory) {
+ this.httpClient = httpClientFactory.getCommonHttpClient();
+ }
+
+ @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_SITE.equals(thingTypeUID)) {
+ return new EaseeSiteHandler((Bridge) thing, httpClient);
+ } else if (THING_TYPE_MASTER_CHARGER.equals(thingTypeUID)) {
+ return new EaseeMasterChargerHandler(thing);
+ } else if (THING_TYPE_CHARGER.equals(thingTypeUID)) {
+ return new EaseeChargerHandler(thing);
+ } else {
+ logger.warn("Unsupported Thing-Type: {}", thingTypeUID.getAsString());
+ }
+
+ return null;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/Utils.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/Utils.java
new file mode 100644
index 00000000000..586042944ef
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/Utils.java
@@ -0,0 +1,196 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import java.time.Instant;
+import java.time.ZoneId;
+import java.time.ZonedDateTime;
+import java.time.format.DateTimeFormatter;
+import java.time.format.DateTimeParseException;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.easee.internal.model.ConfigurationException;
+import org.openhab.core.thing.Channel;
+import org.openhab.core.thing.type.ChannelTypeUID;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import com.google.gson.JsonArray;
+import com.google.gson.JsonElement;
+import com.google.gson.JsonNull;
+import com.google.gson.JsonObject;
+import com.google.gson.JsonPrimitive;
+
+/**
+ * some helper methods.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public final class Utils {
+ private static final Logger LOGGER = LoggerFactory.getLogger(Utils.class);
+
+ /**
+ * only static methods no instance needed
+ */
+ private Utils() {
+ }
+
+ /**
+ * parses a date string in easee format to ZonedDateTime which is used by Openhab.
+ *
+ * @param date
+ * @return
+ */
+ public static ZonedDateTime parseDate(String date) throws DateTimeParseException {
+ final DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ssX");
+ LOGGER.trace("parsing: {}", date);
+ ZonedDateTime zdt = ZonedDateTime.parse(date, formatter);
+ return zdt;
+ }
+
+ /**
+ * returns a date in a readable format
+ *
+ * @param date
+ * @return
+ */
+ public static String formatDate(Instant date) {
+ final DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")
+ .withZone(ZoneId.systemDefault());
+ return formatter.format(date);
+ }
+
+ /**
+ * get element as JsonObject.
+ *
+ * @param jsonObject
+ * @param key
+ * @return
+ */
+ public static @Nullable JsonObject getAsJsonObject(@Nullable JsonObject jsonObject, String key) {
+ JsonElement element = jsonObject == null ? null : jsonObject.get(key);
+ return (element instanceof JsonObject) ? element.getAsJsonObject() : null;
+ }
+
+ /**
+ * get element as String.
+ *
+ * @param jsonObject
+ * @param key
+ * @return
+ */
+ public static @Nullable String getAsString(@Nullable JsonObject jsonObject, String key) {
+ JsonElement element = jsonObject == null ? null : jsonObject.get(key);
+ String text = null;
+ if (element != null) {
+ if (element instanceof JsonPrimitive) {
+ text = element.getAsString();
+ } else if (element instanceof JsonObject || element instanceof JsonArray) {
+ text = element.toString();
+ }
+ }
+ return text;
+ }
+
+ /**
+ * get element as int.
+ *
+ * @param jsonObject
+ * @param key
+ * @return
+ */
+ public static int getAsInt(@Nullable JsonObject jsonObject, String key) {
+ JsonElement element = jsonObject == null ? null : jsonObject.get(key);
+ return (element instanceof JsonPrimitive) ? element.getAsInt() : 0;
+ }
+
+ /**
+ * get element as boolean.
+ *
+ * @param jsonObject
+ * @param key
+ * @return
+ */
+ public static @Nullable Boolean getAsBool(@Nullable JsonObject jsonObject, String key) {
+ JsonElement json = jsonObject == null ? null : jsonObject.get(key);
+ return (json == null || json instanceof JsonNull) ? null : json.getAsBoolean();
+ }
+
+ /**
+ * retrieves typeID of a channel.
+ *
+ * @param channel
+ * @return typeID or empty string if typeUID is null.
+ */
+ public static String getChannelTypeId(Channel channel) {
+ ChannelTypeUID typeUID = channel.getChannelTypeUID();
+ if (typeUID == null) {
+ return "";
+ }
+ return typeUID.getId();
+ }
+
+ /**
+ * retrieves the validation expression which is assigned to this channel, fallback to a public static, if no
+ * validation
+ * is
+ * defined.
+ *
+ * @param channel
+ * @return the validation expression
+ */
+ public static String getValidationExpression(Channel channel) {
+ String expr = getPropertyOrParameter(channel, PARAMETER_NAME_VALIDATION_REGEXP);
+ if (expr == null) {
+ throw new ConfigurationException(
+ "channel (" + channel.getUID().getId() + ") does not have a validation expression configured");
+ }
+ return expr;
+ }
+
+ /**
+ * retrieves the write API url suffix which is assigned to this channel.
+ *
+ * @param channel
+ * @return the url suffix
+ */
+ public static String getWriteCommand(Channel channel) {
+ String command = getPropertyOrParameter(channel, PARAMETER_NAME_WRITE_COMMAND);
+ if (command == null) {
+ throw new ConfigurationException(
+ "channel (" + channel.getUID().getId() + ") does not have a write command configured");
+ }
+ return command;
+ }
+
+ /**
+ * internal utiliy method which returns a property (if found) or a config parameter (if found) otherwise null
+ *
+ * @param channel
+ * @param name
+ * @return
+ */
+ public static @Nullable String getPropertyOrParameter(Channel channel, String name) {
+ String value = channel.getProperties().get(name);
+ // also eclipse says this cannot be null, it definitely can!
+ if (value == null || value.isEmpty()) {
+ Object obj = channel.getConfiguration().get(name);
+ value = obj == null ? null : obj.toString();
+ }
+ return value;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/AbstractCommand.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/AbstractCommand.java
new file mode 100644
index 00000000000..52579640b4a
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/AbstractCommand.java
@@ -0,0 +1,328 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.WEB_REQUEST_BEARER_TOKEN_PREFIX;
+
+import java.net.SocketTimeoutException;
+import java.net.UnknownHostException;
+import java.nio.ByteBuffer;
+import java.nio.charset.StandardCharsets;
+import java.util.ArrayList;
+import java.util.List;
+import java.util.concurrent.TimeUnit;
+import java.util.concurrent.TimeoutException;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.eclipse.jetty.client.HttpClient;
+import org.eclipse.jetty.client.api.Request;
+import org.eclipse.jetty.client.api.Response;
+import org.eclipse.jetty.client.api.Result;
+import org.eclipse.jetty.client.util.BufferingResponseListener;
+import org.eclipse.jetty.http.HttpHeader;
+import org.eclipse.jetty.http.HttpStatus;
+import org.eclipse.jetty.http.HttpStatus.Code;
+import org.openhab.binding.easee.internal.EaseeBindingConstants;
+import org.openhab.binding.easee.internal.connector.CommunicationStatus;
+import org.openhab.binding.easee.internal.handler.EaseeThingHandler;
+import org.openhab.binding.easee.internal.model.GenericResponseTransformer;
+import org.openhab.binding.easee.internal.model.ValidationException;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import com.google.gson.Gson;
+import com.google.gson.GsonBuilder;
+import com.google.gson.JsonObject;
+import com.google.gson.ToNumberPolicy;
+
+/**
+ * base class for all commands. common logic should be implemented here
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public abstract class AbstractCommand extends BufferingResponseListener implements EaseeCommand {
+
+ public static enum RetryOnFailure {
+ YES,
+ NO;
+ }
+
+ public static enum ProcessFailureResponse {
+ YES,
+ NO;
+ }
+
+ /**
+ * logger
+ */
+ private final Logger logger = LoggerFactory.getLogger(AbstractCommand.class);
+
+ /**
+ * the configuration
+ */
+ protected final EaseeThingHandler handler;
+
+ /**
+ * JSON deserializer
+ */
+ protected final Gson gson;
+
+ /**
+ * status code of fulfilled request
+ */
+ private final CommunicationStatus communicationStatus;
+
+ /**
+ * generic transformer which just transfers all values in a plain map.
+ */
+ private final GenericResponseTransformer transformer;
+
+ /**
+ * retry counter.
+ */
+ private int retries = 0;
+
+ /**
+ * retry active
+ */
+ private final RetryOnFailure retryOnFailure;
+
+ /**
+ * process error response, e.g. set handler offline on error
+ */
+ private final ProcessFailureResponse processFailureResponse;
+
+ /**
+ * allows further processing of the json result data, if set.
+ */
+ private List resultProcessors;
+
+ /**
+ * the constructor
+ */
+ public AbstractCommand(EaseeThingHandler handler, RetryOnFailure retryOnFailure,
+ ProcessFailureResponse processFailureResponse) {
+ this.gson = new GsonBuilder().setObjectToNumberStrategy(ToNumberPolicy.LONG_OR_DOUBLE).create();
+ this.communicationStatus = new CommunicationStatus();
+ this.resultProcessors = new ArrayList<>();
+ this.transformer = new GenericResponseTransformer(handler);
+ this.handler = handler;
+ this.processFailureResponse = processFailureResponse;
+ this.retryOnFailure = retryOnFailure;
+ }
+
+ /**
+ * the constructor
+ */
+ public AbstractCommand(EaseeThingHandler handler, RetryOnFailure retryOnFailure,
+ ProcessFailureResponse processFailureResponse, JsonResultProcessor resultProcessor) {
+ this(handler, retryOnFailure, processFailureResponse);
+ this.resultProcessors.add(resultProcessor);
+ }
+
+ /**
+ * Log request success
+ */
+ @Override
+ public final void onSuccess(@Nullable Response response) {
+ super.onSuccess(response);
+ if (response != null) {
+ communicationStatus.setHttpCode(HttpStatus.getCode(response.getStatus()));
+ logger.debug("HTTP response {}", response.getStatus());
+ }
+ }
+
+ /**
+ * Log request failure
+ */
+ @Override
+ public final void onFailure(@Nullable Response response, @Nullable Throwable failure) {
+ super.onFailure(response, failure);
+ if (failure != null) {
+ logger.info("Request failed: {}", failure.toString());
+ communicationStatus.setError((Exception) failure);
+ if (failure instanceof SocketTimeoutException || failure instanceof TimeoutException) {
+ communicationStatus.setHttpCode(Code.REQUEST_TIMEOUT);
+ } else if (failure instanceof UnknownHostException) {
+ communicationStatus.setHttpCode(Code.BAD_GATEWAY);
+ } else {
+ communicationStatus.setHttpCode(Code.INTERNAL_SERVER_ERROR);
+ }
+ } else {
+ logger.info("Request failed");
+ }
+ if (response != null && response.getStatus() > 0) {
+ communicationStatus.setHttpCode(HttpStatus.getCode(response.getStatus()));
+ }
+ }
+
+ /**
+ * just for logging of content
+ */
+ @Override
+ public void onContent(@Nullable Response response, @Nullable ByteBuffer content) {
+ super.onContent(response, content);
+ logger.debug("received content, length: {}", getContentAsString().length());
+ }
+
+ /**
+ * default handling of successful requests.
+ */
+ @Override
+ public void onComplete(@Nullable Result result) {
+ String json = getContentAsString(StandardCharsets.UTF_8);
+
+ logger.debug("JSON String: {}", json);
+ switch (getCommunicationStatus().getHttpCode()) {
+ case OK:
+ case ACCEPTED:
+ onCompleteCodeOk(json);
+ break;
+ default:
+ onCompleteCodeDefault(json);
+ }
+ }
+
+ /**
+ * handling of result in case of HTTP response OK.
+ *
+ * @param json
+ */
+ protected void onCompleteCodeOk(@Nullable String json) {
+ JsonObject jsonObject = transform(json);
+ if (jsonObject != null) {
+ logger.debug("success");
+ handler.updateChannelStatus(transformer.transform(jsonObject, getChannelGroup()));
+ processResult(jsonObject);
+ }
+ }
+
+ /**
+ * handling of result in default case, this means error handling of http codes where no specific handling applies.
+ *
+ * @param json
+ */
+ protected void onCompleteCodeDefault(@Nullable String json) {
+ JsonObject jsonObject = transform(json);
+ if (jsonObject == null) {
+ jsonObject = new JsonObject();
+ }
+ if (processFailureResponse == ProcessFailureResponse.YES) {
+ processResult(jsonObject);
+ } else {
+ logger.info("command failed, url: {} - code: {} - result: {}", getURL(),
+ getCommunicationStatus().getHttpCode(), jsonObject.get(EaseeBindingConstants.JSON_KEY_ERROR_TITLE));
+ }
+
+ if (retryOnFailure == RetryOnFailure.YES && retries++ < MAX_RETRIES) {
+ handler.enqueueCommand(this);
+ }
+ }
+
+ /**
+ * error safe json transformer.
+ *
+ * @param json
+ * @return
+ */
+ private @Nullable JsonObject transform(@Nullable String json) {
+ if (json != null) {
+ try {
+ return gson.fromJson(json, JsonObject.class);
+ } catch (Exception ex) {
+ logger.debug("JSON could not be parsed: {}\nError: {}", json, ex.getMessage());
+ }
+ }
+ return null;
+ }
+
+ /**
+ * preparation of the request. will call a hook (prepareRequest) that has to be implemented in the subclass to add
+ * content to the request.
+ *
+ * @throws ValidationException
+ */
+ @Override
+ public void performAction(HttpClient asyncclient, String accessToken) throws ValidationException {
+ Request request = asyncclient.newRequest(getURL()).timeout(handler.getBridgeConfiguration().getAsyncTimeout(),
+ TimeUnit.SECONDS);
+
+ // we want to send and receive json only, so explicitely set this!
+ request.header(HttpHeader.ACCEPT, "application/json");
+ request.header(HttpHeader.CONTENT_TYPE, "application/json");
+
+ // this should be the default for Easee Cloud API
+ request.followRedirects(false);
+
+ // add authentication data for every request. Handling this here makes it obsolete to implement for each and
+ // every command
+ if (!accessToken.isBlank()) {
+ request.header(HttpHeader.AUTHORIZATION, WEB_REQUEST_BEARER_TOKEN_PREFIX + accessToken);
+ }
+
+ prepareRequest(request).send(this);
+ }
+
+ /**
+ * @return returns Http Status Code
+ */
+ public CommunicationStatus getCommunicationStatus() {
+ return communicationStatus;
+ }
+
+ /**
+ * calls the registered resultPRocessors.
+ *
+ * @param jsonObject
+ */
+ protected final void processResult(JsonObject jsonObject) {
+ for (JsonResultProcessor processor : resultProcessors) {
+ try {
+ processor.processResult(getCommunicationStatus(), jsonObject);
+ } catch (Exception ex) {
+ // this should not happen
+ logger.warn("Exception caught: {}", ex.getMessage(), ex);
+ }
+ }
+ }
+
+ /**
+ * concrete implementation has to prepare the requests with additional parameters, etc
+ *
+ * @param requestToPrepare the request to prepare
+ * @return prepared Request object
+ * @throws ValidationException
+ */
+ protected abstract Request prepareRequest(Request requestToPrepare) throws ValidationException;
+
+ /**
+ * concrete implementation has to provide the channel group.
+ *
+ * @return
+ */
+ protected abstract String getChannelGroup();
+
+ /**
+ * concrete implementation has to provide the URL
+ *
+ * @return Url
+ */
+ protected abstract String getURL();
+
+ @Override
+ public void registerResultProcessor(JsonResultProcessor resultProcessor) {
+ this.resultProcessors.add(resultProcessor);
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/AbstractWriteCommand.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/AbstractWriteCommand.java
new file mode 100644
index 00000000000..5e87d35665f
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/AbstractWriteCommand.java
@@ -0,0 +1,120 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.CHANNEL_GROUP_NONE;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.api.Request;
+import org.openhab.binding.easee.internal.Utils;
+import org.openhab.binding.easee.internal.handler.EaseeThingHandler;
+import org.openhab.binding.easee.internal.model.ValidationException;
+import org.openhab.core.library.types.OnOffType;
+import org.openhab.core.library.types.QuantityType;
+import org.openhab.core.thing.Channel;
+import org.openhab.core.types.Command;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * base class for all write commands. common logic should be implemented here
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public abstract class AbstractWriteCommand extends AbstractCommand {
+ private final Logger logger = LoggerFactory.getLogger(AbstractWriteCommand.class);
+
+ protected final Channel channel;
+ protected Command command;
+
+ /**
+ * the constructor
+ *
+ * @param config
+ */
+ public AbstractWriteCommand(EaseeThingHandler handler, Channel channel, Command command,
+ RetryOnFailure retryOnFailure, ProcessFailureResponse processFailureResponse) {
+ super(handler, retryOnFailure, processFailureResponse);
+ this.channel = channel;
+ this.command = command;
+ }
+
+ /**
+ * helper method for write commands that extracts value from command.
+ *
+ * @return value as String without unit.
+ */
+ protected String getCommandValue() {
+ if (command instanceof QuantityType>) {
+ // this is necessary because we must not send the unit to the backend
+ return String.valueOf(((QuantityType>) command).doubleValue());
+ } else if (command instanceof OnOffType) {
+ // this is necessary because we must send booleans and not ON/OFF to the backend
+ return String.valueOf(command.equals(OnOffType.ON));
+ } else {
+ return command.toString();
+ }
+ }
+
+ /**
+ * helper that transforms channelId + commandvalue in a JSON string that can be added as content to a POST request.
+ *
+ * @return converted JSON string
+ * @throws ValidationException
+ */
+ protected String getJsonContent() throws ValidationException {
+ Map content = new HashMap(1);
+ content.put(channel.getUID().getIdWithoutGroup(), getCommandValue());
+
+ return gson.toJson(content);
+ }
+
+ @Override
+ protected Request prepareRequest(Request requestToPrepare) throws ValidationException {
+ String channelId = channel.getUID().getIdWithoutGroup();
+ String expr = Utils.getValidationExpression(channel);
+ String value = getCommandValue();
+
+ // quantity types are transformed to double and thus we might have decimals which could cause validation error.
+ // So we will shorten here in case no decimals are needed.
+ if (value.endsWith(".0")) {
+ value = value.substring(0, value.length() - 2);
+ }
+
+ if (value.matches(expr)) {
+ return prepareWriteRequest(requestToPrepare);
+ } else {
+ logger.info("channel '{}' does not allow value '{}' - validation rule '{}'", channelId, value, expr);
+ throw new ValidationException("channel (" + channelId + ") could not be updated due to a validation error");
+ }
+ }
+
+ @Override
+ protected String getChannelGroup() {
+ // this is a pure write command, thus no channel group needed.
+ return CHANNEL_GROUP_NONE;
+ }
+
+ /**
+ * concrete implementation has to prepare the write requests with additional parameters, etc
+ *
+ * @param requestToPrepare the request to prepare
+ * @return prepared Request object
+ * @throws ValidationException
+ */
+ protected abstract Request prepareWriteRequest(Request requestToPrepare) throws ValidationException;
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/EaseeCommand.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/EaseeCommand.java
new file mode 100644
index 00000000000..759416a6481
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/EaseeCommand.java
@@ -0,0 +1,47 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.HttpClient;
+import org.eclipse.jetty.client.api.Response.CompleteListener;
+import org.eclipse.jetty.client.api.Response.ContentListener;
+import org.eclipse.jetty.client.api.Response.FailureListener;
+import org.eclipse.jetty.client.api.Response.SuccessListener;
+import org.openhab.binding.easee.internal.model.ValidationException;
+
+/**
+ * public interface for all commands
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public interface EaseeCommand extends SuccessListener, FailureListener, ContentListener, CompleteListener {
+
+ static final int MAX_RETRIES = 5;
+
+ /**
+ * this method is to be called by the UplinkWebinterface class
+ *
+ * @param asyncclient
+ * @throws ValidationException
+ */
+ void performAction(HttpClient asyncclient, String token) throws ValidationException;
+
+ /**
+ * sets a result processor for json result data
+ *
+ * @param resultProcessor
+ */
+ public void registerResultProcessor(JsonResultProcessor resultProcessor);
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/JsonResultProcessor.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/JsonResultProcessor.java
new file mode 100644
index 00000000000..e6d8c8e9850
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/JsonResultProcessor.java
@@ -0,0 +1,31 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.easee.internal.connector.CommunicationStatus;
+
+import com.google.gson.JsonObject;
+
+/**
+ * functional interface that is intended to provide a function for surther result processing of json data retrieved by a
+ * command.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+@FunctionalInterface
+public interface JsonResultProcessor {
+
+ void processResult(CommunicationStatus status, JsonObject jsonObject);
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/account/Login.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/account/Login.java
new file mode 100644
index 00000000000..efa2c3c38c1
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/account/Login.java
@@ -0,0 +1,85 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command.account;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import java.nio.charset.StandardCharsets;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.eclipse.jetty.client.api.Request;
+import org.eclipse.jetty.client.api.Result;
+import org.eclipse.jetty.client.util.StringContentProvider;
+import org.eclipse.jetty.http.HttpMethod;
+import org.openhab.binding.easee.internal.command.AbstractCommand;
+import org.openhab.binding.easee.internal.handler.EaseeBridgeHandler;
+
+import com.google.gson.JsonObject;
+
+/**
+ * implements the login to the webinterface
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class Login extends AbstractCommand {
+
+ class LoginData {
+ final String userName;
+ final String password;
+
+ public LoginData(String userName, String password) {
+ this.userName = userName;
+ this.password = password;
+ }
+ }
+
+ private final LoginData loginData;
+
+ public Login(EaseeBridgeHandler handler) {
+ // flags do not matter as "onComplete" is overwritten in this class.
+ super(handler, RetryOnFailure.NO, ProcessFailureResponse.NO);
+ loginData = new LoginData(handler.getBridgeConfiguration().getUsername(),
+ handler.getBridgeConfiguration().getPassword());
+ }
+
+ @Override
+ protected Request prepareRequest(Request requestToPrepare) {
+ StringContentProvider cp = new StringContentProvider(gson.toJson(loginData));
+ requestToPrepare.content(cp);
+ requestToPrepare.method(HttpMethod.POST);
+
+ return requestToPrepare;
+ }
+
+ @Override
+ protected String getURL() {
+ return LOGIN_URL;
+ }
+
+ @Override
+ public void onComplete(@Nullable Result result) {
+ String json = getContentAsString(StandardCharsets.UTF_8);
+ JsonObject jsonObject = gson.fromJson(json, JsonObject.class);
+
+ if (jsonObject != null) {
+ processResult(jsonObject);
+ }
+ }
+
+ @Override
+ protected String getChannelGroup() {
+ return CHANNEL_GROUP_NONE;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/account/RefreshToken.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/account/RefreshToken.java
new file mode 100644
index 00000000000..092bd763fdd
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/account/RefreshToken.java
@@ -0,0 +1,61 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command.account;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.REFRESH_TOKEN_URL;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.api.Request;
+import org.eclipse.jetty.client.util.StringContentProvider;
+import org.eclipse.jetty.http.HttpMethod;
+import org.openhab.binding.easee.internal.handler.EaseeBridgeHandler;
+
+/**
+ * implements the refresh of the access token.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class RefreshToken extends Login {
+
+ class RefreshData {
+ final String accessToken;
+ final String refreshToken;
+
+ public RefreshData(String accessToken, String refreshToken) {
+ this.accessToken = accessToken;
+ this.refreshToken = refreshToken;
+ }
+ }
+
+ private final RefreshData refreshData;
+
+ public RefreshToken(EaseeBridgeHandler handler, String accessToken, String refreshToken) {
+ super(handler);
+ refreshData = new RefreshData(accessToken, refreshToken);
+ }
+
+ @Override
+ protected Request prepareRequest(Request requestToPrepare) {
+ StringContentProvider cp = new StringContentProvider(gson.toJson(refreshData));
+ requestToPrepare.content(cp);
+ requestToPrepare.method(HttpMethod.POST);
+
+ return requestToPrepare;
+ }
+
+ @Override
+ protected String getURL() {
+ return REFRESH_TOKEN_URL;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/ChangeConfiguration.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/ChangeConfiguration.java
new file mode 100644
index 00000000000..60720e1b3c5
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/ChangeConfiguration.java
@@ -0,0 +1,54 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command.charger;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.CHANGE_CONFIGURATION_URL;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.api.Request;
+import org.eclipse.jetty.client.util.StringContentProvider;
+import org.eclipse.jetty.http.HttpMethod;
+import org.openhab.binding.easee.internal.command.AbstractWriteCommand;
+import org.openhab.binding.easee.internal.handler.EaseeThingHandler;
+import org.openhab.binding.easee.internal.model.ValidationException;
+import org.openhab.core.thing.Channel;
+import org.openhab.core.types.Command;
+
+/**
+ * implements the settings api call of the charger.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class ChangeConfiguration extends AbstractWriteCommand {
+ private final String url;
+
+ public ChangeConfiguration(EaseeThingHandler handler, String chargerId, Channel channel, Command command) {
+ super(handler, channel, command, RetryOnFailure.YES, ProcessFailureResponse.YES);
+ this.url = CHANGE_CONFIGURATION_URL.replaceAll("\\{id\\}", chargerId);
+ }
+
+ @Override
+ protected Request prepareWriteRequest(Request requestToPrepare) throws ValidationException {
+ StringContentProvider cp = new StringContentProvider(getJsonContent());
+ requestToPrepare.content(cp);
+ requestToPrepare.method(HttpMethod.POST);
+
+ return requestToPrepare;
+ }
+
+ @Override
+ protected String getURL() {
+ return url;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/Charger.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/Charger.java
new file mode 100644
index 00000000000..94420e4eeea
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/Charger.java
@@ -0,0 +1,55 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command.charger;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.api.Request;
+import org.eclipse.jetty.http.HttpMethod;
+import org.openhab.binding.easee.internal.command.AbstractCommand;
+import org.openhab.binding.easee.internal.command.JsonResultProcessor;
+import org.openhab.binding.easee.internal.handler.EaseeThingHandler;
+
+/**
+ * implements the charger api call of the charger.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class Charger extends AbstractCommand {
+ private final String url;
+
+ public Charger(EaseeThingHandler handler, String chargerId, JsonResultProcessor resultProcessor) {
+ // retry does not make much sense as it is a polling command, command should always succeed therefore update
+ // handler on failure.
+ super(handler, RetryOnFailure.NO, ProcessFailureResponse.YES, resultProcessor);
+ this.url = CHARGER_URL.replaceAll("\\{id\\}", chargerId);
+ }
+
+ @Override
+ protected Request prepareRequest(Request requestToPrepare) {
+ requestToPrepare.method(HttpMethod.GET);
+ return requestToPrepare;
+ }
+
+ @Override
+ protected String getURL() {
+ return url;
+ }
+
+ @Override
+ protected String getChannelGroup() {
+ return CHANNEL_GROUP_CHARGER;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/ChargerState.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/ChargerState.java
new file mode 100644
index 00000000000..a73a63992f1
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/ChargerState.java
@@ -0,0 +1,54 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command.charger;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.api.Request;
+import org.eclipse.jetty.http.HttpMethod;
+import org.openhab.binding.easee.internal.command.AbstractCommand;
+import org.openhab.binding.easee.internal.handler.EaseeThingHandler;
+
+/**
+ * implements the state api call of the charger.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class ChargerState extends AbstractCommand {
+ private final String url;
+
+ public ChargerState(EaseeThingHandler handler, String chargerId) {
+ // retry does not make much sense as it is a polling command, command should always succeed therefore update
+ // handler on failure.
+ super(handler, RetryOnFailure.NO, ProcessFailureResponse.YES);
+ this.url = STATE_URL.replaceAll("\\{id\\}", chargerId);
+ }
+
+ @Override
+ protected Request prepareRequest(Request requestToPrepare) {
+ requestToPrepare.method(HttpMethod.GET);
+ return requestToPrepare;
+ }
+
+ @Override
+ protected String getURL() {
+ return url;
+ }
+
+ @Override
+ protected String getChannelGroup() {
+ return CHANNEL_GROUP_CHARGER_STATE;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/GetConfiguration.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/GetConfiguration.java
new file mode 100644
index 00000000000..390c7bfc949
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/GetConfiguration.java
@@ -0,0 +1,54 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command.charger;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.api.Request;
+import org.eclipse.jetty.http.HttpMethod;
+import org.openhab.binding.easee.internal.command.AbstractCommand;
+import org.openhab.binding.easee.internal.handler.EaseeThingHandler;
+
+/**
+ * implements the get configuration api call of the charger.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class GetConfiguration extends AbstractCommand {
+ private final String url;
+
+ public GetConfiguration(EaseeThingHandler handler, String chargerId) {
+ // retry does not make much sense as it is a polling command, command should always succeed therefore update
+ // handler on failure.
+ super(handler, RetryOnFailure.NO, ProcessFailureResponse.YES);
+ this.url = GET_CONFIGURATION_URL.replaceAll("\\{id\\}", chargerId);
+ }
+
+ @Override
+ protected Request prepareRequest(Request requestToPrepare) {
+ requestToPrepare.method(HttpMethod.GET);
+ return requestToPrepare;
+ }
+
+ @Override
+ protected String getURL() {
+ return url;
+ }
+
+ @Override
+ protected String getChannelGroup() {
+ return CHANNEL_GROUP_CHARGER_CONFIG;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/LatestChargingSession.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/LatestChargingSession.java
new file mode 100644
index 00000000000..d509434ceb4
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/LatestChargingSession.java
@@ -0,0 +1,54 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command.charger;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.api.Request;
+import org.eclipse.jetty.http.HttpMethod;
+import org.openhab.binding.easee.internal.command.AbstractCommand;
+import org.openhab.binding.easee.internal.handler.EaseeThingHandler;
+
+/**
+ * implements the latest charging session api call of the charger.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class LatestChargingSession extends AbstractCommand {
+ private final String url;
+
+ public LatestChargingSession(EaseeThingHandler handler, String chargerId) {
+ // retry does not make much sense as it is a polling command, command might fail if no charging sessions are
+ // available, therefore just ignore failure.
+ super(handler, RetryOnFailure.NO, ProcessFailureResponse.NO);
+ this.url = LATEST_CHARGING_SESSION_URL.replaceAll("\\{id\\}", chargerId);
+ }
+
+ @Override
+ protected Request prepareRequest(Request requestToPrepare) {
+ requestToPrepare.method(HttpMethod.GET);
+ return requestToPrepare;
+ }
+
+ @Override
+ protected String getURL() {
+ return url;
+ }
+
+ @Override
+ protected String getChannelGroup() {
+ return CHANNEL_GROUP_CHARGER_LATEST_SESSION;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/SendCommand.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/SendCommand.java
new file mode 100644
index 00000000000..f61d6740df5
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/SendCommand.java
@@ -0,0 +1,69 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command.charger;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.COMMANDS_URL;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.api.Request;
+import org.eclipse.jetty.http.HttpMethod;
+import org.openhab.binding.easee.internal.command.AbstractWriteCommand;
+import org.openhab.binding.easee.internal.handler.EaseeThingHandler;
+import org.openhab.core.thing.Channel;
+import org.openhab.core.types.Command;
+
+/**
+ * implements the command api call of the charger.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class SendCommand extends AbstractWriteCommand {
+ String url = COMMANDS_URL;
+
+ /**
+ * general constructor.
+ *
+ * @param handler the ThingHandler which is responsible for the channel
+ * @param chargerId Id of the charger which is adressed by this command
+ * @param channel the channel that triggered this command
+ * @param command the command to be send
+ */
+ public SendCommand(EaseeThingHandler handler, String chargerId, Channel channel, Command command) {
+ this(handler, channel, command);
+ this.url = COMMANDS_URL.replaceAll("\\{id\\}", chargerId).replaceAll("\\{command\\}", getCommandValue());
+ }
+
+ /**
+ * this constructor should only be used by other command implementations that inherit this class.
+ *
+ * @param handler the ThingHandler which is responsible for the channel
+ * @param channel the channel that triggered this command
+ * @param command the command to be send
+ */
+ SendCommand(EaseeThingHandler handler, Channel channel, Command command) {
+ super(handler, channel, command, RetryOnFailure.YES, ProcessFailureResponse.YES);
+ }
+
+ @Override
+ protected Request prepareWriteRequest(Request requestToPrepare) {
+ requestToPrepare.method(HttpMethod.POST);
+
+ return requestToPrepare;
+ }
+
+ @Override
+ protected String getURL() {
+ return url;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/SendCommandPauseResume.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/SendCommandPauseResume.java
new file mode 100644
index 00000000000..85b55cce011
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/SendCommandPauseResume.java
@@ -0,0 +1,41 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command.charger;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.easee.internal.handler.EaseeThingHandler;
+import org.openhab.core.library.types.OnOffType;
+import org.openhab.core.thing.Channel;
+import org.openhab.core.types.Command;
+
+/**
+ * implements the command api call of the charger.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class SendCommandPauseResume extends SendCommand {
+
+ public SendCommandPauseResume(EaseeThingHandler handler, String chargerId, Channel channel, Command command) {
+ super(handler, channel, command);
+ String value;
+ if (command.equals(OnOffType.ON)) {
+ value = CMD_VAL_PAUSE_CHARGING;
+ } else {
+ value = CMD_VAL_RESUME_CHARGING;
+ }
+ this.url = COMMANDS_URL.replaceAll("\\{id\\}", chargerId).replaceAll("\\{command\\}", value);
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/SendCommandStartStop.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/SendCommandStartStop.java
new file mode 100644
index 00000000000..9d9509a3ee4
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/charger/SendCommandStartStop.java
@@ -0,0 +1,41 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command.charger;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.easee.internal.handler.EaseeThingHandler;
+import org.openhab.core.library.types.OnOffType;
+import org.openhab.core.thing.Channel;
+import org.openhab.core.types.Command;
+
+/**
+ * implements the command api call of the charger.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class SendCommandStartStop extends SendCommand {
+
+ public SendCommandStartStop(EaseeThingHandler handler, String chargerId, Channel channel, Command command) {
+ super(handler, channel, command);
+ String value;
+ if (command.equals(OnOffType.ON)) {
+ value = CMD_VAL_START_CHARGING;
+ } else {
+ value = CMD_VAL_STOP_CHARGING;
+ }
+ this.url = COMMANDS_URL.replaceAll("\\{id\\}", chargerId).replaceAll("\\{command\\}", value);
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/circuit/CircuitSettings.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/circuit/CircuitSettings.java
new file mode 100644
index 00000000000..74fbfbdc434
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/circuit/CircuitSettings.java
@@ -0,0 +1,53 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command.circuit;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.api.Request;
+import org.eclipse.jetty.http.HttpMethod;
+import org.openhab.binding.easee.internal.command.AbstractCommand;
+import org.openhab.binding.easee.internal.handler.EaseeThingHandler;
+
+/**
+ * implements the settings api call of the circuit.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class CircuitSettings extends AbstractCommand {
+ private final String url;
+
+ public CircuitSettings(EaseeThingHandler handler, String circuitId) {
+ super(handler, RetryOnFailure.NO, ProcessFailureResponse.YES);
+ String siteId = handler.getBridgeConfiguration().getSiteId();
+ this.url = CIRCUIT_SETTINGS_URL.replaceAll("\\{siteId\\}", siteId).replaceAll("\\{circuitId\\}", circuitId);
+ }
+
+ @Override
+ protected Request prepareRequest(Request requestToPrepare) {
+ requestToPrepare.method(HttpMethod.GET);
+ return requestToPrepare;
+ }
+
+ @Override
+ protected String getURL() {
+ return url;
+ }
+
+ @Override
+ protected String getChannelGroup() {
+ return CHANNEL_GROUP_CIRCUIT_SETTINGS;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/circuit/DynamicCircuitCurrent.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/circuit/DynamicCircuitCurrent.java
new file mode 100644
index 00000000000..39334ec2c6b
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/circuit/DynamicCircuitCurrent.java
@@ -0,0 +1,54 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command.circuit;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.api.Request;
+import org.eclipse.jetty.http.HttpMethod;
+import org.openhab.binding.easee.internal.command.AbstractCommand;
+import org.openhab.binding.easee.internal.handler.EaseeThingHandler;
+
+/**
+ * implements the dynamicCurrent api call of the circuit.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class DynamicCircuitCurrent extends AbstractCommand {
+ private final String url;
+
+ public DynamicCircuitCurrent(EaseeThingHandler handler, String circuitId) {
+ super(handler, RetryOnFailure.NO, ProcessFailureResponse.YES);
+ String siteId = handler.getBridgeConfiguration().getSiteId();
+ this.url = DYNAMIC_CIRCUIT_CURRENT_URL.replaceAll("\\{siteId\\}", siteId).replaceAll("\\{circuitId\\}",
+ circuitId);
+ }
+
+ @Override
+ protected Request prepareRequest(Request requestToPrepare) {
+ requestToPrepare.method(HttpMethod.GET);
+ return requestToPrepare;
+ }
+
+ @Override
+ protected String getURL() {
+ return url;
+ }
+
+ @Override
+ protected String getChannelGroup() {
+ return CHANNEL_GROUP_CIRCUIT_DYNAMIC_CURRENT;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/circuit/SetCircuitSettings.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/circuit/SetCircuitSettings.java
new file mode 100644
index 00000000000..bac998a2e4a
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/circuit/SetCircuitSettings.java
@@ -0,0 +1,55 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command.circuit;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.CIRCUIT_SETTINGS_URL;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.api.Request;
+import org.eclipse.jetty.client.util.StringContentProvider;
+import org.eclipse.jetty.http.HttpMethod;
+import org.openhab.binding.easee.internal.command.AbstractWriteCommand;
+import org.openhab.binding.easee.internal.handler.EaseeThingHandler;
+import org.openhab.binding.easee.internal.model.ValidationException;
+import org.openhab.core.thing.Channel;
+import org.openhab.core.types.Command;
+
+/**
+ * implements the settings api call of the circuit.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class SetCircuitSettings extends AbstractWriteCommand {
+ private final String url;
+
+ public SetCircuitSettings(EaseeThingHandler handler, Channel channel, Command command, String circuitId) {
+ super(handler, channel, command, RetryOnFailure.YES, ProcessFailureResponse.YES);
+ String siteId = handler.getBridgeConfiguration().getSiteId();
+ this.url = CIRCUIT_SETTINGS_URL.replaceAll("\\{siteId\\}", siteId).replaceAll("\\{circuitId\\}", circuitId);
+ }
+
+ @Override
+ protected Request prepareWriteRequest(Request requestToPrepare) throws ValidationException {
+ requestToPrepare.method(HttpMethod.POST);
+ StringContentProvider cp = new StringContentProvider(getJsonContent());
+ requestToPrepare.content(cp);
+
+ return requestToPrepare;
+ }
+
+ @Override
+ protected String getURL() {
+ return url;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/circuit/SetDynamicCircuitCurrents.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/circuit/SetDynamicCircuitCurrents.java
new file mode 100644
index 00000000000..133a3114934
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/circuit/SetDynamicCircuitCurrents.java
@@ -0,0 +1,84 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command.circuit;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.DYNAMIC_CIRCUIT_CURRENT_URL;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.api.Request;
+import org.eclipse.jetty.client.util.StringContentProvider;
+import org.eclipse.jetty.http.HttpMethod;
+import org.openhab.binding.easee.internal.command.AbstractWriteCommand;
+import org.openhab.binding.easee.internal.handler.EaseeThingHandler;
+import org.openhab.binding.easee.internal.model.ValidationException;
+import org.openhab.core.thing.Channel;
+import org.openhab.core.types.Command;
+
+/**
+ * implements the dynamicCurrent api call of the circuit.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class SetDynamicCircuitCurrents extends AbstractWriteCommand {
+ private static final String PHASE1 = "phase1";
+ private static final String PHASE2 = "phase2";
+ private static final String PHASE3 = "phase3";
+ private final String url;
+
+ public SetDynamicCircuitCurrents(EaseeThingHandler handler, Channel channel, Command command, String circuitId) {
+ super(handler, channel, command, RetryOnFailure.YES, ProcessFailureResponse.YES);
+ String siteId = handler.getBridgeConfiguration().getSiteId();
+ this.url = DYNAMIC_CIRCUIT_CURRENT_URL.replaceAll("\\{siteId\\}", siteId).replaceAll("\\{circuitId\\}",
+ circuitId);
+ }
+
+ /**
+ * helper that transforms channelId + commandvalue in a JSON string that can be added as content to a POST request.
+ *
+ * @return converted JSON string
+ * @throws ValidationException
+ */
+ @Override
+ protected String getJsonContent() throws ValidationException {
+ Map content = new HashMap(3);
+ String rawCommand = getCommandValue();
+ String[] tokens = rawCommand.split(";");
+ if (tokens.length == 3) {
+ content.put(PHASE1, tokens[0]);
+ content.put(PHASE2, tokens[1]);
+ content.put(PHASE3, tokens[2]);
+ } else {
+ throw new ValidationException(
+ "malformed command string, expected: ';;', actual: " + rawCommand);
+ }
+ return gson.toJson(content);
+ }
+
+ @Override
+ protected Request prepareWriteRequest(Request requestToPrepare) throws ValidationException {
+ requestToPrepare.method(HttpMethod.POST);
+ StringContentProvider cp = new StringContentProvider(getJsonContent());
+ requestToPrepare.content(cp);
+
+ return requestToPrepare;
+ }
+
+ @Override
+ protected String getURL() {
+ return url;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/circuit/SetMaxCircuitCurrents.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/circuit/SetMaxCircuitCurrents.java
new file mode 100644
index 00000000000..cfb04ed6357
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/circuit/SetMaxCircuitCurrents.java
@@ -0,0 +1,60 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command.circuit;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.easee.internal.handler.EaseeThingHandler;
+import org.openhab.binding.easee.internal.model.ValidationException;
+import org.openhab.core.thing.Channel;
+import org.openhab.core.types.Command;
+
+/**
+ * implements the settings api call of the circuit.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class SetMaxCircuitCurrents extends SetCircuitSettings {
+ private static final String PHASE1 = "maxCircuitCurrentP1";
+ private static final String PHASE2 = "maxCircuitCurrentP2";
+ private static final String PHASE3 = "maxCircuitCurrentP3";
+
+ public SetMaxCircuitCurrents(EaseeThingHandler handler, Channel channel, Command command, String circuitId) {
+ super(handler, channel, command, circuitId);
+ }
+
+ /**
+ * helper that transforms channelId + commandvalue in a JSON string that can be added as content to a POST request.
+ *
+ * @return converted JSON string
+ * @throws ValidationException
+ */
+ @Override
+ protected String getJsonContent() throws ValidationException {
+ Map content = new HashMap(3);
+ String rawCommand = getCommandValue();
+ String[] tokens = rawCommand.split(";");
+ if (tokens.length == 3) {
+ content.put(PHASE1, tokens[0]);
+ content.put(PHASE2, tokens[1]);
+ content.put(PHASE3, tokens[2]);
+ } else {
+ throw new ValidationException(
+ "malformed command string, expected: ';;', actual: " + rawCommand);
+ }
+ return gson.toJson(content);
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/circuit/SetOfflineMaxCircuitCurrents.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/circuit/SetOfflineMaxCircuitCurrents.java
new file mode 100644
index 00000000000..28515f2c8c5
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/circuit/SetOfflineMaxCircuitCurrents.java
@@ -0,0 +1,60 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command.circuit;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.easee.internal.handler.EaseeThingHandler;
+import org.openhab.binding.easee.internal.model.ValidationException;
+import org.openhab.core.thing.Channel;
+import org.openhab.core.types.Command;
+
+/**
+ * implements the settings api call of the circuit.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class SetOfflineMaxCircuitCurrents extends SetCircuitSettings {
+ private static final String PHASE1 = "offlineMaxCircuitCurrentP1";
+ private static final String PHASE2 = "offlineMaxCircuitCurrentP2";
+ private static final String PHASE3 = "offlineMaxCircuitCurrentP3";
+
+ public SetOfflineMaxCircuitCurrents(EaseeThingHandler handler, Channel channel, Command command, String circuitId) {
+ super(handler, channel, command, circuitId);
+ }
+
+ /**
+ * helper that transforms channelId + commandvalue in a JSON string that can be added as content to a POST request.
+ *
+ * @return converted JSON string
+ * @throws ValidationException
+ */
+ @Override
+ protected String getJsonContent() throws ValidationException {
+ Map content = new HashMap(3);
+ String rawCommand = getCommandValue();
+ String[] tokens = rawCommand.split(";");
+ if (tokens.length == 3) {
+ content.put(PHASE1, tokens[0]);
+ content.put(PHASE2, tokens[1]);
+ content.put(PHASE3, tokens[2]);
+ } else {
+ throw new ValidationException(
+ "malformed command string, expected: ';;', actual: " + rawCommand);
+ }
+ return gson.toJson(content);
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/site/GetSite.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/site/GetSite.java
new file mode 100644
index 00000000000..02b373329d8
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/command/site/GetSite.java
@@ -0,0 +1,55 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.command.site;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.api.Request;
+import org.eclipse.jetty.http.HttpMethod;
+import org.openhab.binding.easee.internal.command.AbstractCommand;
+import org.openhab.binding.easee.internal.command.JsonResultProcessor;
+import org.openhab.binding.easee.internal.handler.EaseeBridgeHandler;
+
+/**
+ * implements the get sites api call of the site.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class GetSite extends AbstractCommand {
+
+ public GetSite(EaseeBridgeHandler handler, JsonResultProcessor resultProcessor) {
+ // retry does not make much sense as it is a polling command, command should always succeed therefore update
+ // handler on failure.
+ super(handler, RetryOnFailure.NO, ProcessFailureResponse.YES, resultProcessor);
+ }
+
+ @Override
+ protected Request prepareRequest(Request requestToPrepare) {
+ requestToPrepare.method(HttpMethod.GET);
+ return requestToPrepare;
+ }
+
+ @Override
+ protected String getURL() {
+ String url = GET_SITE_URL;
+ url = url.replaceAll("\\{siteId\\}", handler.getBridgeConfiguration().getSiteId());
+ return url;
+ }
+
+ @Override
+ protected String getChannelGroup() {
+ return CHANNEL_GROUP_SITE_INFO;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/config/EaseeConfiguration.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/config/EaseeConfiguration.java
new file mode 100644
index 00000000000..e51da25a6d5
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/config/EaseeConfiguration.java
@@ -0,0 +1,90 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.config;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * The {@link EaseeConfiguration} class contains fields mapping thing configuration parameters.
+ *
+ * @author Alexander Friese - Initial contribution
+ */
+@NonNullByDefault
+public class EaseeConfiguration {
+
+ private String username = "";
+ private String password = "";
+ private String siteId = "";
+
+ private Integer asyncTimeout = 120;
+ private Integer syncTimeout = 120;
+ private Integer dataPollingInterval = 120;
+
+ public String getUsername() {
+ return username;
+ }
+
+ public void setUsername(String username) {
+ this.username = username;
+ }
+
+ public String getPassword() {
+ return password;
+ }
+
+ public void setPassword(String password) {
+ this.password = password;
+ }
+
+ public String getSiteId() {
+ return siteId;
+ }
+
+ public void setSiteId(String siteId) {
+ this.siteId = siteId;
+ }
+
+ public Integer getAsyncTimeout() {
+ return asyncTimeout;
+ }
+
+ public void setAsyncTimeout(Integer asyncTimeout) {
+ this.asyncTimeout = asyncTimeout;
+ }
+
+ public Integer getSyncTimeout() {
+ return syncTimeout;
+ }
+
+ public void setSyncTimeout(Integer syncTimeout) {
+ this.syncTimeout = syncTimeout;
+ }
+
+ public Integer getDataPollingInterval() {
+ return dataPollingInterval;
+ }
+
+ public void setDataPollingInterval(Integer dataPollingInterval) {
+ this.dataPollingInterval = dataPollingInterval;
+ }
+
+ @Override
+ public String toString() {
+ StringBuilder builder = new StringBuilder();
+ builder.append("EaseeConfiguration [username=").append(username).append(", password=").append(password)
+ .append(", siteId=").append(siteId).append(", asyncTimeout=").append(asyncTimeout)
+ .append(", syncTimeout=").append(syncTimeout).append(", dataPollingInterval=")
+ .append(dataPollingInterval).append("]");
+ return builder.toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/connector/CommunicationStatus.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/connector/CommunicationStatus.java
new file mode 100644
index 00000000000..72657d781c9
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/connector/CommunicationStatus.java
@@ -0,0 +1,59 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.connector;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.eclipse.jetty.http.HttpStatus.Code;
+
+/**
+ * this class contains the HTTP status of the communication and an optional exception that might occoured during
+ * communication
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class CommunicationStatus {
+
+ private @Nullable Code httpCode;
+ private @Nullable Exception error;
+
+ public final Code getHttpCode() {
+ Code code = httpCode;
+ return code == null ? Code.INTERNAL_SERVER_ERROR : code;
+ }
+
+ public final void setHttpCode(Code httpCode) {
+ this.httpCode = httpCode;
+ }
+
+ public final @Nullable Exception getError() {
+ return error;
+ }
+
+ public final void setError(Exception error) {
+ this.error = error;
+ }
+
+ public final String getMessage() {
+ Exception err = error;
+ String errMsg = err == null ? null : err.getMessage();
+ String msg = getHttpCode().getMessage();
+ if (errMsg != null && !errMsg.isEmpty()) {
+ return errMsg;
+ } else if (msg != null && !msg.isEmpty()) {
+ return msg;
+ }
+ return "";
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/connector/WebInterface.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/connector/WebInterface.java
new file mode 100644
index 00000000000..49a54660fa3
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/connector/WebInterface.java
@@ -0,0 +1,346 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.connector;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import java.time.Instant;
+import java.time.temporal.ChronoUnit;
+import java.util.Queue;
+import java.util.concurrent.Future;
+import java.util.concurrent.ScheduledExecutorService;
+import java.util.concurrent.TimeUnit;
+import java.util.concurrent.atomic.AtomicReference;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.eclipse.jetty.client.HttpClient;
+import org.eclipse.jetty.util.BlockingArrayQueue;
+import org.openhab.binding.easee.internal.AtomicReferenceTrait;
+import org.openhab.binding.easee.internal.Utils;
+import org.openhab.binding.easee.internal.command.EaseeCommand;
+import org.openhab.binding.easee.internal.command.account.Login;
+import org.openhab.binding.easee.internal.command.account.RefreshToken;
+import org.openhab.binding.easee.internal.handler.EaseeBridgeHandler;
+import org.openhab.binding.easee.internal.handler.StatusHandler;
+import org.openhab.binding.easee.internal.model.ValidationException;
+import org.openhab.core.thing.ThingStatus;
+import org.openhab.core.thing.ThingStatusDetail;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import com.google.gson.JsonObject;
+
+/**
+ * The connector is responsible for communication with the Easee Cloud API
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class WebInterface implements AtomicReferenceTrait {
+
+ private final Logger logger = LoggerFactory.getLogger(WebInterface.class);
+
+ /**
+ * bridge handler
+ */
+ private final EaseeBridgeHandler handler;
+
+ /**
+ * handler for updating bridge status
+ */
+ private final StatusHandler statusHandler;
+
+ /**
+ * holds authentication status
+ */
+ private boolean authenticated = false;
+
+ /**
+ * access token returned by login, needed to authenticate all requests send to API.
+ */
+ private String accessToken;
+
+ /**
+ * refresh token returned by login, needed for refreshing the access token.
+ */
+ private String refreshToken;
+
+ /**
+ * expiry of the access token.
+ */
+ private Instant tokenExpiry;
+
+ /**
+ * last refresh of the access token.
+ */
+ private Instant tokenRefreshDate;
+
+ /**
+ * HTTP client for asynchronous calls
+ */
+ private final HttpClient httpClient;
+
+ /**
+ * the scheduler which periodically sends web requests to the Easee Cloud API. Should be initiated with the thing's
+ * existing scheduler instance.
+ */
+ private final ScheduledExecutorService scheduler;
+
+ /**
+ * request executor
+ */
+ private final WebRequestExecutor requestExecutor;
+
+ /**
+ * periodic request executor job
+ */
+ private final AtomicReference<@Nullable Future>> requestExecutorJobReference;
+
+ /**
+ * this class is responsible for executing periodic web requests. This ensures that only one request is executed
+ * at the same time and there will be a guaranteed minimum delay between subsequent requests.
+ *
+ * @author afriese - initial contribution
+ */
+ private class WebRequestExecutor implements Runnable {
+
+ /**
+ * queue which holds the commands to execute
+ */
+ private final Queue commandQueue;
+
+ /**
+ * constructor
+ */
+ WebRequestExecutor() {
+ this.commandQueue = new BlockingArrayQueue<>(WEB_REQUEST_QUEUE_MAX_SIZE);
+ }
+
+ private void processAuthenticationResult(CommunicationStatus status, JsonObject jsonObject) {
+ String msg = Utils.getAsString(jsonObject, JSON_KEY_ERROR_TITLE);
+ if (msg == null || msg.isBlank()) {
+ msg = status.getMessage();
+ }
+
+ switch (status.getHttpCode()) {
+ case BAD_REQUEST:
+ statusHandler.updateStatusInfo(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR, msg);
+ setAuthenticated(false);
+ break;
+ case OK:
+ String accessToken = Utils.getAsString(jsonObject, JSON_KEY_AUTH_ACCESS_TOKEN);
+ String refreshToken = Utils.getAsString(jsonObject, JSON_KEY_AUTH_REFRESH_TOKEN);
+ int expiresInSeconds = Utils.getAsInt(jsonObject, JSON_KEY_AUTH_EXPIRES_IN);
+ if (accessToken != null && refreshToken != null && expiresInSeconds != 0) {
+ WebInterface.this.accessToken = accessToken;
+ WebInterface.this.refreshToken = refreshToken;
+ tokenRefreshDate = Instant.now();
+ tokenExpiry = tokenRefreshDate.plusSeconds(expiresInSeconds);
+
+ logger.debug("access token refreshed: {}, expiry: {}", Utils.formatDate(tokenRefreshDate),
+ Utils.formatDate(tokenExpiry));
+
+ statusHandler.updateStatusInfo(ThingStatus.ONLINE, ThingStatusDetail.NONE,
+ STATUS_TOKEN_VALIDATED);
+ setAuthenticated(true);
+ handler.startDiscovery();
+ break;
+ }
+ default:
+ statusHandler.updateStatusInfo(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR, msg);
+ setAuthenticated(false);
+ }
+ }
+
+ /**
+ * puts a command into the queue
+ *
+ * @param command
+ */
+ void enqueue(EaseeCommand command) {
+ try {
+ commandQueue.add(command);
+ } catch (IllegalStateException ex) {
+ if (commandQueue.size() >= WEB_REQUEST_QUEUE_MAX_SIZE) {
+ logger.debug(
+ "Could not add command to command queue because queue is already full. Maybe Easee Cloud is down?");
+ } else {
+ logger.warn("Could not add command to queue - IllegalStateException");
+ }
+ }
+ }
+
+ /**
+ * executes the web request
+ */
+ @Override
+ public void run() {
+ logger.debug("run queued commands, queue size is {}", commandQueue.size());
+ if (!isAuthenticated()) {
+ authenticate();
+ } else {
+ refreshAccessToken();
+
+ if (isAuthenticated() && !commandQueue.isEmpty()) {
+ try {
+ executeCommand();
+ } catch (Exception ex) {
+ logger.warn("command execution ended with exception:", ex);
+ }
+ }
+ }
+ }
+
+ /**
+ * authenticates with the Easee Cloud interface.
+ */
+ private synchronized void authenticate() {
+ setAuthenticated(false);
+ EaseeCommand loginCommand = new Login(handler);
+ loginCommand.registerResultProcessor(this::processAuthenticationResult);
+ try {
+ loginCommand.performAction(httpClient, accessToken);
+ } catch (ValidationException e) {
+ // this cannot happen
+ }
+ }
+
+ /**
+ * periodically refreshed the access token.
+ */
+ private synchronized void refreshAccessToken() {
+ Instant now = Instant.now();
+
+ if (now.isAfter(tokenExpiry.minus(WEB_REQUEST_TOKEN_EXPIRY_BUFFER_MINUTES, ChronoUnit.MINUTES))
+ || now.isAfter(tokenRefreshDate.plus(WEB_REQUEST_TOKEN_MAX_AGE_MINUTES, ChronoUnit.MINUTES))) {
+ logger.debug("access token needs to be refreshed, last refresh: {}, expiry: {}",
+ Utils.formatDate(tokenRefreshDate), Utils.formatDate(tokenExpiry));
+
+ EaseeCommand refreshCommand = new RefreshToken(handler, accessToken, refreshToken);
+ refreshCommand.registerResultProcessor(this::processAuthenticationResult);
+ try {
+ refreshCommand.performAction(httpClient, accessToken);
+ } catch (ValidationException e) {
+ // this cannot happen
+ }
+ }
+ }
+
+ /**
+ * executes the next command in the queue. requires authenticated session.
+ *
+ * @throws ValidationException
+ */
+ private void executeCommand() throws ValidationException {
+ EaseeCommand command = commandQueue.poll();
+ if (command != null) {
+ command.registerResultProcessor(this::processExecutionResult);
+ command.performAction(httpClient, accessToken);
+ }
+ }
+
+ private void processExecutionResult(CommunicationStatus status, JsonObject jsonObject) {
+ String msg = Utils.getAsString(jsonObject, JSON_KEY_ERROR_TITLE);
+ if (msg == null || msg.isBlank()) {
+ msg = status.getMessage();
+ }
+
+ switch (status.getHttpCode()) {
+ case OK:
+ case ACCEPTED:
+ // no action needed as the thing is already online.
+ break;
+ default:
+ statusHandler.updateStatusInfo(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR, msg);
+ setAuthenticated(false);
+
+ }
+ }
+ }
+
+ /**
+ * Constructor to set up interface
+ *
+ * @param config Bridge configuration
+ */
+ public WebInterface(ScheduledExecutorService scheduler, EaseeBridgeHandler handler, HttpClient httpClient,
+ StatusHandler statusHandler) {
+ this.handler = handler;
+ this.statusHandler = statusHandler;
+ this.scheduler = scheduler;
+ this.httpClient = httpClient;
+ this.tokenExpiry = OUTDATED_DATE;
+ this.tokenRefreshDate = OUTDATED_DATE;
+ this.accessToken = "";
+ this.refreshToken = "";
+ this.requestExecutor = new WebRequestExecutor();
+ this.requestExecutorJobReference = new AtomicReference<>(null);
+ }
+
+ public void start() {
+ setAuthenticated(false);
+ updateJobReference(requestExecutorJobReference, scheduler.scheduleWithFixedDelay(requestExecutor,
+ WEB_REQUEST_INITIAL_DELAY, WEB_REQUEST_INTERVAL, TimeUnit.SECONDS));
+ }
+
+ /**
+ * queues any command for execution
+ *
+ * @param command
+ */
+ public void enqueueCommand(EaseeCommand command) {
+ requestExecutor.enqueue(command);
+ }
+
+ /**
+ * will be called by the ThingHandler to abort periodic jobs.
+ */
+ public void dispose() {
+ logger.debug("Webinterface disposed.");
+ cancelJobReference(requestExecutorJobReference);
+ setAuthenticated(false);
+ }
+
+ /**
+ * returns authentication status.
+ *
+ * @return
+ */
+ private boolean isAuthenticated() {
+ return authenticated;
+ }
+
+ /**
+ * update the authentication status, also resets token data.
+ *
+ * @param authenticated
+ */
+ private void setAuthenticated(boolean authenticated) {
+ this.authenticated = authenticated;
+ if (!authenticated) {
+ this.tokenExpiry = OUTDATED_DATE;
+ this.accessToken = "";
+ this.refreshToken = "";
+ }
+ }
+
+ /**
+ * returns the current access token.
+ *
+ * @return
+ */
+ public String getAccessToken() {
+ return accessToken;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/discovery/EaseeSiteDiscoveryService.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/discovery/EaseeSiteDiscoveryService.java
new file mode 100644
index 00000000000..62e8bd65118
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/discovery/EaseeSiteDiscoveryService.java
@@ -0,0 +1,169 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.discovery;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.easee.internal.EaseeBindingConstants;
+import org.openhab.binding.easee.internal.Utils;
+import org.openhab.binding.easee.internal.command.site.GetSite;
+import org.openhab.binding.easee.internal.connector.CommunicationStatus;
+import org.openhab.binding.easee.internal.handler.EaseeSiteHandler;
+import org.openhab.core.config.discovery.AbstractDiscoveryService;
+import org.openhab.core.config.discovery.DiscoveryResultBuilder;
+import org.openhab.core.thing.ThingTypeUID;
+import org.openhab.core.thing.ThingUID;
+import org.openhab.core.thing.binding.ThingHandler;
+import org.openhab.core.thing.binding.ThingHandlerService;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import com.google.gson.JsonArray;
+import com.google.gson.JsonElement;
+import com.google.gson.JsonObject;
+
+/**
+ * this class will handle discovery of wallboxes and circuits within the site configured.
+ *
+ * @author Alexander Friese - initial contribution
+ *
+ */
+@NonNullByDefault
+public class EaseeSiteDiscoveryService extends AbstractDiscoveryService implements ThingHandlerService {
+
+ private final Logger logger = LoggerFactory.getLogger(EaseeSiteDiscoveryService.class);
+ private @NonNullByDefault({}) EaseeSiteHandler bridgeHandler;
+
+ public EaseeSiteDiscoveryService() throws IllegalArgumentException {
+ super(EaseeBindingConstants.SUPPORTED_THING_TYPES_UIDS, 300, false);
+ }
+
+ @Override
+ protected void startScan() {
+ bridgeHandler.enqueueCommand(new GetSite(bridgeHandler, this::processSiteDiscoveryResult));
+ }
+
+ @Override
+ public void setThingHandler(ThingHandler handler) {
+ if (handler instanceof EaseeSiteHandler) {
+ this.bridgeHandler = (EaseeSiteHandler) handler;
+ this.bridgeHandler.setDiscoveryService(this);
+ }
+ }
+
+ @Override
+ public @Nullable ThingHandler getThingHandler() {
+ return bridgeHandler;
+ }
+
+ // method is defined in both implemented interface and inherited class, thus we must define a behaviour here.
+ @Override
+ public void deactivate() {
+ super.deactivate();
+ }
+
+ /**
+ * callback that handles json result data to provide discovery result.
+ *
+ * @param site
+ */
+ private void processSiteDiscoveryResult(CommunicationStatus status, JsonObject site) {
+ logger.debug("processDiscoveryResult {}", site);
+
+ JsonArray circuits = site.getAsJsonArray(JSON_KEY_CIRCUITS);
+ if (circuits == null) {
+ logger.info("Site discovery failed, no circuits found.");
+ } else {
+ circuits.forEach(this::handleCircuitDiscovery);
+ }
+ }
+
+ /**
+ * handles each circuit discovery result.
+ *
+ * @param circuit
+ */
+ private void handleCircuitDiscovery(JsonElement json) {
+ logger.debug("handleCircuitDiscovery {}", json);
+
+ JsonObject circuit = json.getAsJsonObject();
+ final String circuitId = Utils.getAsString(circuit, JSON_KEY_GENERIC_ID);
+ final String circuitName = Utils.getAsString(circuit, JSON_KEY_CIRCUIT_NAME);
+
+ if (circuitId != null) {
+ final String circuitLabel = circuitName != null ? circuitName : circuitId;
+
+ // handle contained chargers
+ JsonArray chargers = circuit.getAsJsonArray(JSON_KEY_CHARGERS);
+ if (chargers == null) {
+ logger.info("Site discovery failed, no chargers found.");
+ } else {
+ chargers.forEach(charger -> handleChargerDiscovery(charger, circuitId, circuitLabel));
+ }
+ }
+ }
+
+ /**
+ * handles each charger discovery result.
+ *
+ * @param charger
+ */
+ private void handleChargerDiscovery(JsonElement json, String circuitId, String circuitLabel) {
+ logger.debug("handleChargerDiscovery {}", json);
+
+ JsonObject charger = json.getAsJsonObject();
+ String chargerId = Utils.getAsString(charger, JSON_KEY_GENERIC_ID);
+ String backPlateId = Utils.getAsString(charger.getAsJsonObject(JSON_KEY_BACK_PLATE), JSON_KEY_GENERIC_ID);
+ String masterBackPlateId = Utils.getAsString(charger.getAsJsonObject(JSON_KEY_BACK_PLATE),
+ JSON_KEY_MASTER_BACK_PLATE_ID);
+ String chargerName = Utils.getAsString(charger, JSON_KEY_GENERIC_NAME);
+
+ if (chargerId != null && backPlateId != null && masterBackPlateId != null) {
+ DiscoveryResultBuilder builder;
+
+ if (backPlateId.equals(masterBackPlateId)) {
+ builder = initDiscoveryResultBuilder(DEVICE_MASTER_CHARGER, chargerId, chargerName)
+ .withProperty(THING_CONFIG_IS_MASTER, GENERIC_YES);
+ } else {
+ builder = initDiscoveryResultBuilder(DEVICE_CHARGER, chargerId, chargerName)
+ .withProperty(THING_CONFIG_IS_MASTER, GENERIC_NO);
+ }
+ builder.withProperty(THING_CONFIG_CIRCUIT_ID, circuitId);
+ builder.withProperty(THING_CONFIG_CIRCUIT_NAME, circuitLabel);
+ builder.withProperty(THING_CONFIG_BACK_PLATE_ID, backPlateId);
+ builder.withProperty(THING_CONFIG_MASTER_BACK_PLATE_ID, masterBackPlateId);
+ thingDiscovered(builder.build());
+ }
+ }
+
+ /**
+ * sends discovery notification to the framework.
+ *
+ * @param deviceType
+ * @param deviceId
+ * @param deviceName
+ */
+ private DiscoveryResultBuilder initDiscoveryResultBuilder(String deviceType, String deviceId,
+ @Nullable String deviceName) {
+ ThingUID bridgeUID = bridgeHandler.getThing().getUID();
+ ThingTypeUID typeUid = new ThingTypeUID(BINDING_ID, deviceType);
+
+ ThingUID thingUID = new ThingUID(typeUid, bridgeUID, deviceId);
+ String label = deviceName != null ? deviceName : deviceId;
+
+ return DiscoveryResultBuilder.create(thingUID).withBridge(bridgeUID).withLabel(label)
+ .withProperty(THING_CONFIG_ID, deviceId).withRepresentationProperty(THING_CONFIG_ID);
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/ChannelProvider.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/ChannelProvider.java
new file mode 100644
index 00000000000..7758f8e6672
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/ChannelProvider.java
@@ -0,0 +1,29 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.handler;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.core.thing.Channel;
+
+/**
+ * this interface provides all methods which deal with channels
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public interface ChannelProvider {
+
+ @Nullable
+ Channel getChannel(String groupId, String channelId);
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/EaseeBridgeHandler.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/EaseeBridgeHandler.java
new file mode 100644
index 00000000000..5e8dd13e0fa
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/EaseeBridgeHandler.java
@@ -0,0 +1,30 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.handler;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.core.thing.binding.BridgeHandler;
+
+/**
+ * public interface of the {@link EaseeBridgeHandler}
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public interface EaseeBridgeHandler extends BridgeHandler, EaseeThingHandler {
+
+ /**
+ * starts discovery of wallboxes and circuits
+ */
+ void startDiscovery();
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/EaseeChargerHandler.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/EaseeChargerHandler.java
new file mode 100644
index 00000000000..c5ea6a7688f
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/EaseeChargerHandler.java
@@ -0,0 +1,234 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.handler;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import java.util.Map;
+import java.util.concurrent.Future;
+import java.util.concurrent.TimeUnit;
+import java.util.concurrent.atomic.AtomicReference;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.easee.internal.AtomicReferenceTrait;
+import org.openhab.binding.easee.internal.EaseeBindingConstants;
+import org.openhab.binding.easee.internal.Utils;
+import org.openhab.binding.easee.internal.command.EaseeCommand;
+import org.openhab.binding.easee.internal.command.charger.ChangeConfiguration;
+import org.openhab.binding.easee.internal.command.charger.Charger;
+import org.openhab.binding.easee.internal.command.charger.ChargerState;
+import org.openhab.binding.easee.internal.command.charger.GetConfiguration;
+import org.openhab.binding.easee.internal.command.charger.LatestChargingSession;
+import org.openhab.binding.easee.internal.command.charger.SendCommand;
+import org.openhab.binding.easee.internal.command.charger.SendCommandStartStop;
+import org.openhab.binding.easee.internal.config.EaseeConfiguration;
+import org.openhab.binding.easee.internal.connector.CommunicationStatus;
+import org.openhab.core.thing.Bridge;
+import org.openhab.core.thing.Channel;
+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.openhab.core.types.State;
+import org.openhab.core.types.UnDefType;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import com.google.gson.JsonObject;
+
+/**
+ * The {@link EaseeChargerHandler} is responsible for handling commands, which are
+ * sent to one of the channels.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class EaseeChargerHandler extends BaseThingHandler implements EaseeThingHandler, AtomicReferenceTrait {
+ private final Logger logger = LoggerFactory.getLogger(EaseeChargerHandler.class);
+
+ /**
+ * Schedule for polling live data
+ */
+ private final AtomicReference<@Nullable Future>> dataPollingJobReference;
+
+ public EaseeChargerHandler(Thing thing) {
+ super(thing);
+ this.dataPollingJobReference = new AtomicReference<>(null);
+ }
+
+ @Override
+ public void initialize() {
+ logger.debug("About to initialize Charger");
+ String chargerId = getConfig().get(EaseeBindingConstants.THING_CONFIG_ID).toString();
+ logger.debug("Easee Charger initialized with id: {}", chargerId);
+
+ updateStatus(ThingStatus.UNKNOWN, ThingStatusDetail.NONE, STATUS_WAITING_FOR_BRIDGE);
+ startPolling();
+
+ enqueueCommand(new Charger(this, chargerId, this::updateProperties));
+ }
+
+ private void updateProperties(CommunicationStatus status, JsonObject charger) {
+ Map properties = editProperties();
+
+ String backPlateId = Utils.getAsString(charger.getAsJsonObject(JSON_KEY_BACK_PLATE), JSON_KEY_GENERIC_ID);
+ String masterBackPlateId = Utils.getAsString(charger.getAsJsonObject(JSON_KEY_BACK_PLATE),
+ JSON_KEY_MASTER_BACK_PLATE_ID);
+ if (backPlateId != null && masterBackPlateId != null) {
+ if (backPlateId.equals(masterBackPlateId)) {
+ properties.put(THING_CONFIG_IS_MASTER, GENERIC_YES);
+ } else {
+ properties.put(THING_CONFIG_IS_MASTER, GENERIC_NO);
+ }
+ properties.put(THING_CONFIG_BACK_PLATE_ID, backPlateId);
+ properties.put(THING_CONFIG_MASTER_BACK_PLATE_ID, masterBackPlateId);
+ }
+ String chargerName = Utils.getAsString(charger, JSON_KEY_GENERIC_NAME);
+ if (chargerName != null) {
+ properties.put(JSON_KEY_GENERIC_NAME, chargerName);
+ }
+ String circuitId = Utils.getAsString(charger.getAsJsonObject(JSON_KEY_BACK_PLATE), JSON_KEY_CIRCUIT_ID);
+ if (circuitId != null) {
+ properties.put(JSON_KEY_CIRCUIT_ID, circuitId);
+ }
+
+ updateProperties(properties);
+ }
+
+ /**
+ * Start the polling.
+ */
+ private void startPolling() {
+ updateJobReference(dataPollingJobReference, scheduler.scheduleWithFixedDelay(this::pollingRun,
+ POLLING_INITIAL_DELAY, getBridgeConfiguration().getDataPollingInterval(), TimeUnit.SECONDS));
+ }
+
+ /**
+ * Poll the Easee Cloud API one time.
+ */
+ void pollingRun() {
+ String chargerId = getConfig().get(EaseeBindingConstants.THING_CONFIG_ID).toString();
+ logger.debug("polling charger data for {}", chargerId);
+
+ ChargerState state = new ChargerState(this, chargerId);
+ state.registerResultProcessor(this::updateStatusInfo);
+ enqueueCommand(state);
+
+ // proceed if charger is online
+ if (getThing().getStatus() == ThingStatus.ONLINE) {
+ enqueueCommand(new GetConfiguration(this, chargerId));
+ enqueueCommand(new LatestChargingSession(this, chargerId));
+ }
+ }
+
+ /**
+ * updates status depending on online information received from the API.
+ *
+ * @param status
+ * @param jsonObject
+ */
+ private void updateStatusInfo(CommunicationStatus status, JsonObject jsonObject) {
+ Boolean isOnline = Utils.getAsBool(jsonObject, JSON_KEY_ONLINE);
+
+ if (isOnline == null) {
+ super.updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR, STATUS_NO_VALID_DATA);
+ } else if (isOnline) {
+ super.updateStatus(ThingStatus.ONLINE, ThingStatusDetail.NONE);
+ } else {
+ super.updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR, STATUS_NO_CONNECTION);
+ }
+ }
+
+ /**
+ * Disposes the thing.
+ */
+ @Override
+ public void dispose() {
+ logger.debug("Handler disposed.");
+ cancelJobReference(dataPollingJobReference);
+ }
+
+ /**
+ * will update all channels provided in the map
+ */
+ @Override
+ public void updateChannelStatus(Map values) {
+ logger.debug("Handling charger channel update.");
+
+ for (Channel channel : values.keySet()) {
+ if (getThing().getChannels().contains(channel)) {
+ State value = values.get(channel);
+ if (value != null) {
+ logger.debug("Channel is to be updated: {}: {}", channel.getUID().getAsString(), value);
+ updateState(channel.getUID(), value);
+ } else {
+ logger.debug("Value is null or not provided by Easee Cloud (channel: {})",
+ channel.getUID().getAsString());
+ updateState(channel.getUID(), UnDefType.UNDEF);
+ }
+ } else {
+ logger.debug("Could not identify channel: {} for model {}", channel.getUID().getAsString(),
+ getThing().getThingTypeUID().getAsString());
+ }
+ }
+ }
+
+ @Override
+ public void enqueueCommand(EaseeCommand command) {
+ EaseeBridgeHandler bridgeHandler = getBridgeHandler();
+ if (bridgeHandler != null) {
+ bridgeHandler.enqueueCommand(command);
+ } else {
+ // this should not happen
+ logger.warn("no bridge handler found");
+ }
+ }
+
+ private @Nullable EaseeBridgeHandler getBridgeHandler() {
+ Bridge bridge = getBridge();
+ return bridge == null ? null : ((EaseeBridgeHandler) bridge.getHandler());
+ }
+
+ @Override
+ public EaseeConfiguration getBridgeConfiguration() {
+ EaseeBridgeHandler bridgeHandler = getBridgeHandler();
+ return bridgeHandler == null ? new EaseeConfiguration() : bridgeHandler.getBridgeConfiguration();
+ }
+
+ @Override
+ public EaseeCommand buildEaseeCommand(Command command, Channel channel) {
+ String chargerId = getConfig().get(EaseeBindingConstants.THING_CONFIG_ID).toString();
+
+ switch (Utils.getWriteCommand(channel)) {
+ case COMMAND_CHANGE_CONFIGURATION:
+ return new ChangeConfiguration(this, chargerId, channel, command);
+ case COMMAND_SEND_COMMAND:
+ return new SendCommand(this, chargerId, channel, command);
+ case COMMAND_SEND_COMMAND_START_STOP:
+ return new SendCommandStartStop(this, chargerId, channel, command);
+ default:
+ // this should not happen
+ logger.error("write command '{}' not found for channel '{}'", command.toString(),
+ channel.getUID().getIdWithoutGroup());
+ throw new UnsupportedOperationException(
+ "write command not found for channel: " + channel.getUID().getIdWithoutGroup());
+ }
+ }
+
+ @Override
+ public Logger getLogger() {
+ return logger;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/EaseeMasterChargerHandler.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/EaseeMasterChargerHandler.java
new file mode 100644
index 00000000000..9476a36deb4
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/EaseeMasterChargerHandler.java
@@ -0,0 +1,82 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.handler;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.easee.internal.EaseeBindingConstants;
+import org.openhab.binding.easee.internal.Utils;
+import org.openhab.binding.easee.internal.command.EaseeCommand;
+import org.openhab.binding.easee.internal.command.circuit.CircuitSettings;
+import org.openhab.binding.easee.internal.command.circuit.DynamicCircuitCurrent;
+import org.openhab.binding.easee.internal.command.circuit.SetCircuitSettings;
+import org.openhab.binding.easee.internal.command.circuit.SetDynamicCircuitCurrents;
+import org.openhab.binding.easee.internal.command.circuit.SetMaxCircuitCurrents;
+import org.openhab.binding.easee.internal.command.circuit.SetOfflineMaxCircuitCurrents;
+import org.openhab.core.thing.Channel;
+import org.openhab.core.thing.Thing;
+import org.openhab.core.thing.ThingStatus;
+import org.openhab.core.types.Command;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * The {@link EaseeMasterChargerHandler} is responsible for handling commands, which are
+ * sent to one of the channels.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class EaseeMasterChargerHandler extends EaseeChargerHandler {
+ private final Logger logger = LoggerFactory.getLogger(EaseeMasterChargerHandler.class);
+
+ public EaseeMasterChargerHandler(Thing thing) {
+ super(thing);
+ }
+
+ /**
+ * Poll the Easee Cloud API one time.
+ */
+ @Override
+ void pollingRun() {
+ super.pollingRun();
+
+ // proceed if charger is online, otherwise circuit data is not in sync.
+ if (getThing().getStatus() == ThingStatus.ONLINE) {
+ String circuitId = getConfig().get(EaseeBindingConstants.THING_CONFIG_CIRCUIT_ID).toString();
+ logger.debug("polling circuit data for {}", circuitId);
+
+ enqueueCommand(new DynamicCircuitCurrent(this, circuitId));
+ enqueueCommand(new CircuitSettings(this, circuitId));
+ }
+ }
+
+ @Override
+ public EaseeCommand buildEaseeCommand(Command command, Channel channel) {
+ String circuitId = getConfig().get(EaseeBindingConstants.THING_CONFIG_CIRCUIT_ID).toString();
+
+ switch (Utils.getWriteCommand(channel)) {
+ case COMMAND_SET_CIRCUIT_SETTINGS:
+ return new SetCircuitSettings(this, channel, command, circuitId);
+ case COMMAND_SET_DYNAMIC_CIRCUIT_CURRENTS:
+ return new SetDynamicCircuitCurrents(this, channel, command, circuitId);
+ case COMMAND_SET_MAX_CIRCUIT_CURRENTS:
+ return new SetMaxCircuitCurrents(this, channel, command, circuitId);
+ case COMMAND_SET_OFFLINE_MAX_CIRCUIT_CURRENTS:
+ return new SetOfflineMaxCircuitCurrents(this, channel, command, circuitId);
+ default:
+ return super.buildEaseeCommand(command, channel);
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/EaseeSiteHandler.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/EaseeSiteHandler.java
new file mode 100644
index 00000000000..0bba6787315
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/EaseeSiteHandler.java
@@ -0,0 +1,129 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.handler;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import java.util.Collection;
+import java.util.Collections;
+import java.util.Map;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.eclipse.jetty.client.HttpClient;
+import org.openhab.binding.easee.internal.Utils;
+import org.openhab.binding.easee.internal.command.EaseeCommand;
+import org.openhab.binding.easee.internal.command.site.GetSite;
+import org.openhab.binding.easee.internal.config.EaseeConfiguration;
+import org.openhab.binding.easee.internal.connector.CommunicationStatus;
+import org.openhab.binding.easee.internal.connector.WebInterface;
+import org.openhab.binding.easee.internal.discovery.EaseeSiteDiscoveryService;
+import org.openhab.core.config.discovery.DiscoveryService;
+import org.openhab.core.thing.Bridge;
+import org.openhab.core.thing.ThingStatus;
+import org.openhab.core.thing.ThingStatusDetail;
+import org.openhab.core.thing.binding.BaseBridgeHandler;
+import org.openhab.core.thing.binding.ThingHandlerService;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import com.google.gson.JsonObject;
+
+/**
+ * The {@link EaseeSiteHandler} is responsible for handling commands, which are
+ * sent to one of the channels.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class EaseeSiteHandler extends BaseBridgeHandler implements EaseeBridgeHandler {
+ private final Logger logger = LoggerFactory.getLogger(EaseeSiteHandler.class);
+
+ private @Nullable DiscoveryService discoveryService;
+
+ /**
+ * Interface object for querying the Easee web interface
+ */
+ private WebInterface webInterface;
+
+ public EaseeSiteHandler(Bridge bridge, HttpClient httpClient) {
+ super(bridge);
+ this.webInterface = new WebInterface(scheduler, this, httpClient, super::updateStatus);
+ }
+
+ @Override
+ public void initialize() {
+ logger.debug("About to initialize Easee Site");
+ EaseeConfiguration config = getBridgeConfiguration();
+ logger.debug("Easee Site initialized with configuration: {}", config.toString());
+
+ updateStatus(ThingStatus.UNKNOWN, ThingStatusDetail.NONE, STATUS_WAITING_FOR_LOGIN);
+ webInterface.start();
+
+ enqueueCommand(new GetSite(this, this::updateProperties));
+ }
+
+ private void updateProperties(CommunicationStatus status, JsonObject jsonObject) {
+ Map properties = editProperties();
+ String name = Utils.getAsString(jsonObject, JSON_KEY_GENERIC_NAME);
+ if (name != null) {
+ properties.put(JSON_KEY_GENERIC_NAME, name);
+ }
+ String siteKey = Utils.getAsString(jsonObject, JSON_KEY_SITE_KEY);
+ if (siteKey != null) {
+ properties.put(JSON_KEY_SITE_KEY, siteKey);
+ }
+ updateProperties(properties);
+ }
+
+ /**
+ * Disposes the bridge.
+ */
+ @Override
+ public void dispose() {
+ logger.debug("Handler disposed.");
+ webInterface.dispose();
+ }
+
+ @Override
+ public EaseeConfiguration getBridgeConfiguration() {
+ return this.getConfigAs(EaseeConfiguration.class);
+ }
+
+ @Override
+ public Collection> getServices() {
+ return Collections.singleton(EaseeSiteDiscoveryService.class);
+ }
+
+ public void setDiscoveryService(EaseeSiteDiscoveryService discoveryService) {
+ this.discoveryService = discoveryService;
+ }
+
+ @Override
+ public void startDiscovery() {
+ DiscoveryService discoveryService = this.discoveryService;
+ if (discoveryService != null) {
+ discoveryService.startScan(null);
+ }
+ }
+
+ @Override
+ public void enqueueCommand(EaseeCommand command) {
+ webInterface.enqueueCommand(command);
+ }
+
+ @Override
+ public Logger getLogger() {
+ return logger;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/EaseeThingHandler.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/EaseeThingHandler.java
new file mode 100644
index 00000000000..445b18d199b
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/EaseeThingHandler.java
@@ -0,0 +1,136 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.handler;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.CHANNEL_TYPEPREFIX_RW;
+
+import java.util.Map;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.easee.internal.Utils;
+import org.openhab.binding.easee.internal.command.EaseeCommand;
+import org.openhab.binding.easee.internal.config.EaseeConfiguration;
+import org.openhab.core.thing.Channel;
+import org.openhab.core.thing.ChannelGroupUID;
+import org.openhab.core.thing.ChannelUID;
+import org.openhab.core.thing.ThingStatus;
+import org.openhab.core.thing.ThingUID;
+import org.openhab.core.thing.binding.ThingHandler;
+import org.openhab.core.types.Command;
+import org.openhab.core.types.RefreshType;
+import org.openhab.core.types.State;
+import org.slf4j.Logger;
+
+/**
+ * public interface of the {@link EaseeThingHandler}. provides some default implementations which can be used by both
+ * BridgeHandlers and ThingHandlers.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public interface EaseeThingHandler extends ThingHandler, ChannelProvider {
+
+ /**
+ * just to avoid usage of static loggers.
+ *
+ * @return
+ */
+ Logger getLogger();
+
+ /**
+ * method which updates the channels. needs to be implemented by thing types which have channels.
+ *
+ * @param values key-value list where key is the channel
+ */
+ default void updateChannelStatus(Map values) {
+ getLogger().debug("updateChannelStatus not implemented/supported by this thing type");
+ }
+
+ /**
+ * return the bridge's configuration
+ *
+ * @return
+ */
+ EaseeConfiguration getBridgeConfiguration();
+
+ /**
+ * passes a new command o the command queue
+ *
+ * @param command to be queued/executed
+ */
+ void enqueueCommand(EaseeCommand command);
+
+ /**
+ * default implementation to handle commands
+ *
+ * @param channelUID
+ * @param command
+ */
+ @Override
+ default void handleCommand(ChannelUID channelUID, Command command) {
+ getLogger().debug("command for {}: {}", channelUID, command);
+
+ if (command instanceof RefreshType) {
+ return;
+ }
+
+ String group = channelUID.getGroupId();
+ group = group == null ? "" : group;
+ Channel channel = getChannel(group, channelUID.getIdWithoutGroup());
+ if (channel == null) {
+ // this should not happen
+ getLogger().warn("channel not found: {}", channelUID);
+ return;
+ }
+
+ String channelType = Utils.getChannelTypeId(channel);
+ if (!channelType.startsWith(CHANNEL_TYPEPREFIX_RW)) {
+ getLogger().info("channel '{}' does not support write access - value to set '{}'",
+ channelUID.getIdWithoutGroup(), command);
+ throw new UnsupportedOperationException(
+ "channel (" + channelUID.getIdWithoutGroup() + ") does not support write access");
+ }
+
+ if (getThing().getStatus() != ThingStatus.ONLINE) {
+ getLogger().debug("Thing is not online, thus no commands will be handled");
+ return;
+ }
+ enqueueCommand(buildEaseeCommand(command, channel));
+ }
+
+ /**
+ * builds the EaseeCommand which can be send to the webinterface.
+ *
+ * @param command the openhab raw command received from the framework
+ * @param channel the channel which belongs to the command.
+ * @return
+ */
+ default EaseeCommand buildEaseeCommand(Command command, Channel channel) {
+ throw new UnsupportedOperationException("buildEaseeCommand not implemented/supported by this thing type");
+ }
+
+ /**
+ * determines the channel for a given groupId and channelId.
+ *
+ * @param groupId groupId of the channel
+ * @param channelId channelId of the channel
+ */
+ @Override
+ default @Nullable Channel getChannel(String groupId, String channelId) {
+ ThingUID thingUID = this.getThing().getUID();
+ ChannelGroupUID channelGroupUID = new ChannelGroupUID(thingUID, groupId);
+ Channel channel = getThing().getChannel(new ChannelUID(channelGroupUID, channelId));
+ return channel;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/StatusHandler.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/StatusHandler.java
new file mode 100644
index 00000000000..c68da39615a
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/handler/StatusHandler.java
@@ -0,0 +1,36 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.handler;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.core.thing.ThingStatus;
+import org.openhab.core.thing.ThingStatusDetail;
+
+/**
+ * functional interface to provide an function to update status of a thing or bridge.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@FunctionalInterface
+@NonNullByDefault
+public interface StatusHandler {
+ /**
+ * Called from WebInterface#authenticate() to update
+ * the thing status because updateStatus is protected.
+ *
+ * @param status Thing status
+ * @param statusDetail Thing status detail
+ * @param description Thing status description
+ */
+ void updateStatusInfo(ThingStatus status, ThingStatusDetail statusDetail, String description);
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/model/ConfigurationException.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/model/ConfigurationException.java
new file mode 100644
index 00000000000..1389bf6dd43
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/model/ConfigurationException.java
@@ -0,0 +1,41 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.model;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * exception whichs is used to state a validation error
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class ConfigurationException extends RuntimeException {
+ private static final long serialVersionUID = 5736865225953884234L;
+
+ public ConfigurationException() {
+ super();
+ }
+
+ public ConfigurationException(String message) {
+ super(message);
+ }
+
+ public ConfigurationException(Throwable cause) {
+ super(cause);
+ }
+
+ public ConfigurationException(String message, Throwable cause) {
+ super(message, cause);
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/model/CustomResponseTransformer.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/model/CustomResponseTransformer.java
new file mode 100644
index 00000000000..db43bea1090
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/model/CustomResponseTransformer.java
@@ -0,0 +1,122 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.model;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.easee.internal.Utils;
+import org.openhab.binding.easee.internal.handler.ChannelProvider;
+import org.openhab.core.library.types.OnOffType;
+import org.openhab.core.library.types.StringType;
+import org.openhab.core.thing.Channel;
+import org.openhab.core.types.State;
+
+import com.google.gson.JsonObject;
+
+/**
+ * transforms the http response into the openhab datamodel (instances of State).
+ * This class is used to handle special cases which cannot be mapped by the generic transformer.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+class CustomResponseTransformer {
+ private final ChannelProvider channelProvider;
+
+ CustomResponseTransformer(ChannelProvider channelProvider) {
+ this.channelProvider = channelProvider;
+ }
+
+ /**
+ * allows additional updates of special/composite channels.
+ *
+ * @param triggerChannel the channel which triggers the additional update
+ * @param value updated value of the triggering channel
+ * @param rawData raw json data provided by the API
+ */
+ Map transform(Channel triggerChannel, String value, JsonObject rawData) {
+ Map result = new HashMap<>(20);
+
+ switch (triggerChannel.getUID().getId()) {
+ case CHANNEL_GROUP_CHARGER_STATE + "#" + CHANNEL_CHARGER_OP_MODE:
+ updateChargerStartStop(result, value, rawData);
+ break;
+ case CHANNEL_GROUP_CHARGER_STATE + "#" + CHANNEL_CHARGER_DYNAMIC_CURRENT:
+ updateChargerPauseResume(result, value);
+ break;
+ case CHANNEL_GROUP_CIRCUIT_DYNAMIC_CURRENT + "#" + CHANNEL_CIRCUIT_DYNAMIC_CURRENT_PHASE1:
+ updateCompositePhaseChannel(result, rawData, CHANNEL_GROUP_CIRCUIT_DYNAMIC_CURRENT,
+ CHANNEL_CIRCUIT_DYNAMIC_CURRENTS, CHANNEL_CIRCUIT_DYNAMIC_CURRENT_PHASE1,
+ CHANNEL_CIRCUIT_DYNAMIC_CURRENT_PHASE2, CHANNEL_CIRCUIT_DYNAMIC_CURRENT_PHASE3);
+ break;
+ case CHANNEL_GROUP_CIRCUIT_SETTINGS + "#" + CHANNEL_CIRCUIT_MAX_CURRENT_PHASE1:
+ updateCompositePhaseChannel(result, rawData, CHANNEL_GROUP_CIRCUIT_SETTINGS,
+ CHANNEL_CIRCUIT_MAX_CURRENTS, CHANNEL_CIRCUIT_MAX_CURRENT_PHASE1,
+ CHANNEL_CIRCUIT_MAX_CURRENT_PHASE2, CHANNEL_CIRCUIT_MAX_CURRENT_PHASE3);
+ break;
+ case CHANNEL_GROUP_CIRCUIT_SETTINGS + "#" + CHANNEL_CIRCUIT_OFFLINE_MAX_CURRENT_PHASE1:
+ updateCompositePhaseChannel(result, rawData, CHANNEL_GROUP_CIRCUIT_SETTINGS,
+ CHANNEL_CIRCUIT_OFFLINE_MAX_CURRENTS, CHANNEL_CIRCUIT_OFFLINE_MAX_CURRENT_PHASE1,
+ CHANNEL_CIRCUIT_OFFLINE_MAX_CURRENT_PHASE2, CHANNEL_CIRCUIT_OFFLINE_MAX_CURRENT_PHASE3);
+ break;
+ }
+
+ return result;
+ }
+
+ private void updateChargerStartStop(Map result, String value, JsonObject rawData) {
+ Channel channel = channelProvider.getChannel(CHANNEL_GROUP_CHARGER_COMMANDS, CHANNEL_CHARGER_START_STOP);
+ if (channel != null) {
+ int val = Integer.valueOf(value);
+ // state >= 3 will mean charging, ready to charge or charging finished
+ boolean charging = val >= CHARGER_OP_STATE_CHARGING;
+
+ String rfnc = Utils.getAsString(rawData, CHANNEL_CHARGER_REASON_FOR_NO_CURRENT);
+ int reasonForNoCurrent = Integer.valueOf(rfnc == null ? "-1" : rfnc);
+ boolean paused = (val == CHARGER_OP_STATE_WAITING
+ && reasonForNoCurrent == CHARGER_REASON_FOR_NO_CURRENT_PAUSED);
+
+ result.put(channel, OnOffType.from(charging || paused));
+ }
+ }
+
+ private void updateChargerPauseResume(Map result, String value) {
+ Channel channel = channelProvider.getChannel(CHANNEL_GROUP_CHARGER_COMMANDS, CHANNEL_CHARGER_PAUSE_RESUME);
+ if (channel != null) {
+ int val = Integer.valueOf(value);
+ // value == 0 will mean paused
+ boolean paused = val == CHARGER_DYNAMIC_CURRENT_PAUSE;
+
+ result.put(channel, OnOffType.from(paused));
+ }
+ }
+
+ private void updateCompositePhaseChannel(Map result, JsonObject rawData, final String group,
+ final String targetChannel, final String channelPhase1, final String channelPhase2,
+ final String channelPhase3) {
+ Channel channel = channelProvider.getChannel(group, targetChannel);
+ String phase1 = Utils.getAsString(rawData, channelPhase1);
+ String phase2 = Utils.getAsString(rawData, channelPhase2);
+ String phase3 = Utils.getAsString(rawData, channelPhase3);
+ if (channel != null && phase1 != null && phase2 != null && phase3 != null) {
+ phase1 = phase1.replaceAll("\\.0", "");
+ phase2 = phase2.replaceAll("\\.0", "");
+ phase3 = phase3.replaceAll("\\.0", "");
+ result.put(channel, new StringType(phase1 + ";" + phase2 + ";" + phase3));
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/model/GenericResponseTransformer.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/model/GenericResponseTransformer.java
new file mode 100644
index 00000000000..a8e7a832867
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/model/GenericResponseTransformer.java
@@ -0,0 +1,127 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.model;
+
+import static org.openhab.binding.easee.internal.EaseeBindingConstants.*;
+
+import java.time.format.DateTimeParseException;
+import java.util.HashMap;
+import java.util.Map;
+
+import javax.measure.MetricPrefix;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.easee.internal.Utils;
+import org.openhab.binding.easee.internal.handler.ChannelProvider;
+import org.openhab.core.library.types.DateTimeType;
+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.Units;
+import org.openhab.core.thing.Channel;
+import org.openhab.core.types.State;
+import org.openhab.core.types.UnDefType;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import com.google.gson.JsonObject;
+
+/**
+ * transforms the http response into the openhab datamodel (instances of State)
+ * this is a generic trnasformer which tries to map json fields 1:1 to channels.
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class GenericResponseTransformer {
+ private final Logger logger = LoggerFactory.getLogger(GenericResponseTransformer.class);
+ private final ChannelProvider channelProvider;
+ private final CustomResponseTransformer customResponseTransformer;
+
+ public GenericResponseTransformer(ChannelProvider channelProvider) {
+ this.channelProvider = channelProvider;
+ this.customResponseTransformer = new CustomResponseTransformer(channelProvider);
+ }
+
+ public Map transform(JsonObject jsonData, String group) {
+ Map result = new HashMap<>(20);
+
+ for (String channelId : jsonData.keySet()) {
+ String value = Utils.getAsString(jsonData, channelId);
+
+ Channel channel = channelProvider.getChannel(group, channelId);
+ if (channel == null) {
+ // As we have a generic response mapper it ould happen that a subset of key/values in the response
+ // cannot be mapped to openhab channels.
+ logger.debug("Channel not found: {}#{}", group, channelId);
+ } else {
+ logger.debug("mapping value '{}' to channel {}", value, channel.getUID().getId());
+ String channelType = channel.getAcceptedItemType();
+
+ if (value == null || channelType == null) {
+ result.put(channel, UnDefType.NULL);
+ } else {
+ try {
+ switch (channelType) {
+ case CHANNEL_TYPE_SWITCH:
+ result.put(channel, OnOffType.from(Boolean.parseBoolean(value)));
+ break;
+ case CHANNEL_TYPE_VOLT:
+ result.put(channel, new QuantityType<>(Double.parseDouble(value), Units.VOLT));
+ break;
+ case CHANNEL_TYPE_AMPERE:
+ result.put(channel, new QuantityType<>(Double.parseDouble(value), Units.AMPERE));
+ break;
+ case CHANNEL_TYPE_KWH:
+ result.put(channel, new QuantityType<>(Double.parseDouble(value),
+ MetricPrefix.KILO(Units.WATT_HOUR)));
+ break;
+ case CHANNEL_TYPE_KW:
+ result.put(channel,
+ new QuantityType<>(Double.parseDouble(value), MetricPrefix.KILO(Units.WATT)));
+ break;
+ case CHANNEL_TYPE_DATE:
+ result.put(channel, new DateTimeType(Utils.parseDate(value)));
+ break;
+ case CHANNEL_TYPE_STRING:
+ result.put(channel, new StringType(value));
+ break;
+ case CHANNEL_TYPE_NUMBER:
+ if (Utils.getChannelTypeId(channel).contains(CHANNEL_TYPENAME_INTEGER)) {
+ // explicit type long is needed in case of integer/long values otherwise automatic
+ // transformation to a decimal type is applied.
+ result.put(channel, new DecimalType(Long.parseLong(value)));
+ } else {
+ result.put(channel, new DecimalType(Double.parseDouble(value)));
+ }
+ break;
+ default:
+ logger.warn("no mapping implemented for channel type '{}'", channelType);
+ }
+
+ // call the custom handler to handle specific / composite channels which do not map 1:1 to JSON
+ // fields.
+ result.putAll(customResponseTransformer.transform(channel, value, jsonData));
+
+ } catch (NumberFormatException | DateTimeParseException ex) {
+ logger.warn("caught exception while parsing data for channel {} (value '{}'). Exception: {}",
+ channel.getUID().getId(), value, ex.getMessage());
+ }
+ }
+ }
+ }
+
+ return result;
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/model/ValidationException.java b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/model/ValidationException.java
new file mode 100644
index 00000000000..d6f0d5ad317
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/java/org/openhab/binding/easee/internal/model/ValidationException.java
@@ -0,0 +1,41 @@
+/**
+ * Copyright (c) 2010-2022 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.easee.internal.model;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * exception whichs is used to state a validation error
+ *
+ * @author Alexander Friese - initial contribution
+ */
+@NonNullByDefault
+public class ValidationException extends Exception {
+ private static final long serialVersionUID = -6479556472780307224L;
+
+ public ValidationException() {
+ super();
+ }
+
+ public ValidationException(String message) {
+ super(message);
+ }
+
+ public ValidationException(Throwable cause) {
+ super(cause);
+ }
+
+ public ValidationException(String message, Throwable cause) {
+ super(message, cause);
+ }
+}
diff --git a/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/binding/binding.xml b/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/binding/binding.xml
new file mode 100644
index 00000000000..177514fdc7c
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/binding/binding.xml
@@ -0,0 +1,7 @@
+
+
+ Easee Binding
+ This is the binding for Easee Wallboxes.
+
diff --git a/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/config/config.xml b/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/config/config.xml
new file mode 100644
index 00000000000..60930c40b5d
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/config/config.xml
@@ -0,0 +1,56 @@
+
+
+
+
+
+ Authentication settings.
+
+
+
+ Connection settings.
+
+
+
+ General settings.
+
+
+
+
+ The username to login at Easee Cloud service. This should be an e-mail address or phone number.
+
+
+
+ Your password to login at Easee Cloud service.
+ password
+
+
+
+ The ID of the site containing the charger(s) and circuit(s) that should be integrated into openHAB.
+
+
+
+ Interval in which data is polled from EaseeCloud (in seconds).
+ 60
+
+
+
+
+
+ The ID of the charger that should be integrated into openHAB.
+
+
+
+
+
+ The ID of the charger that should be integrated into openHAB.
+
+
+
+ The ID of the circuit which is controlled by this charger.
+
+
+
diff --git a/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/i18n/easee.properties b/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/i18n/easee.properties
new file mode 100644
index 00000000000..198cefd2889
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/i18n/easee.properties
@@ -0,0 +1,167 @@
+# binding
+
+binding.easee.name = Easee Binding
+binding.easee.description = This is the binding for Easee Wallboxes.
+
+# thing types
+
+thing-type.easee.charger.label = Easee Charger
+thing-type.easee.charger.description = Cloud connected Easee Charger.
+thing-type.easee.mastercharger.label = Easee Master Charger
+thing-type.easee.mastercharger.description = Cloud connected Easee Master Charger which also controls the circuit.
+thing-type.easee.site.label = Easee Site
+thing-type.easee.site.description = Cloud connection to an Easee site.
+
+# thing types config
+
+thing-type.config.easee.charger.id.label = ID
+thing-type.config.easee.charger.id.description = The ID of the charger that should be integrated into openHAB.
+thing-type.config.easee.mastercharger.circuitId.label = Circuit ID
+thing-type.config.easee.mastercharger.circuitId.description = The ID of the circuit which is controlled by this charger.
+thing-type.config.easee.mastercharger.id.label = ID
+thing-type.config.easee.mastercharger.id.description = The ID of the charger that should be integrated into openHAB.
+thing-type.config.easee.site.dataPollingInterval.label = Polling Interval
+thing-type.config.easee.site.dataPollingInterval.description = Interval in which data is polled from EaseeCloud (in seconds).
+thing-type.config.easee.site.group.authentication.label = Authentication
+thing-type.config.easee.site.group.authentication.description = Authentication settings.
+thing-type.config.easee.site.group.connection.label = Connection
+thing-type.config.easee.site.group.connection.description = Connection settings.
+thing-type.config.easee.site.group.general.label = General
+thing-type.config.easee.site.group.general.description = General settings.
+thing-type.config.easee.site.password.label = Password
+thing-type.config.easee.site.password.description = Your password to login at Easee Cloud service.
+thing-type.config.easee.site.siteId.label = Site ID
+thing-type.config.easee.site.siteId.description = The ID of the site containing the charger(s) and circuit(s) that should be integrated into openHAB.
+thing-type.config.easee.site.username.label = Username
+thing-type.config.easee.site.username.description = The username to login at Easee Cloud service. This should be an e-mail address or phone number.
+
+# channel group types
+
+channel-group-type.easee.charger-commands.label = Charger Commands
+channel-group-type.easee.charger-commands.channel.genericCommand.label = Generic Charger Command
+channel-group-type.easee.charger-commands.channel.genericCommand.description = Sends a command to the charger. Write only channel.
+channel-group-type.easee.charger-commands.channel.pauseResume.label = Pause/Resume Charging
+channel-group-type.easee.charger-commands.channel.pauseResume.description = Pauses/Resumes charging. Pausing works by setting Dynamic Charger Current to 0.
+channel-group-type.easee.charger-commands.channel.startStop.label = Start/Stop Charging
+channel-group-type.easee.charger-commands.channel.startStop.description = Starts/Stops charging. Only relevant if authorization is required.
+channel-group-type.easee.charger-config.label = Charger Configuration
+channel-group-type.easee.charger-config.channel.authorizationRequired.label = Authorization Required
+channel-group-type.easee.charger-config.channel.authorizationRequired.description = Indicates if authorization is required to start charging.
+channel-group-type.easee.charger-config.channel.limitToSinglePhaseCharging.label = Limit To Single Phase Charging
+channel-group-type.easee.charger-config.channel.limitToSinglePhaseCharging.description = Indicates if charging should be limited to single phase mode.
+channel-group-type.easee.charger-config.channel.lockCablePermanently.label = Lock Cable Permanently
+channel-group-type.easee.charger-config.channel.lockCablePermanently.description = Lock Cable Permanently status of the wallbox.
+channel-group-type.easee.charger-config.channel.maxChargerCurrent.label = Max Charger Current
+channel-group-type.easee.charger-config.channel.maxChargerCurrent.description = Max charging current.
+channel-group-type.easee.charger-config.channel.phaseMode.label = Phase Mode
+channel-group-type.easee.charger-config.channel.phaseMode.description = Phase Mode.
+channel-group-type.easee.charger-latestSession.label = Last Charging Session
+channel-group-type.easee.charger-latestSession.channel.sessionEnd.label = Session End
+channel-group-type.easee.charger-latestSession.channel.sessionEnd.description = Date/Time when session ended.
+channel-group-type.easee.charger-latestSession.channel.sessionEnergy.label = Total Session Energy
+channel-group-type.easee.charger-latestSession.channel.sessionEnergy.description = Total Energy for last session.
+channel-group-type.easee.charger-latestSession.channel.sessionStart.label = Session Start
+channel-group-type.easee.charger-latestSession.channel.sessionStart.description = Date/Time when session started.
+channel-group-type.easee.charger-state.label = Charger Status
+channel-group-type.easee.charger-state.channel.cableLocked.label = Cable Locked
+channel-group-type.easee.charger-state.channel.cableLocked.description = Cable Locked status of the wallbox.
+channel-group-type.easee.charger-state.channel.chargerFirmware.label = Current Firmware
+channel-group-type.easee.charger-state.channel.chargerFirmware.description = Current Firmware of the wallbox.
+channel-group-type.easee.charger-state.channel.chargerOpMode.label = Charger Operation Mode
+channel-group-type.easee.charger-state.channel.chargerOpMode.description = Current operation mode.
+channel-group-type.easee.charger-state.channel.dynamicChargerCurrent.label = Dynamic Charger Current
+channel-group-type.easee.charger-state.channel.dynamicChargerCurrent.description = Dynamic set charging current.
+channel-group-type.easee.charger-state.channel.dynamicCircuitCurrentP1.label = Dynamic Circuit Current P1
+channel-group-type.easee.charger-state.channel.dynamicCircuitCurrentP1.description = Dynamic set circuit current for phase 1.
+channel-group-type.easee.charger-state.channel.dynamicCircuitCurrentP2.label = Dynamic Circuit Current P2
+channel-group-type.easee.charger-state.channel.dynamicCircuitCurrentP2.description = Dynamic set circuit current for phase 2.
+channel-group-type.easee.charger-state.channel.dynamicCircuitCurrentP3.label = Dynamic Circuit Current P3
+channel-group-type.easee.charger-state.channel.dynamicCircuitCurrentP3.description = Dynamic set circuit current for phase 3.
+channel-group-type.easee.charger-state.channel.errorCode.label = Error Code
+channel-group-type.easee.charger-state.channel.errorCode.description = Error Code.
+channel-group-type.easee.charger-state.channel.fatalErrorCode.label = Fatal Error Code
+channel-group-type.easee.charger-state.channel.fatalErrorCode.description = Fatal Error Code.
+channel-group-type.easee.charger-state.channel.isOnline.label = Online
+channel-group-type.easee.charger-state.channel.isOnline.description = Online status of the wallbox.
+channel-group-type.easee.charger-state.channel.latestFirmware.label = Latest Firmware
+channel-group-type.easee.charger-state.channel.latestFirmware.description = Latest Firmware which is available for the wallbox.
+channel-group-type.easee.charger-state.channel.latestPulse.label = Latest Pulse
+channel-group-type.easee.charger-state.channel.latestPulse.description = Last data received from charger.
+channel-group-type.easee.charger-state.channel.lifetimeEnergy.label = Lifetime Energy
+channel-group-type.easee.charger-state.channel.lifetimeEnergy.description = Total lifetime energy.
+channel-group-type.easee.charger-state.channel.outputCurrent.label = Output Current
+channel-group-type.easee.charger-state.channel.outputCurrent.description = Actual charging current.
+channel-group-type.easee.charger-state.channel.reasonForNoCurrent.label = Reason for no Current
+channel-group-type.easee.charger-state.channel.reasonForNoCurrent.description = Reason for no Current.
+channel-group-type.easee.charger-state.channel.sessionEnergy.label = Session Energy
+channel-group-type.easee.charger-state.channel.sessionEnergy.description = Energy for current session.
+channel-group-type.easee.charger-state.channel.smartCharging.label = Smart Charging
+channel-group-type.easee.charger-state.channel.smartCharging.description = Smart Charging status of the wallbox.
+channel-group-type.easee.charger-state.channel.totalPower.label = Total Power
+channel-group-type.easee.charger-state.channel.totalPower.description = Total Power for all active phases.
+channel-group-type.easee.charger-state.channel.voltage.label = Voltage
+channel-group-type.easee.charger-state.channel.voltage.description = Voltage
+channel-group-type.easee.circuit-dynamicCurrent.label = Circuit Dynamic Current
+channel-group-type.easee.circuit-dynamicCurrent.channel.dynamicCurrents.label = Dynamic Currents
+channel-group-type.easee.circuit-dynamicCurrent.channel.dynamicCurrents.description = Dynamic circuit currents for phases 1,2,3.
+channel-group-type.easee.circuit-dynamicCurrent.channel.phase1.label = Dynamic Current P1
+channel-group-type.easee.circuit-dynamicCurrent.channel.phase1.description = Dynamic set circuit current for phase 1.
+channel-group-type.easee.circuit-dynamicCurrent.channel.phase2.label = Dynamic Current P2
+channel-group-type.easee.circuit-dynamicCurrent.channel.phase2.description = Dynamic set circuit current for phase 2.
+channel-group-type.easee.circuit-dynamicCurrent.channel.phase3.label = Dynamic Current P3
+channel-group-type.easee.circuit-dynamicCurrent.channel.phase3.description = Dynamic set circuit current for phase 3.
+channel-group-type.easee.circuit-settings.label = Circuit Settings
+channel-group-type.easee.circuit-settings.channel.allowOfflineMaxCircuitCurrent.label = Allow Offline Max Circuit Current
+channel-group-type.easee.circuit-settings.channel.allowOfflineMaxCircuitCurrent.description = Allow Offline Max Circuit Current.
+channel-group-type.easee.circuit-settings.channel.enableIdleCurrent.label = Enable Idle Current
+channel-group-type.easee.circuit-settings.channel.enableIdleCurrent.description = This will block 6A Idle current for each charger in the current.
+channel-group-type.easee.circuit-settings.channel.maxCircuitCurrentP1.label = Max Current P1
+channel-group-type.easee.circuit-settings.channel.maxCircuitCurrentP1.description = Max circuit current for phase 1.
+channel-group-type.easee.circuit-settings.channel.maxCircuitCurrentP2.label = Max Current P2
+channel-group-type.easee.circuit-settings.channel.maxCircuitCurrentP2.description = Max circuit current for phase 2.
+channel-group-type.easee.circuit-settings.channel.maxCircuitCurrentP3.label = Max Current P3
+channel-group-type.easee.circuit-settings.channel.maxCircuitCurrentP3.description = Max circuit current for phase 3.
+channel-group-type.easee.circuit-settings.channel.maxCurrents.label = Max Currents
+channel-group-type.easee.circuit-settings.channel.maxCurrents.description = Max circuit currents for phases 1,2,3.
+channel-group-type.easee.circuit-settings.channel.offlineMaxCircuitCurrentP1.label = Offline Max Current P1
+channel-group-type.easee.circuit-settings.channel.offlineMaxCircuitCurrentP1.description = Max circuit current for phase 1 in offline mode.
+channel-group-type.easee.circuit-settings.channel.offlineMaxCircuitCurrentP2.label = Offline Max Current P2
+channel-group-type.easee.circuit-settings.channel.offlineMaxCircuitCurrentP2.description = Max circuit current for phase 2 in offline mode.
+channel-group-type.easee.circuit-settings.channel.offlineMaxCircuitCurrentP3.label = Offline Max Current P3
+channel-group-type.easee.circuit-settings.channel.offlineMaxCircuitCurrentP3.description = Max circuit current for phase 3 in offline mode.
+channel-group-type.easee.circuit-settings.channel.offlineMaxCurrents.label = Offline Max Currents
+channel-group-type.easee.circuit-settings.channel.offlineMaxCurrents.description = Offline Max circuit currents for phases 1,2,3.
+
+# channel types
+
+channel-type.easee.rwtype-charger-command.label = Charger Command
+channel-type.easee.rwtype-current.label = Electric Current
+channel-type.easee.rwtype-currents.label = Currents Phase 1;2;3
+channel-type.easee.rwtype-integer-phase-mode.label = Phase Mode
+channel-type.easee.rwtype-integer-phase-mode.state.option.1 = 1-Phase
+channel-type.easee.rwtype-integer-phase-mode.state.option.2 = Auto
+channel-type.easee.rwtype-integer-phase-mode.state.option.3 = 3-Phase
+channel-type.easee.rwtype-switch.label = Switch
+channel-type.easee.type-current.label = ElectricCurrent
+channel-type.easee.type-date.label = DateTime
+channel-type.easee.type-energy.label = Energy
+channel-type.easee.type-integer-charger-op-mode.label = Charger Operation Mode
+channel-type.easee.type-integer-charger-op-mode.state.option.0 = Offline
+channel-type.easee.type-integer-charger-op-mode.state.option.1 = Disconnected
+channel-type.easee.type-integer-charger-op-mode.state.option.2 = AwaitingStart
+channel-type.easee.type-integer-charger-op-mode.state.option.3 = Charging
+channel-type.easee.type-integer-charger-op-mode.state.option.4 = Completed
+channel-type.easee.type-integer-charger-op-mode.state.option.5 = Error
+channel-type.easee.type-integer-charger-op-mode.state.option.6 = ReadyToCharge
+channel-type.easee.type-integer.label = Generic Integer
+channel-type.easee.type-power.label = Power
+channel-type.easee.type-switch.label = Switch
+channel-type.easee.type-volt.label = ElectricPotential
+
+# status translations
+
+status.token.validated = "Access token validated"
+status.waiting.for.bridge = "Waiting for bridge to go online"
+status.waiting.for.login = "Waiting for web api login"
+status.no.valid.data = "No valid data received. This is most likely a configuration error"
+status.no.connection = "Charger might have no internet connection or fuse tripped"
diff --git a/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/thing/charger-channel-groups.xml b/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/thing/charger-channel-groups.xml
new file mode 100644
index 00000000000..184a9b266ce
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/thing/charger-channel-groups.xml
@@ -0,0 +1,179 @@
+
+
+
+
+
+
+
+ Smart Charging status of the wallbox.
+
+
+
+ Cable Locked status of the wallbox.
+
+
+
+ Current operation mode.
+
+
+
+ Total Power for all active phases.
+
+
+
+ Energy for current session.
+
+
+
+ Dynamic set circuit current for phase 1.
+
+
+
+ Dynamic set circuit current for phase 2.
+
+
+
+ Dynamic set circuit current for phase 3.
+
+
+
+ Last data received from charger.
+
+
+
+ Current Firmware of the wallbox.
+
+
+
+ Latest Firmware which is available for the wallbox.
+
+
+
+ Voltage
+
+
+
+ Actual charging current.
+
+
+
+ Online status of the wallbox.
+
+
+
+ Dynamic set charging current.
+
+ ChangeConfiguration
+ (0|6|7|8|9|10|11|12|13|14|15|16|17|18|19|20|21|22|23|24|25|26|27|28|29|30|31|32)
+
+
+
+
+ Reason for no Current.
+
+
+
+ Total lifetime energy.
+
+
+
+ Error Code.
+
+
+
+ Fatal Error Code.
+
+
+
+
+
+
+
+
+ Lock Cable Permanently status of the wallbox.
+
+ ChangeConfiguration
+ .*
+
+
+
+
+ Indicates if authorization is required to start charging.
+
+ ChangeConfiguration
+ .*
+
+
+
+
+ Indicates if charging should be limited to single phase mode.
+
+ ChangeConfiguration
+ .*
+
+
+
+
+ Phase Mode.
+
+ ChangeConfiguration
+ (1|2|3)
+
+
+
+
+ Max charging current.
+
+
+
+
+
+
+
+
+ Total Energy for last session.
+
+
+
+ Date/Time when session started.
+
+
+
+ Date/Time when session ended.
+
+
+
+
+
+
+
+
+ Sends a command to the charger. Write only channel.
+
+ SendCommand
+ (reboot|update_firmware|poll_all|smart_charging|start_charging|stop_charging|pause_charging|resume_charging|toggle_charging|override_schedule)
+
+
+
+
+ Starts/Stops charging. Only relevant if authorization is required.
+
+ SendCommandStartStop
+ .*
+
+
+
+
+ Pauses/Resumes charging. Pausing works by setting Dynamic Charger Current to 0.
+
+ SendCommandPauseResume
+ .*
+
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/thing/circuit-channel-groups.xml b/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/thing/circuit-channel-groups.xml
new file mode 100644
index 00000000000..838dc097f1e
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/thing/circuit-channel-groups.xml
@@ -0,0 +1,88 @@
+
+
+
+
+
+
+
+ Max circuit current for phase 1.
+
+
+
+ Max circuit current for phase 2.
+
+
+
+ Max circuit current for phase 3.
+
+
+
+ Max circuit currents for phases 1,2,3.
+
+ SetMaxCircuitCurrents
+ (0|6|7|8|9|10|11|12|13|14|15|16|17|18|19|20|21|22|23|24|25|26|27|28|29|30|31|32);(0|6|7|8|9|10|11|12|13|14|15|16|17|18|19|20|21|22|23|24|25|26|27|28|29|30|31|32);(0|6|7|8|9|10|11|12|13|14|15|16|17|18|19|20|21|22|23|24|25|26|27|28|29|30|31|32)
+
+
+
+
+ Max circuit current for phase 1 in offline mode.
+
+
+
+ Max circuit current for phase 2 in offline mode.
+
+
+
+ Max circuit current for phase 3 in offline mode.
+
+
+
+ Offline Max circuit currents for phases 1,2,3.
+
+ SetOfflineMaxCircuitCurrents
+ (0|6|7|8|9|10|11|12|13|14|15|16|17|18|19|20|21|22|23|24|25|26|27|28|29|30|31|32);(0|6|7|8|9|10|11|12|13|14|15|16|17|18|19|20|21|22|23|24|25|26|27|28|29|30|31|32);(0|6|7|8|9|10|11|12|13|14|15|16|17|18|19|20|21|22|23|24|25|26|27|28|29|30|31|32)
+
+
+
+
+ This will block 6A Idle current for each charger in the current.
+
+ SetCircuitSettings
+ .*
+
+
+
+
+ Allow Offline Max Circuit Current.
+
+
+
+
+
+
+
+
+ Dynamic set circuit current for phase 1.
+
+
+
+ Dynamic set circuit current for phase 2.
+
+
+
+ Dynamic set circuit current for phase 3.
+
+
+
+ Dynamic circuit currents for phases 1,2,3.
+
+ SetDynamicCircuitCurrents
+ (0|6|7|8|9|10|11|12|13|14|15|16|17|18|19|20|21|22|23|24|25|26|27|28|29|30|31|32);(0|6|7|8|9|10|11|12|13|14|15|16|17|18|19|20|21|22|23|24|25|26|27|28|29|30|31|32);(0|6|7|8|9|10|11|12|13|14|15|16|17|18|19|20|21|22|23|24|25|26|27|28|29|30|31|32)
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/thing/easee-readonly-channel-types.xml b/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/thing/easee-readonly-channel-types.xml
new file mode 100644
index 00000000000..3f2e613fab3
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/thing/easee-readonly-channel-types.xml
@@ -0,0 +1,62 @@
+
+
+
+ Number:Power
+
+
+
+
+
+ Number:Energy
+
+
+
+
+
+ Number:ElectricCurrent
+
+
+
+
+
+ Switch
+
+
+
+
+ Number:ElectricPotential
+
+
+
+
+
+ Number
+
+
+
+
+
+ Number
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ DateTime
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/thing/easee-readwrite-channel-types.xml b/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/thing/easee-readwrite-channel-types.xml
new file mode 100644
index 00000000000..5c0fbaadf10
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/thing/easee-readwrite-channel-types.xml
@@ -0,0 +1,40 @@
+
+
+
+ Switch
+
+
+
+
+ Number:ElectricCurrent
+
+
+
+
+
+ Number
+
+
+
+
+
+
+
+
+
+
+ String
+
+
+
+
+
+ String
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/thing/things.xml b/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/thing/things.xml
new file mode 100644
index 00000000000..d0f1049eaf4
--- /dev/null
+++ b/bundles/org.openhab.binding.easee/src/main/resources/OH-INF/thing/things.xml
@@ -0,0 +1,41 @@
+
+
+
+
+ Cloud connection to an Easee site.
+
+
+
+
+
+
+
+ Cloud connected Easee Master Charger which also controls the circuit.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Cloud connected Easee Charger.
+
+
+
+
+
+
+
+
+
diff --git a/bundles/pom.xml b/bundles/pom.xml
index 6f924d28826..99ec72c728c 100644
--- a/bundles/pom.xml
+++ b/bundles/pom.xml
@@ -109,6 +109,7 @@
org.openhab.binding.dsmr
org.openhab.binding.dwdpollenflug
org.openhab.binding.dwdunwetter
+ org.openhab.binding.easee
org.openhab.binding.ecobee
org.openhab.binding.ecotouch
org.openhab.binding.ekey