diff --git a/bom/openhab-addons/pom.xml b/bom/openhab-addons/pom.xml
index f28e8b7bbe8..739c821ddd4 100644
--- a/bom/openhab-addons/pom.xml
+++ b/bom/openhab-addons/pom.xml
@@ -546,6 +546,11 @@
org.openhab.binding.groheondus
${project.version}
+
+ org.openhab.addons.bundles
+ org.openhab.binding.groupepsa
+ ${project.version}
+
org.openhab.addons.bundles
org.openhab.binding.guntamatic
diff --git a/bundles/org.openhab.binding.groupepsa/NOTICE b/bundles/org.openhab.binding.groupepsa/NOTICE
new file mode 100644
index 00000000000..cdc134b4460
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/NOTICE
@@ -0,0 +1,20 @@
+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
+
+== Third-party Content
+
+GeoJSON support for Gson
+* License: Apache License 2.0
+* Project: https://github.com/filosganga/geogson
+* Source: https://github.com/filosganga/geogson
diff --git a/bundles/org.openhab.binding.groupepsa/README.md b/bundles/org.openhab.binding.groupepsa/README.md
new file mode 100644
index 00000000000..590164803ab
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/README.md
@@ -0,0 +1,151 @@
+# Groupe PSA Binding
+
+Binding to retrieve information via the Groupe PSA Web API for cars from Opel, Peugeot, Citroen, DS and Vauxhall.
+
+## Supported Things
+
+bridge - Groupe PSA Web Api Bridge: The Thing to auto discover your cars
+
+vehicle - Groupe PSA Car: The actual car thing.
+
+## Discovery
+
+Use the "Groupe PSA Web Api bridge" to auto discover your cars. You need to select the brand for the bridge binding and only cars for the brand will be auto discovered. If you need to add for multiple brands or multiple different users, add multiple bridges.
+
+## Bridge Configuration
+
+You need to select a brand and enter the User Name and Password.
+The Polling interval (in minutes) determines how often the API will polled for new cars.
+The Client ID and Client Secret should not need to be updated. (However you can register your own app via https://developer.groupe-psa.com/inc/ and use this client information if you wish.)
+
+### parameters
+
+| Property | Default | Required | Description |
+| --------------- | ------- | -------- | ------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| vendor | None | Yes | The brand of the car (PEUGEOT, CITROEN, DS, OPEL or VAUXHALL) |
+| userName | None | Yes | The user name for the mypeugot/mycitroen/myds/myopel/myvauxhall website or app |
+| password | None | Yes | The password for the given user |
+| pollingInterval | 60 | No | The Polling interval (in minutes) determines how often the available vehicles are queried |
+| clientId | | Yes | The client ID for API access: can normally left at the default value. (see: https://developer.groupe-psa.io/webapi/b2c/quickstart/connect/#article) |
+| clientSecret | | Yes | The client secret for API access: can normally left at the default value. (see: https://developer.groupe-psa.io/webapi/b2c/quickstart/connect/#article) |
+
+## Vehicle Configuration
+
+Normally the vehicles will be autodiscovered. The Polling Interval and Online Timeout can be adjusted.
+
+### parameters
+
+| Property | Default | Required | Description |
+| --------------- | ------- | -------- | ------------------------------------------------------------------------------------------------ | ------------------------------------------------------------------------------------------ |
+| id | None | Yes | Vehicle API ID | The ID is the vehicle API ID (not equal to the VIN), which is autodiscoverd by the bridge. |
+| pollingInterval | 5 | No | The Polling interval (in minutes) determines how often the car is polled for updated information |
+| onlineInterval | 15 | No | The Online Timeout (in minutes) determines when the car is deemed to be offline. |
+
+## Channels
+
+| Channel Type ID | Item Type | Description |
+| ----------------------- | ------------------------- | ------------------------------------------------ |
+| current | Number:ElectricCurrent | Electrical current |
+| voltage | Number:ElectricPotential | Voltage |
+| temperature | Number:Temperature | Temperature |
+| daytime | Contact | Enabled if it is daytime |
+| doorLock | String | Door lock state |
+| doorOpen | Contact | Door is open |
+| ignition | String | Ignition state |
+| moving | Contact | Vehicle is moving |
+| acceleration | Number:Acceleration | Current acceleration |
+| speed | Number:Speed | Current speed |
+| mileage | Number:Length | Total travelled distance |
+| position | Location | Last known position |
+| heading | Number:Angle | Direction of travel |
+| type | String | Position acquisition type |
+| signal | Number:Dimensionless | Strength of the position localization signal |
+| lastUpdated | DateTime | Last time the results were updated on the server |
+| privacy | String | Privacy status |
+| belt | String | Seat belt status |
+| emergency | String | Emergency call status |
+| service | String | Service Type |
+| preconditioning | String | Air conditioning status |
+| preconditioningFailure | String | Airr conditioning failure cause |
+| level | Number:Dimensionless | Fuel level |
+| autonomy | Number:Length | Remaining distance |
+| consumption | Number:VolumetricFlowRate | Fuel consumption |
+| residual | Number:Energy | Remaining battery charge |
+| capacity | Number:Energy | Battery capacity |
+| healthCapacity | Number:Dimensionless | Health of the battery capacity |
+| healthResistance | Number:Dimensionless | Health of the battery resistance |
+| chargingStatus | String | Battery charging status |
+| chargingMode | String | Battery charging mode |
+| chargingPlugged | Contact | Vehicle plugged in to charger |
+| chargingRate | Number:Speed | Battery Charging Rate |
+| chargingRemainingTime | Number:Time | Time remaining till charged |
+| chargingNextDelayedTime | Number:Time | Time till the next charging starts |
+
+Further documentation can be found at: https://developer.groupe-psa.io/webapi/b2c/api-reference/specification/#article
+
+## Full Example
+
+### Things file
+
+```
+Bridge groupepsa:bridge:opel "Auto Interface" [
+ pollingInterval=60,
+ userName="anonymous@anonymous.email",
+ password="password",
+ vendor="OPEL"
+] {
+ Things:
+ vehicle zafira "Auto" @ "Outdoors"
+ [
+ id="",
+ pollingInterval=5,
+ onlineInterval=1440
+ ]
+
+}
+```
+
+### Items file
+
+```
+Group Auto
+
+Number:ElectricCurrent Auto_Aux_Current "Auxillliary Battery Current [%.1f %unit%]" (Auto) ["Measurement","Current"] {channel="groupepsa:vehicle:opel:zafira:battery#current"}
+Number Auto_Aux_Level "Auxillliary Battery Level [%.1f %unit%]" (Auto) ["Measurement","Level"] {channel="groupepsa:vehicle:opel:zafira:battery#voltage"}
+
+Number:Temperature Auto_Outside_Temperature "Outside Temperature [%.1f %unit%]" (Auto) ["Measurement","Temperature"] {channel="groupepsa:vehicle:opel:zafira:environment#temperature"}
+
+Number:Length Auto_Mileage "Mileage [%.1f %unit%]" (Auto) ["Measurement"] {channel="groupepsa:vehicle:opel:zafira:motion#mileage"}
+Number:Speed Auto_Speed "Speed [%.1f %unit%]" (Auto) ["Measurement"] {channel="groupepsa:vehicle:opel:zafira:motion#speed"}
+Number:Acceleration Auto_Acceleration "Acceleration [%.1f %unit%]" (Auto) ["Measurement"] {channel="groupepsa:vehicle:opel:zafira:motion#acceleration"}
+
+Location Auto_Location "Locatie" (Auto) ["Point"] {channel="groupepsa:vehicle:opel:zafira:position#position"}
+
+Number:Angle Auto_Heading "Richting" (Auto) ["Measurement"] {channel="groupepsa:vehicle:opel:zafira:position#heading"}
+String Auto_Location_Type "Locatie Type" (Auto) ["Measurement"] {channel="groupepsa:vehicle:opel:zafira:position#type"}
+Number Auto_Signal_Strength "Signaal Sterkte [%.1f %unit%]" (Auto) ["Measurement"] {channel="groupepsa:vehicle:opel:zafira:position#signal"}
+
+DateTime Auto_Last_Update "Laatst bijgewerkt [%1$tA, %1$te %1$tb %1$tY %1$tR]" (Auto) ["Measurement", "Timestamp"] {channel="groupepsa:vehicle:opel:zafira:various#lastUpdated"}
+String Auto_Privacy_Status "Privacy Status" (Auto) ["Status"] {channel="groupepsa:vehicle:opel:zafira:various#privacyStatus"}
+String Auto_Belt_Status "Belt Status" (Auto) ["Status"] {channel="groupepsa:vehicle:opel:zafira:various#beltStatus"}
+String Auto_Emergency_Call "Emergency Call" (Auto) ["Status"] {channel="groupepsa:vehicle:opel:zafira:various#emergencyCall"}
+String Auto_Service "Service" (Auto) ["Status"] {channel="groupepsa:vehicle:opel:zafira:various#service"}
+String Auto_Air_Conditioning "Air Conditioning" (Auto) ["Status"] {channel="groupepsa:vehicle:opel:zafira:various#preconditioning"}
+String Auto_Air_Conditioning_Failure "Air Conditioning Failure" (Auto) ["Status"] {channel="groupepsa:vehicle:opel:zafira:various#preconditioningFailure"}
+
+Number:Length Auto_Autonomy "Autonomy [%.1f %unit%]" (Auto) ["Measurement"] {channel="groupepsa:vehicle:opel:zafira:electric#autonomy"}
+Number Auto_Level "Battery Level [%.1f %unit%]" (Auto) ["Measurement", "Level"] {channel="groupepsa:vehicle:opel:zafira:electric#level"}
+Number:Energy Auto_Residual "Battery Residual [%.1f %unit%]" (Auto) ["Measurement", "Level"] {channel="groupepsa:vehicle:opel:zafira:electric#residual"}
+
+Number:Energy Auto_Capacity "Battery Capacity [%.1f %unit%]" (Auto) ["Measurement"] {channel="groupepsa:vehicle:opel:zafira:electric#capacity"}
+Number Auto_Health_Capacity "Battery Health Capacity [%.1f %unit%]" (Auto) ["Measurement"] {channel="groupepsa:vehicle:opel:zafira:electric#batteryHealthCapacity"}
+Number Auto_Health_Resistance "Battery Health Resistance [%.1f %unit%]" (Auto) ["Measurement"] {channel="groupepsa:vehicle:opel:zafira:electric#batteryHealthResistance"}
+
+String Auto_Charging_Status "Charging Status" (Auto) ["Status"] {channel="groupepsa:vehicle:opel:zafira:electric#chargingStatus"}
+String Auto_Charging_Mode "Charging Mode" (Auto) ["Status"] {channel="groupepsa:vehicle:opel:zafira:electric#chargingMode"}
+Contact Auto_Plugged_In "Plugged In" (Auto) ["Status"] {channel="groupepsa:vehicle:opel:zafira:electric#chargingPlugged"}
+Number:Speed Auto_Charging_Rate "Charging Rate [%.1f %unit%]" (Auto) ["Measurement"] {channel="groupepsa:vehicle:opel:zafira:electric#chargingRate"}
+
+Number:Time Auto_Charging_Time_Remaining "Charging Time Remaining [%.1f %unit%]" (Auto) ["Measurement", "Duration"] {channel="groupepsa:vehicle:opel:zafira:electric#chargingRemainingTime"}
+Number:Time Auto_Charging_Time_Till_Next "Charging Time Till Next Charging [%.1f %unit%]" (Auto) ["Measurement", "Duration"] {channel="groupepsa:vehicle:opel:zafira:electric#chargingNextDelayedTime"}
+```
diff --git a/bundles/org.openhab.binding.groupepsa/pom.xml b/bundles/org.openhab.binding.groupepsa/pom.xml
new file mode 100644
index 00000000000..b384ef993c9
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/pom.xml
@@ -0,0 +1,25 @@
+
+
+
+ 4.0.0
+
+
+ org.openhab.addons.bundles
+ org.openhab.addons.reactor.bundles
+ 3.3.0-SNAPSHOT
+
+
+ org.openhab.binding.groupepsa
+
+ openHAB Add-ons :: Bundles :: Groupe PSA Binding
+
+
+
+ com.github.filosganga
+ geogson-core
+ 1.4.31
+
+
+
+
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/feature/feature.xml b/bundles/org.openhab.binding.groupepsa/src/main/feature/feature.xml
new file mode 100644
index 00000000000..fc30b11e051
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/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.groupepsa/${project.version}
+
+
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/GroupePSABindingConstants.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/GroupePSABindingConstants.java
new file mode 100644
index 00000000000..d2e7623351d
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/GroupePSABindingConstants.java
@@ -0,0 +1,109 @@
+/**
+ * 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.groupepsa.internal;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.core.thing.ThingTypeUID;
+
+/**
+ * The {@link GroupePSABindingConstants} class defines common constants, which
+ * are used across the whole binding.
+ *
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class GroupePSABindingConstants {
+
+ public static final String BINDING_ID = "groupepsa";
+
+ // List of all Thing Type UIDs
+ public static final ThingTypeUID THING_TYPE_BRIDGE = new ThingTypeUID(BINDING_ID, "bridge");
+ public static final ThingTypeUID THING_TYPE_VEHICLE = new ThingTypeUID(BINDING_ID, "vehicle");
+
+ // Vehicle properties
+ public static final String VEHICLE_ID = "id";
+ public static final String VEHICLE_VIN = "vin";
+ public static final String VEHICLE_VENDOR = "vendor";
+ public static final String VEHICLE_MODEL = "model";
+
+ // List of all Channel ids
+ public static final String CHANNEL_BATTERY_CURRENT = "battery#current";
+ public static final String CHANNEL_BATTERY_VOLTAGE = "battery#voltage";
+
+ public static final String CHANNEL_TYPE_DOORLOCK = "doorLock";
+ public static final String CHANNEL_TYPE_DOOROPEN = "doorOpen";
+
+ public static final String CHANNEL_GROUP_DOORS = "doors";
+ public static final String CHANNEL_DOORS_LOCK = "doors#locked";
+
+ public static final String CHANNEL_ENVIRONMENT_TEMPERATURE = "environment#temperature";
+ public static final String CHANNEL_ENVIRONMENT_DAYTIME = "environment#daytime";
+
+ public static final String CHANNEL_MOTION_IGNITION = "motion#ignition";
+ public static final String CHANNEL_MOTION_ACCELERATION = "motion#acceleration";
+ public static final String CHANNEL_MOTION_MOVING = "motion#moving";
+ public static final String CHANNEL_MOTION_SPEED = "motion#speed";
+ public static final String CHANNEL_MOTION_MILEAGE = "motion#mileage";
+
+ public static final String CHANNEL_POSITION_POSITION = "position#position";
+ public static final String CHANNEL_POSITION_HEADING = "position#heading";
+ public static final String CHANNEL_POSITION_TYPE = "position#type";
+ public static final String CHANNEL_POSITION_SIGNALSTRENGTH = "position#signal";
+
+ public static final String CHANNEL_VARIOUS_LAST_UPDATED = "various#lastUpdated";
+ public static final String CHANNEL_VARIOUS_PRIVACY = "various#privacy";
+ public static final String CHANNEL_VARIOUS_BELT = "various#belt";
+ public static final String CHANNEL_VARIOUS_EMERGENCY = "various#emergency";
+ public static final String CHANNEL_VARIOUS_SERVICE = "various#service";
+ public static final String CHANNEL_VARIOUS_PRECONDITINING = "various#preconditioning";
+ public static final String CHANNEL_VARIOUS_PRECONDITINING_FAILURE = "various#preconditioningFailure";
+
+ public static final String CHANNEL_FUEL_AUTONOMY = "fuel#autonomy";
+ public static final String CHANNEL_FUEL_CONSUMPTION = "fuel#consumption";
+ public static final String CHANNEL_FUEL_LEVEL = "fuel#level";
+
+ public static final String CHANNEL_ELECTRIC_AUTONOMY = "electric#autonomy";
+ public static final String CHANNEL_ELECTRIC_LEVEL = "electric#level";
+ public static final String CHANNEL_ELECTRIC_RESIDUAL = "electric#residual";
+
+ public static final String CHANNEL_ELECTRIC_BATTERY_CAPACITY = "electric#batteryCapacity";
+ public static final String CHANNEL_ELECTRIC_BATTERY_HEALTH_CAPACITY = "electric#batteryHealthCapacity";
+ public static final String CHANNEL_ELECTRIC_BATTERY_HEALTH_RESISTANCE = "electric#batteryHealthResistance";
+
+ public static final String CHANNEL_ELECTRIC_CHARGING_STATUS = "electric#chargingStatus";
+ public static final String CHANNEL_ELECTRIC_CHARGING_MODE = "electric#chargingMode";
+ public static final String CHANNEL_ELECTRIC_CHARGING_PLUGGED = "electric#chargingPlugged";
+ public static final String CHANNEL_ELECTRIC_CHARGING_RATE = "electric#chargingRate";
+ public static final String CHANNEL_ELECTRIC_CHARGING_REMAININGTIME = "electric#chargingRemainingTime";
+ public static final String CHANNEL_ELECTRIC_CHARGING_NEXTDELAYEDTIME = "electric#chargingNextDelayedTime";
+
+ public enum VendorConstants {
+ PEUGEOT("https://idpcvs.peugeot.com/am/oauth2/access_token", "clientsB2CPeugeot"),
+ CITROEN("https://idpcvs.citroen.com/am/oauth2/access_token", "clientsB2CCitroen"),
+ DS("https://idpcvs.driveds.com/am/oauth2/access_token", "clientsB2CDS"),
+ OPEL("https://idpcvs.opel.com/am/oauth2/access_token", "clientsB2COpel"),
+ VAUXHALL("https://idpcvs.vauxhall.co.uk/am/oauth2/access_token", "clientsB2CVauxhall");
+
+ public final String url;
+ public final String realm;
+ public final String scope;
+
+ VendorConstants(String url, String realm) {
+ this.url = url;
+ this.realm = realm;
+ this.scope = "profile openid";
+ }
+ }
+
+ public static final String API_URL = "https://api.groupe-psa.com/connectedcar/v4";
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/GroupePSAHandlerFactory.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/GroupePSAHandlerFactory.java
new file mode 100644
index 00000000000..4db404a27dd
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/GroupePSAHandlerFactory.java
@@ -0,0 +1,75 @@
+/**
+ * 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.groupepsa.internal;
+
+import static org.openhab.binding.groupepsa.internal.GroupePSABindingConstants.*;
+
+import java.util.Set;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.eclipse.jetty.client.HttpClient;
+import org.openhab.binding.groupepsa.internal.bridge.GroupePSABridgeHandler;
+import org.openhab.binding.groupepsa.internal.things.GroupePSAHandler;
+import org.openhab.core.auth.client.oauth2.OAuthFactory;
+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;
+
+/**
+ * The {@link GroupePSAHandlerFactory} is responsible for creating things and
+ * thing handlers.
+ *
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+@Component(configurationPid = "binding.groupepsa", service = ThingHandlerFactory.class)
+public class GroupePSAHandlerFactory extends BaseThingHandlerFactory {
+
+ private static final Set SUPPORTED_THING_TYPES_UIDS = Set.of(THING_TYPE_BRIDGE, THING_TYPE_VEHICLE);
+
+ private final OAuthFactory oAuthFactory;
+ protected final HttpClient httpClient;
+
+ @Activate
+ public GroupePSAHandlerFactory(@Reference OAuthFactory oAuthFactory,
+ @Reference HttpClientFactory httpClientFactory) {
+ this.oAuthFactory = oAuthFactory;
+ 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_VEHICLE.equals(thingTypeUID)) {
+ return new GroupePSAHandler(thing);
+ } else if (THING_TYPE_BRIDGE.equals(thingTypeUID)) {
+ return new GroupePSABridgeHandler((Bridge) thing, oAuthFactory, httpClient);
+ }
+
+ return null;
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/bridge/GroupePSABridgeConfiguration.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/bridge/GroupePSABridgeConfiguration.java
new file mode 100644
index 00000000000..861bfaef5bf
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/bridge/GroupePSABridgeConfiguration.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.groupepsa.internal.bridge;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * The {@link GroupePSABridgeConfiguration} class contains fields mapping thing configuration parameters.
+ *
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public final class GroupePSABridgeConfiguration {
+ private String vendor = "";
+ private String userName = "";
+ private String password = "";
+ private String clientId = "";
+ private String clientSecret = "";
+
+ private Integer pollingInterval = 5;
+
+ /**
+ * @return The polling interval for the groupepsa state in s
+ */
+ public Integer getPollingInterval() {
+ return pollingInterval;
+ }
+
+ public void setPollingInterval(Integer pollingInterval) {
+ this.pollingInterval = pollingInterval;
+ }
+
+ public String getVendor() {
+ return vendor;
+ }
+
+ public void setVendor(String vendor) {
+ this.vendor = vendor;
+ }
+
+ 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 getClientId() {
+ return clientId;
+ }
+
+ public void setClientId(String clientId) {
+ this.clientId = clientId;
+ }
+
+ public String getClientSecret() {
+ return clientSecret;
+ }
+
+ public void setClientSecret(String clientSecret) {
+ this.clientSecret = clientSecret;
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/bridge/GroupePSABridgeHandler.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/bridge/GroupePSABridgeHandler.java
new file mode 100644
index 00000000000..b5d02e9a7a6
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/bridge/GroupePSABridgeHandler.java
@@ -0,0 +1,225 @@
+/**
+ * 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.groupepsa.internal.bridge;
+
+import static org.openhab.binding.groupepsa.internal.GroupePSABindingConstants.THING_TYPE_BRIDGE;
+import static org.openhab.binding.groupepsa.internal.GroupePSABindingConstants.VendorConstants;
+
+import java.io.IOException;
+import java.util.Collection;
+import java.util.Collections;
+import java.util.List;
+import java.util.Set;
+import java.util.concurrent.ScheduledFuture;
+import java.util.concurrent.TimeUnit;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.eclipse.jetty.client.HttpClient;
+import org.openhab.binding.groupepsa.internal.discovery.GroupePSADiscoveryService;
+import org.openhab.binding.groupepsa.internal.rest.api.GroupePSAConnectApi;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Vehicle;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.VehicleStatus;
+import org.openhab.binding.groupepsa.internal.rest.exceptions.GroupePSACommunicationException;
+import org.openhab.core.auth.client.oauth2.AccessTokenResponse;
+import org.openhab.core.auth.client.oauth2.OAuthClientService;
+import org.openhab.core.auth.client.oauth2.OAuthException;
+import org.openhab.core.auth.client.oauth2.OAuthFactory;
+import org.openhab.core.auth.client.oauth2.OAuthResponseException;
+import org.openhab.core.thing.Bridge;
+import org.openhab.core.thing.ChannelUID;
+import org.openhab.core.thing.ThingStatus;
+import org.openhab.core.thing.ThingStatusDetail;
+import org.openhab.core.thing.ThingTypeUID;
+import org.openhab.core.thing.binding.BaseBridgeHandler;
+import org.openhab.core.thing.binding.ThingHandlerService;
+import org.openhab.core.types.Command;
+
+/**
+ * The {@link GroupePSABridgeHandler} is responsible for handling commands,
+ * which are sent to one of the channels.
+ *
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class GroupePSABridgeHandler extends BaseBridgeHandler {
+ public static final Set SUPPORTED_THING_TYPES = Collections.singleton(THING_TYPE_BRIDGE);
+ private static final long DEFAULT_POLLING_INTERVAL_M = TimeUnit.HOURS.toMinutes(1);
+
+ private final OAuthFactory oAuthFactory;
+
+ private @Nullable OAuthClientService oAuthService;
+ private @Nullable ScheduledFuture> groupepsaBridgePollingJob;
+ private final HttpClient httpClient;
+
+ private String vendor = "";
+ private @Nullable VendorConstants vendorConstants;
+ private String userName = "";
+ private String password = "";
+ private String clientId = "";
+ private String clientSecret = "";
+
+ private @Nullable GroupePSAConnectApi groupePSAApi;
+
+ public GroupePSABridgeHandler(Bridge bridge, OAuthFactory oAuthFactory, HttpClient httpClient) {
+ super(bridge);
+ this.oAuthFactory = oAuthFactory;
+ this.httpClient = httpClient;
+ }
+
+ private void pollGroupePSAs() {
+ try {
+ List vehicles = getVehicles();
+ if (vehicles != null) {
+ updateStatus(ThingStatus.ONLINE);
+ } else {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR,
+ "@text/comm-error-query-vehicles-failed");
+ }
+ } catch (GroupePSACommunicationException e) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR, e.getMessage());
+ }
+ }
+
+ @Override
+ public void dispose() {
+ stopGroupePSABridgePolling();
+ oAuthFactory.ungetOAuthService(thing.getUID().getAsString());
+ }
+
+ @Override
+ public void initialize() {
+ GroupePSABridgeConfiguration bridgeConfiguration = getConfigAs(GroupePSABridgeConfiguration.class);
+
+ vendor = bridgeConfiguration.getVendor();
+ userName = bridgeConfiguration.getUserName();
+ password = bridgeConfiguration.getPassword();
+ clientId = bridgeConfiguration.getClientId();
+ clientSecret = bridgeConfiguration.getClientSecret();
+
+ final Integer pollingIntervalM = bridgeConfiguration.getPollingInterval();
+
+ if (vendor.isEmpty()) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR, "@text/conf-error-no-vendor");
+ } else if (userName.isEmpty()) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR, "@text/conf-error-no-username");
+ } else if (password.isEmpty()) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR, "@text/conf-error-no-password");
+ } else if (clientId.isEmpty()) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR, "@text/conf-error-no-clientid");
+ } else if (clientSecret.isEmpty()) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
+ "@text/conf-error-no-clientsecret");
+ } else if (pollingIntervalM < 1) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
+ "@text/conf-error-invalid-polling-interval");
+ } else {
+ VendorConstants localVendorConstants = VendorConstants.valueOf(vendor);
+ vendorConstants = localVendorConstants;
+
+ oAuthService = oAuthFactory.createOAuthClientService(thing.getUID().getAsString(), localVendorConstants.url,
+ null, clientId, clientSecret, localVendorConstants.scope, true);
+
+ groupePSAApi = new GroupePSAConnectApi(httpClient, this, clientId, localVendorConstants.realm);
+
+ startGroupePSABridgePolling(pollingIntervalM);
+
+ updateStatus(ThingStatus.UNKNOWN);
+ }
+ }
+
+ private void startGroupePSABridgePolling(@Nullable Integer pollingIntervalM) {
+ if (groupepsaBridgePollingJob == null) {
+ final long pollingIntervalToUse = pollingIntervalM == null ? DEFAULT_POLLING_INTERVAL_M : pollingIntervalM;
+ groupepsaBridgePollingJob = scheduler.scheduleWithFixedDelay(() -> pollGroupePSAs(), 1,
+ TimeUnit.MINUTES.toSeconds(pollingIntervalToUse), TimeUnit.SECONDS);
+ }
+ }
+
+ private void stopGroupePSABridgePolling() {
+ final ScheduledFuture> job = groupepsaBridgePollingJob;
+ if (job != null) {
+ job.cancel(true);
+ groupepsaBridgePollingJob = null;
+ }
+ }
+
+ @Override
+ public void handleCommand(ChannelUID channelUID, Command command) {
+ }
+
+ static Throwable getRootCause(Throwable e) {
+ Throwable nextE;
+ do {
+ nextE = e.getCause();
+ if (nextE != null) {
+ e = nextE;
+ }
+ } while (nextE != null);
+ return e;
+ }
+
+ public String authenticate() throws GroupePSACommunicationException {
+ OAuthClientService localOAuthService = oAuthService;
+ VendorConstants localVendorConstants = vendorConstants;
+ if (localOAuthService == null) {
+ throw new GroupePSACommunicationException("OAuth service is unexpectedly null");
+ }
+ if (localVendorConstants == null) {
+ throw new GroupePSACommunicationException("Vendor constants are unexpectedly null");
+ }
+ try {
+ AccessTokenResponse result = localOAuthService.getAccessTokenResponse();
+ if (result == null) {
+ result = localOAuthService.getAccessTokenByResourceOwnerPasswordCredentials(userName, password,
+ localVendorConstants.scope);
+ }
+ return result.getAccessToken();
+ } catch (OAuthException | IOException | OAuthResponseException e) {
+ throw new GroupePSACommunicationException("Unable to authenticate: " + getRootCause(e).getMessage(), e);
+ }
+ }
+
+ public GroupePSAConnectApi getAPI() throws GroupePSACommunicationException {
+ GroupePSAConnectApi localGroupePSAApi = groupePSAApi;
+ if (localGroupePSAApi == null) {
+ throw new GroupePSACommunicationException("groupePSAApi is unexpectedly null");
+ } else {
+ return localGroupePSAApi;
+ }
+ }
+
+ /**
+ * @return A list of vehicles
+ * @throws GroupePSACommunicationException In case the query cannot be executed
+ * successfully
+ */
+ public @Nullable List getVehicles() throws GroupePSACommunicationException {
+ return getAPI().getVehicles();
+ }
+
+ /**
+ * @param id The id of the mower to query
+ * @return A detailed status of the mower with the specified id
+ * @throws GroupePSACommunicationException In case the query cannot be executed
+ * successfully
+ */
+ public @Nullable VehicleStatus getVehicleStatus(String vin) throws GroupePSACommunicationException {
+ return getAPI().getVehicleStatus(vin);
+ }
+
+ @Override
+ public Collection> getServices() {
+ return Collections.singleton(GroupePSADiscoveryService.class);
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/discovery/GroupePSADiscoveryService.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/discovery/GroupePSADiscoveryService.java
new file mode 100644
index 00000000000..1c55d530aed
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/discovery/GroupePSADiscoveryService.java
@@ -0,0 +1,118 @@
+/**
+ * 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.groupepsa.internal.discovery;
+
+import static org.openhab.binding.groupepsa.internal.GroupePSABindingConstants.THING_TYPE_VEHICLE;
+
+import java.util.Collections;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.groupepsa.internal.GroupePSABindingConstants;
+import org.openhab.binding.groupepsa.internal.bridge.GroupePSABridgeHandler;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Vehicle;
+import org.openhab.binding.groupepsa.internal.rest.exceptions.GroupePSACommunicationException;
+import org.openhab.core.config.discovery.AbstractDiscoveryService;
+import org.openhab.core.config.discovery.DiscoveryResult;
+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.osgi.service.component.annotations.Component;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * The {@link GroupePSADiscoveryService} is responsible for discovering new
+ * vehicles available for the configured app key.
+ *
+ * @author Arjan Mels - Initial contribution
+ */
+@Component(service = ThingHandlerService.class)
+@NonNullByDefault
+public class GroupePSADiscoveryService extends AbstractDiscoveryService implements ThingHandlerService {
+ private final Logger logger = LoggerFactory.getLogger(GroupePSADiscoveryService.class);
+
+ private @Nullable GroupePSABridgeHandler bridgeHandler;
+
+ public GroupePSADiscoveryService() {
+ super(Collections.singleton(THING_TYPE_VEHICLE), 10, false);
+ }
+
+ @Override
+ public void setThingHandler(@Nullable ThingHandler handler) {
+ if (handler instanceof GroupePSABridgeHandler) {
+ bridgeHandler = (GroupePSABridgeHandler) handler;
+ }
+ }
+
+ @Override
+ public @Nullable ThingHandler getThingHandler() {
+ return bridgeHandler;
+ }
+
+ @Override
+ public void deactivate() {
+ super.deactivate();
+ }
+
+ @Override
+ protected void startScan() {
+ try {
+ GroupePSABridgeHandler localBridgeHandler = bridgeHandler;
+ if (localBridgeHandler == null) {
+ return;
+ }
+ List vehicles = localBridgeHandler.getVehicles();
+ if (vehicles == null || vehicles.isEmpty()) {
+ logger.warn("No vehicles found");
+ return;
+ }
+ for (Vehicle vehicle : vehicles) {
+ ThingUID bridgeUID = localBridgeHandler.getThing().getUID();
+ ThingTypeUID thingTypeUID = THING_TYPE_VEHICLE;
+ String id = vehicle.getId();
+ if (id != null) {
+ ThingUID vehicleThingUid = new ThingUID(THING_TYPE_VEHICLE, bridgeUID, id);
+
+ Map properties = new HashMap<>();
+ putProperty(properties, GroupePSABindingConstants.VEHICLE_ID, id);
+ putProperty(properties, GroupePSABindingConstants.VEHICLE_VIN, vehicle.getVin());
+ putProperty(properties, GroupePSABindingConstants.VEHICLE_VENDOR, vehicle.getBrand());
+ putProperty(properties, GroupePSABindingConstants.VEHICLE_MODEL, vehicle.getLabel());
+
+ DiscoveryResult discoveryResult = DiscoveryResultBuilder.create(vehicleThingUid)
+ .withThingType(thingTypeUID).withProperties(properties).withBridge(bridgeUID)
+ .withRepresentationProperty(GroupePSABindingConstants.VEHICLE_VIN)
+ .withLabel(vehicle.getBrand() + " (" + vehicle.getVin() + ")").build();
+
+ thingDiscovered(discoveryResult);
+ }
+ }
+ } catch (GroupePSACommunicationException e) {
+ logger.warn("No vehicles found", e);
+ return;
+ }
+ }
+
+ private void putProperty(Map properties, String key, @Nullable String value) {
+ if (value == null) {
+ value = "Unknown";
+ }
+ properties.put(key, value);
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/GroupePSAConnectApi.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/GroupePSAConnectApi.java
new file mode 100644
index 00000000000..15bfcff264a
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/GroupePSAConnectApi.java
@@ -0,0 +1,200 @@
+/**
+ * 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.groupepsa.internal.rest.api;
+
+import static org.openhab.binding.groupepsa.internal.GroupePSABindingConstants.API_URL;
+
+import java.lang.reflect.Type;
+import java.time.Duration;
+import java.time.ZonedDateTime;
+import java.util.List;
+import java.util.concurrent.ExecutionException;
+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.ContentResponse;
+import org.eclipse.jetty.client.api.Request;
+import org.eclipse.jetty.http.HttpMethod;
+import org.eclipse.jetty.http.HttpStatus;
+import org.openhab.binding.groupepsa.internal.bridge.GroupePSABridgeHandler;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.ErrorObject;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.User;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Vehicle;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.VehicleStatus;
+import org.openhab.binding.groupepsa.internal.rest.exceptions.GroupePSACommunicationException;
+import org.openhab.binding.groupepsa.internal.rest.exceptions.UnauthorizedException;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import com.github.filosganga.geogson.gson.GeometryAdapterFactory;
+import com.google.gson.Gson;
+import com.google.gson.GsonBuilder;
+import com.google.gson.JsonDeserializationContext;
+import com.google.gson.JsonDeserializer;
+import com.google.gson.JsonElement;
+import com.google.gson.JsonParseException;
+import com.google.gson.JsonSyntaxException;
+
+/**
+ * Allows access to the GroupePSAConnectApi
+ *
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class GroupePSAConnectApi {
+ private final Logger logger = LoggerFactory.getLogger(GroupePSAConnectApi.class);
+
+ private final HttpClient httpClient;
+ private final GroupePSABridgeHandler bridge;
+ private final String clientId;
+ private final String realm;
+
+ protected final Gson gson;
+
+ public GroupePSAConnectApi(HttpClient httpClient, GroupePSABridgeHandler bridge, String clientId, String realm) {
+ this.httpClient = httpClient;
+ this.bridge = bridge;
+ this.clientId = clientId;
+ this.realm = realm;
+
+ gson = new GsonBuilder().registerTypeAdapterFactory(new GeometryAdapterFactory())
+ .registerTypeAdapter(ZonedDateTime.class, new JsonDeserializer() {
+ @Override
+ public @Nullable ZonedDateTime deserialize(JsonElement json, Type typeOfT,
+ JsonDeserializationContext context) throws JsonParseException {
+ return ZonedDateTime.parse(json.getAsJsonPrimitive().getAsString());
+ }
+ }).registerTypeAdapter(Duration.class, new JsonDeserializer() {
+ @Override
+ public @Nullable Duration deserialize(JsonElement json, Type typeOfT,
+ JsonDeserializationContext context) throws JsonParseException {
+ return Duration.parse(json.getAsJsonPrimitive().getAsString());
+ }
+ }).create();
+ }
+
+ protected HttpClient getHttpClient() {
+ return httpClient;
+ }
+
+ protected GroupePSABridgeHandler getBridge() {
+ return bridge;
+ }
+
+ public String getBaseUrl() {
+ return API_URL;
+ }
+
+ private ContentResponse executeRequest(final String uri) throws GroupePSACommunicationException {
+ return executeRequest(uri, "application/hal+json");
+ }
+
+ static Throwable getRootCause(Throwable e) {
+ Throwable nextE;
+ do {
+ nextE = e.getCause();
+ if (nextE != null) {
+ e = nextE;
+ }
+ } while (nextE != null);
+ return e;
+ }
+
+ public ContentResponse executeRequest(final String uri, final String accept)
+ throws GroupePSACommunicationException {
+ Request request = getHttpClient().newRequest(uri);
+
+ String token = getBridge().authenticate();
+
+ request.timeout(10, TimeUnit.SECONDS);
+
+ request.param("client_id", this.clientId);
+
+ request.header("Authorization", "Bearer " + token);
+ request.header("Accept", accept);
+ request.header("x-introspect-realm", this.realm);
+
+ request.method(HttpMethod.GET);
+
+ logger.trace("HttpRequest {}", request.getURI());
+ logger.trace("HttpRequest Headers:\n{}", request.getHeaders());
+
+ try {
+ ContentResponse response = request.send();
+ logger.trace("HttpResponse {}", response);
+ logger.trace("HttpResponse Headers:\n{}", response.getHeaders());
+ logger.trace("HttpResponse Content: {}", response.getContentAsString());
+ return response;
+ } catch (InterruptedException | TimeoutException | ExecutionException e) {
+ throw new GroupePSACommunicationException("Unable to perform Http Request: " + getRootCause(e).getMessage(),
+ e);
+ }
+ }
+
+ private void checkForError(ContentResponse response, int statusCode) throws GroupePSACommunicationException {
+ if (statusCode >= 200 && statusCode < 300) {
+ return;
+ }
+
+ switch (statusCode) {
+ case HttpStatus.NOT_FOUND_404:
+ ErrorObject error = gson.fromJson(response.getContentAsString(), ErrorObject.class);
+ String message = (error != null) ? error.getMessage() : null;
+ if (message == null) {
+ message = "Unknown";
+ }
+ throw new GroupePSACommunicationException(statusCode, message);
+
+ case HttpStatus.FORBIDDEN_403:
+ case HttpStatus.UNAUTHORIZED_401:
+ throw new UnauthorizedException(statusCode, response.getContentAsString());
+
+ default:
+ throw new GroupePSACommunicationException(statusCode, response.getContentAsString());
+ }
+ }
+
+ private @Nullable T parseResponse(ContentResponse response, Class type)
+ throws GroupePSACommunicationException {
+ int statusCode = response.getStatus();
+
+ checkForError(response, statusCode);
+
+ try {
+ return gson.fromJson(response.getContentAsString(), type);
+ } catch (JsonSyntaxException e) {
+ throw new GroupePSACommunicationException("Error in received JSON: " + getRootCause(e).getMessage(), e);
+ }
+ }
+
+ public @Nullable List getVehicles() throws GroupePSACommunicationException {
+ ContentResponse response = executeRequest(getBaseUrl() + "/user");
+ User user = parseResponse(response, User.class);
+
+ if (user != null) {
+ return user.getVehicles();
+ } else {
+ return null;
+ }
+ }
+
+ public @Nullable VehicleStatus getVehicleStatus(String vin) throws GroupePSACommunicationException {
+ ContentResponse responseOdometer = executeRequest(getBaseUrl() + "/user/vehicles/" + vin + "/status");
+ VehicleStatus status = parseResponse(responseOdometer, VehicleStatus.class);
+
+ return status;
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Air.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Air.java
new file mode 100644
index 00000000000..ff25eb9b34e
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Air.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.groupepsa.internal.rest.api.dto;
+
+import java.math.BigDecimal;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Air {
+
+ private @Nullable BigDecimal temp;
+
+ public @Nullable BigDecimal getTemp() {
+ return temp;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("temp", temp).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/AirConditioning.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/AirConditioning.java
new file mode 100644
index 00000000000..0b6e5704ef6
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/AirConditioning.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.groupepsa.internal.rest.api.dto;
+
+import java.time.ZonedDateTime;
+import java.util.List;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class AirConditioning {
+
+ private @Nullable String failureCause;
+ private @Nullable List programs = null;
+ private @Nullable String status;
+ private @Nullable ZonedDateTime updatedAt;
+
+ public @Nullable String getFailureCause() {
+ return failureCause;
+ }
+
+ public @Nullable List getPrograms() {
+ return programs;
+ }
+
+ public @Nullable String getStatus() {
+ return status;
+ }
+
+ public @Nullable ZonedDateTime getUpdatedAt() {
+ return updatedAt;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("failureCause", failureCause).append("programs", programs)
+ .append("status", status).append("updatedAt", updatedAt).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Battery.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Battery.java
new file mode 100644
index 00000000000..472ad0d764e
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Battery.java
@@ -0,0 +1,48 @@
+/**
+ * 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.groupepsa.internal.rest.api.dto;
+
+import java.math.BigDecimal;
+import java.time.ZonedDateTime;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Battery {
+
+ private @Nullable ZonedDateTime createdAt;
+ private @Nullable BigDecimal current;
+ private @Nullable BigDecimal voltage;
+
+ public @Nullable ZonedDateTime getCreatedAt() {
+ return createdAt;
+ }
+
+ public @Nullable BigDecimal getCurrent() {
+ return current;
+ }
+
+ public @Nullable BigDecimal getVoltage() {
+ return voltage;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("createdAt", createdAt).append("current", current)
+ .append("voltage", voltage).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/BatteryStatus.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/BatteryStatus.java
new file mode 100644
index 00000000000..9014bce8f53
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/BatteryStatus.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.groupepsa.internal.rest.api.dto;
+
+import java.math.BigDecimal;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class BatteryStatus {
+
+ private @Nullable BigDecimal capacity;
+ private @Nullable Health health;
+
+ public @Nullable BigDecimal getCapacity() {
+ return capacity;
+ }
+
+ public @Nullable Health getHealth() {
+ return health;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("capacity", capacity).append("health", health).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Charging.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Charging.java
new file mode 100644
index 00000000000..d6ce386beed
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Charging.java
@@ -0,0 +1,68 @@
+/**
+ * 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.groupepsa.internal.rest.api.dto;
+
+import java.math.BigDecimal;
+import java.time.Duration;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Charging {
+
+ private @Nullable String chargingMode;
+ private @Nullable BigDecimal chargingRate;
+ private @Nullable Duration nextDelayedTime;
+ private @Nullable Boolean plugged;
+ private @Nullable Duration remainingTime;
+ private @Nullable String status;
+
+ public @Nullable String getChargingMode() {
+ return chargingMode;
+ }
+
+ public @Nullable BigDecimal getChargingRate() {
+ return chargingRate;
+ }
+
+ public void setChargingRate(BigDecimal chargingRate) {
+ this.chargingRate = chargingRate;
+ }
+
+ public @Nullable Duration getNextDelayedTime() {
+ return nextDelayedTime;
+ }
+
+ public @Nullable Boolean isPlugged() {
+ return plugged;
+ }
+
+ public @Nullable Duration getRemainingTime() {
+ return remainingTime;
+ }
+
+ public @Nullable String getStatus() {
+ return status;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("chargingMode", chargingMode).append("chargingRate", chargingRate)
+ .append("nextDelayedTime", nextDelayedTime).append("plugged", plugged)
+ .append("remainingTime", remainingTime).append("status", status).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/DoorsState.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/DoorsState.java
new file mode 100644
index 00000000000..3c22b842113
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/DoorsState.java
@@ -0,0 +1,48 @@
+/**
+ * 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.groupepsa.internal.rest.api.dto;
+
+import java.time.ZonedDateTime;
+import java.util.List;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class DoorsState {
+
+ private @Nullable List lockedState = null;
+ private @Nullable List opening = null;
+ private @Nullable ZonedDateTime updatedAt;
+
+ public @Nullable List getLockedState() {
+ return lockedState;
+ }
+
+ public @Nullable List getOpening() {
+ return opening;
+ }
+
+ public @Nullable ZonedDateTime getUpdatedAt() {
+ return updatedAt;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("lockedState", lockedState).append("opening", opening)
+ .append("updatedAt", updatedAt).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Energy.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Energy.java
new file mode 100644
index 00000000000..5f272989463
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Energy.java
@@ -0,0 +1,80 @@
+/**
+ * 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.groupepsa.internal.rest.api.dto;
+
+import java.math.BigDecimal;
+import java.time.ZonedDateTime;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Energy {
+
+ private @Nullable ZonedDateTime updatedAt;
+ private @Nullable ZonedDateTime createdAt;
+ private @Nullable BigDecimal autonomy;
+ private @Nullable BatteryStatus battery;
+ private @Nullable Charging charging;
+ private @Nullable BigDecimal consumption;
+ private @Nullable BigDecimal level;
+ private @Nullable BigDecimal residual;
+ private @Nullable String type;
+
+ public @Nullable ZonedDateTime getUpdatedAt() {
+ return updatedAt;
+ }
+
+ public @Nullable ZonedDateTime getCreatedAt() {
+ return createdAt;
+ }
+
+ public @Nullable BigDecimal getAutonomy() {
+ return autonomy;
+ }
+
+ public @Nullable BatteryStatus getBattery() {
+ return battery;
+ }
+
+ public @Nullable Charging getCharging() {
+ return charging;
+ }
+
+ public @Nullable BigDecimal getConsumption() {
+ return consumption;
+ }
+
+ public @Nullable BigDecimal getLevel() {
+ return level;
+ }
+
+ public @Nullable BigDecimal getResidual() {
+ return residual;
+ }
+
+ public @Nullable String getType() {
+ return type;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("updatedAt", updatedAt).append("createdAt", createdAt)
+ .append("autonomy", autonomy).append("battery", battery).append("charging", charging)
+ .append("consumption", consumption).append("level", level).append("residual", residual)
+ .append("type", type).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Engine.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Engine.java
new file mode 100644
index 00000000000..7f9e9ee1ae1
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Engine.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.groupepsa.internal.rest.api.dto;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Engine {
+ @SerializedName("class")
+ private @Nullable String type;
+ private @Nullable String energy;
+
+ public @Nullable String getType() {
+ return type;
+ }
+
+ public @Nullable String getEnergy() {
+ return energy;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("type", type).append("energy", energy).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Environment.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Environment.java
new file mode 100644
index 00000000000..e4274020012
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Environment.java
@@ -0,0 +1,52 @@
+/**
+ * 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.groupepsa.internal.rest.api.dto;
+
+import java.time.ZonedDateTime;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Environment {
+
+ private @Nullable ZonedDateTime updatedAt;
+ private @Nullable ZonedDateTime createdAt;
+ private @Nullable Air air;
+ private @Nullable Luminosity luminosity;
+
+ public @Nullable ZonedDateTime getUpdatedAt() {
+ return updatedAt;
+ }
+
+ public @Nullable ZonedDateTime getCreatedAt() {
+ return createdAt;
+ }
+
+ public @Nullable Air getAir() {
+ return air;
+ }
+
+ public @Nullable Luminosity getLuminosity() {
+ return luminosity;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("updatedAt", updatedAt).append("createdAt", createdAt)
+ .append("air", air).append("luminosity", luminosity).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/ErrorObject.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/ErrorObject.java
new file mode 100644
index 00000000000..6debe370536
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/ErrorObject.java
@@ -0,0 +1,44 @@
+/**
+ * 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.groupepsa.internal.rest.api.dto;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class ErrorObject {
+ private @Nullable String uuid;
+ private @Nullable String code;
+ private @Nullable String message;
+
+ public @Nullable String getUuid() {
+ return uuid;
+ }
+
+ public @Nullable String getMessage() {
+ return message;
+ }
+
+ public @Nullable String getCode() {
+ return code;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("uuid", uuid).append("code", code).append("message", message)
+ .toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Health.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Health.java
new file mode 100644
index 00000000000..2d7578ec5f7
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Health.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.groupepsa.internal.rest.api.dto;
+
+import java.math.BigDecimal;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Health {
+
+ private @Nullable BigDecimal capacity;
+ private @Nullable BigDecimal resistance;
+
+ public @Nullable BigDecimal getCapacity() {
+ return capacity;
+ }
+
+ public @Nullable BigDecimal getResistance() {
+ return resistance;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("capacity", capacity).append("resistance", resistance).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Ignition.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Ignition.java
new file mode 100644
index 00000000000..a0d9b426467
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Ignition.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.groupepsa.internal.rest.api.dto;
+
+import java.time.ZonedDateTime;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Ignition {
+
+ private @Nullable ZonedDateTime updatedAt;
+ private @Nullable String type;
+
+ public @Nullable ZonedDateTime getUpdatedAt() {
+ return updatedAt;
+ }
+
+ public @Nullable String getType() {
+ return type;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("updatedAt", updatedAt).append("type", type).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Kinetic.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Kinetic.java
new file mode 100644
index 00000000000..9b7992f4d68
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Kinetic.java
@@ -0,0 +1,58 @@
+/**
+ * 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.groupepsa.internal.rest.api.dto;
+
+import java.math.BigDecimal;
+import java.time.ZonedDateTime;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Kinetic {
+
+ private @Nullable ZonedDateTime createdAt;
+ private @Nullable BigDecimal acceleration;
+ private @Nullable Boolean moving;
+ private @Nullable BigDecimal pace;
+ private @Nullable BigDecimal speed;
+
+ public @Nullable ZonedDateTime getCreatedAt() {
+ return createdAt;
+ }
+
+ public @Nullable BigDecimal getAcceleration() {
+ return acceleration;
+ }
+
+ public @Nullable Boolean isMoving() {
+ return moving;
+ }
+
+ public @Nullable BigDecimal getPace() {
+ return pace;
+ }
+
+ public @Nullable BigDecimal getSpeed() {
+ return speed;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("createdAt", createdAt).append("acceleration", acceleration)
+ .append("moving", moving).append("pace", pace).append("speed", speed).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Luminosity.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Luminosity.java
new file mode 100644
index 00000000000..252ce93e496
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Luminosity.java
@@ -0,0 +1,34 @@
+/**
+ * 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.groupepsa.internal.rest.api.dto;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Luminosity {
+
+ private @Nullable Boolean day;
+
+ public @Nullable Boolean isDay() {
+ return day;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("day", day).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Occurence.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Occurence.java
new file mode 100644
index 00000000000..166f0cdf917
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Occurence.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.groupepsa.internal.rest.api.dto;
+
+import java.util.List;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Occurence {
+
+ private @Nullable List day = null;
+ private @Nullable List week = null;
+
+ public @Nullable List getDay() {
+ return day;
+ }
+
+ public @Nullable List getWeek() {
+ return week;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("day", day).append("week", week).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Odometer.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Odometer.java
new file mode 100644
index 00000000000..319385c7f4d
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Odometer.java
@@ -0,0 +1,42 @@
+/**
+ * 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.groupepsa.internal.rest.api.dto;
+
+import java.math.BigDecimal;
+import java.time.ZonedDateTime;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Odometer {
+
+ private @Nullable ZonedDateTime createdAt;
+ private @Nullable BigDecimal mileage;
+
+ public @Nullable ZonedDateTime getCreatedAt() {
+ return createdAt;
+ }
+
+ public @Nullable BigDecimal getMileage() {
+ return mileage;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("createdAt", createdAt).append("mileage", mileage).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Opening.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Opening.java
new file mode 100644
index 00000000000..73d8b9f6899
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Opening.java
@@ -0,0 +1,39 @@
+/**
+ * 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.groupepsa.internal.rest.api.dto;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Opening {
+
+ private @Nullable String identifier;
+ private @Nullable String state;
+
+ public @Nullable String getIdentifier() {
+ return identifier;
+ }
+
+ public @Nullable String getState() {
+ return state;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("identifier", identifier).append("state", state).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Position.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Position.java
new file mode 100644
index 00000000000..12218d84b5e
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Position.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.groupepsa.internal.rest.api.dto;
+
+import java.time.ZonedDateTime;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.github.filosganga.geogson.model.Geometry;
+import com.github.filosganga.geogson.model.positions.SinglePosition;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Position {
+
+ private @Nullable ZonedDateTime updatedAt;
+ private @Nullable ZonedDateTime createdAt;
+ private @Nullable Geometry geometry;
+ private @Nullable Properties properties;
+ private @Nullable String type;
+
+ public @Nullable ZonedDateTime getUpdatedAt() {
+ return updatedAt;
+ }
+
+ public @Nullable ZonedDateTime getCreatedAt() {
+ return createdAt;
+ }
+
+ public @Nullable Geometry getGeometry() {
+ return geometry;
+ }
+
+ public @Nullable Properties getProperties() {
+ return properties;
+ }
+
+ public @Nullable String getType() {
+ return type;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("updatedAt", updatedAt).append("createdAt", createdAt)
+ .append("geometry", geometry).append("properties", properties).append("type", type).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Preconditionning.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Preconditionning.java
new file mode 100644
index 00000000000..75a8cc4f273
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Preconditionning.java
@@ -0,0 +1,34 @@
+/**
+ * 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.groupepsa.internal.rest.api.dto;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Preconditionning {
+
+ private @Nullable AirConditioning airConditioning;
+
+ public @Nullable AirConditioning getAirConditioning() {
+ return airConditioning;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("airConditioning", airConditioning).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Privacy.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Privacy.java
new file mode 100644
index 00000000000..7c97fab6c1d
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Privacy.java
@@ -0,0 +1,42 @@
+/**
+ * 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.groupepsa.internal.rest.api.dto;
+
+import java.time.ZonedDateTime;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Privacy {
+
+ private @Nullable ZonedDateTime createdAt;
+ private @Nullable String state;
+
+ public @Nullable ZonedDateTime getCreatedAt() {
+ return createdAt;
+ }
+
+ public @Nullable String getState() {
+ return state;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("createdAt", createdAt).append("state", state).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Program.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Program.java
new file mode 100644
index 00000000000..503dc682260
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Program.java
@@ -0,0 +1,57 @@
+/**
+ * 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.groupepsa.internal.rest.api.dto;
+
+import java.math.BigDecimal;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Program {
+
+ private @Nullable Occurence occurence;
+ private @Nullable String recurrence;
+ private @Nullable String start;
+ private @Nullable Boolean enabled;
+ private @Nullable BigDecimal slot;
+
+ public @Nullable Occurence getOccurence() {
+ return occurence;
+ }
+
+ public @Nullable String getRecurrence() {
+ return recurrence;
+ }
+
+ public @Nullable String getStart() {
+ return start;
+ }
+
+ public @Nullable Boolean isEnabled() {
+ return enabled;
+ }
+
+ public @Nullable BigDecimal getSlot() {
+ return slot;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("occurence", occurence).append("recurrence", recurrence)
+ .append("start", start).append("enabled", enabled).append("slot", slot).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Properties.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Properties.java
new file mode 100644
index 00000000000..9ec7f657fdb
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Properties.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.groupepsa.internal.rest.api.dto;
+
+import java.math.BigDecimal;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Properties {
+
+ private @Nullable BigDecimal heading;
+ private @Nullable BigDecimal signalQuality;
+ private @Nullable String type;
+
+ public @Nullable BigDecimal getHeading() {
+ return heading;
+ }
+
+ public @Nullable BigDecimal getSignalQuality() {
+ return signalQuality;
+ }
+
+ public @Nullable String getType() {
+ return type;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("heading", heading).append("signalQuality", signalQuality)
+ .append("type", type).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Safety.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Safety.java
new file mode 100644
index 00000000000..cd9a6a830cb
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Safety.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.groupepsa.internal.rest.api.dto;
+
+import java.time.ZonedDateTime;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Safety {
+
+ private @Nullable ZonedDateTime createdAt;
+ private @Nullable String beltWarning;
+ private @Nullable String eCallTriggeringRequest;
+
+ public @Nullable ZonedDateTime getCreatedAt() {
+ return createdAt;
+ }
+
+ public @Nullable String getBeltWarning() {
+ return beltWarning;
+ }
+
+ public @Nullable String getECallTriggeringRequest() {
+ return eCallTriggeringRequest;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("createdAt", createdAt).append("beltWarning", beltWarning)
+ .append("eCallTriggeringRequest", eCallTriggeringRequest).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Service.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Service.java
new file mode 100644
index 00000000000..20eecd712c2
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Service.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.groupepsa.internal.rest.api.dto;
+
+import java.time.ZonedDateTime;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Service {
+
+ private @Nullable String type;
+ private @Nullable ZonedDateTime updatedAt;
+
+ public @Nullable String getType() {
+ return type;
+ }
+
+ public @Nullable ZonedDateTime getUpdatedAt() {
+ return updatedAt;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("type", type).append("updatedAt", updatedAt).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/ToStringBuilder.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/ToStringBuilder.java
new file mode 100644
index 00000000000..de427b38646
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/ToStringBuilder.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.groupepsa.internal.rest.api.dto;
+
+import java.io.IOException;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+class ToStringBuilder implements Appendable, CharSequence {
+ protected StringBuilder stringBuilder = new StringBuilder();
+
+ ToStringBuilder(Object obj) {
+ }
+
+ @Override
+ public int length() {
+ return stringBuilder.length();
+ }
+
+ @Override
+ public char charAt(int index) {
+ return stringBuilder.charAt(index);
+ }
+
+ @Override
+ public CharSequence subSequence(int start, int end) {
+ return stringBuilder.subSequence(start, end);
+ }
+
+ @Override
+ public ToStringBuilder append(@Nullable CharSequence csq) throws IOException {
+ if (stringBuilder.length() != 0) {
+ stringBuilder.append(", ");
+ }
+ stringBuilder.append(csq);
+ return this;
+ }
+
+ @Override
+ public ToStringBuilder append(@Nullable CharSequence csq, int start, int end) throws IOException {
+ if (stringBuilder.length() != 0) {
+ stringBuilder.append(", ");
+ }
+ stringBuilder.append(csq, start, end);
+ return this;
+ }
+
+ @Override
+ public ToStringBuilder append(char c) throws IOException {
+ if (stringBuilder.length() != 0) {
+ stringBuilder.append(", ");
+ }
+ stringBuilder.append(c);
+ return this;
+ }
+
+ public ToStringBuilder append(Object key, @Nullable Object value) {
+ if (stringBuilder.length() != 0) {
+ stringBuilder.append(", ");
+ }
+ stringBuilder.append(key.toString() + ": " + ((value == null) ? "(null)" : value.toString()));
+ return this;
+ }
+
+ @Override
+ public String toString() {
+ return "{ " + stringBuilder.toString() + " }";
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/User.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/User.java
new file mode 100644
index 00000000000..6c2d6ae2e40
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/User.java
@@ -0,0 +1,80 @@
+/**
+ * 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.groupepsa.internal.rest.api.dto;
+
+import java.time.ZonedDateTime;
+import java.util.List;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class User {
+ private @Nullable String email;
+ private @Nullable String firstName;
+ private @Nullable String lastName;
+ @SerializedName("_embedded")
+ private @Nullable Embedded embedded;
+ private @Nullable ZonedDateTime createdAt;
+ private @Nullable ZonedDateTime updatedAt;
+
+ private static class Embedded {
+ private @Nullable List vehicles;
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("vehicles", vehicles).toString();
+ }
+ }
+
+ public @Nullable String getEmail() {
+ return email;
+ }
+
+ public @Nullable String getLastName() {
+ return lastName;
+ }
+
+ public @Nullable String getFirstName() {
+ return firstName;
+ }
+
+ public @Nullable List getVehicles() {
+ final Embedded resEmbedded = embedded;
+ if (resEmbedded != null) {
+ return resEmbedded.vehicles;
+ } else {
+ return null;
+ }
+ }
+
+ public @Nullable ZonedDateTime getCreatedAt() {
+ return createdAt;
+ }
+
+ public @Nullable ZonedDateTime getUpdatedAt() {
+ return updatedAt;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("createdAt", createdAt).append("updatedAt", createdAt)
+ .append("email", email).append("firstName", firstName).append("lastName", lastName)
+ .append("vehicles", embedded != null ? embedded : null).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Vehicle.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Vehicle.java
new file mode 100644
index 00000000000..ab9a1687e97
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/Vehicle.java
@@ -0,0 +1,70 @@
+/**
+ * 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.groupepsa.internal.rest.api.dto;
+
+import java.time.ZonedDateTime;
+import java.util.List;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class Vehicle {
+ private @Nullable String id;
+ private @Nullable String vin;
+ private @Nullable String brand;
+ private @Nullable String label;
+ @SerializedName("engine")
+ private @Nullable List engines;
+ private @Nullable ZonedDateTime createdAt;
+ private @Nullable ZonedDateTime updatedAt;
+
+ public @Nullable String getId() {
+ return id;
+ }
+
+ public @Nullable String getVin() {
+ return vin;
+ }
+
+ public @Nullable String getBrand() {
+ return brand;
+ }
+
+ public @Nullable String getLabel() {
+ return label;
+ }
+
+ public @Nullable List getEngines() {
+ return engines;
+ }
+
+ public @Nullable ZonedDateTime getCreatedAt() {
+ return createdAt;
+ }
+
+ public @Nullable ZonedDateTime getUpdatedAt() {
+ return updatedAt;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("createdAt", createdAt).append("updatedAt", createdAt).append("id", id)
+ .append("vin", vin).append("brand", brand).append("label", label).append("engines", engines).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/VehicleStatus.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/VehicleStatus.java
new file mode 100644
index 00000000000..44d7d65cbbd
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/api/dto/VehicleStatus.java
@@ -0,0 +1,148 @@
+/**
+ * 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.groupepsa.internal.rest.api.dto;
+
+import java.time.ZonedDateTime;
+import java.util.List;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class VehicleStatus {
+
+ private @Nullable ZonedDateTime updatedAt;
+ @SerializedName("_embedded")
+ private @Nullable Embedded embedded;
+ private @Nullable Battery battery;
+ private @Nullable DoorsState doorsState;
+ private @Nullable List energy = null;
+ private @Nullable Environment environment;
+ private @Nullable Ignition ignition;
+ private @Nullable Kinetic kinetic;
+ @SerializedName("timed.odometer")
+ private @Nullable Odometer odometer;
+ private @Nullable Position lastPosition;
+ private @Nullable Preconditionning preconditionning;
+ private @Nullable Privacy privacy;
+ private @Nullable Safety safety;
+ private @Nullable Service service;
+
+ private static class Embedded {
+ private @Nullable Extension extension;
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("extension", extension).toString();
+ }
+ }
+
+ private static class Extension {
+ private @Nullable Kinetic kinetic;
+ private @Nullable Odometer odometer;
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("kinetic", kinetic).append("odometer", odometer).toString();
+ }
+ }
+
+ public @Nullable Kinetic getKinetic() {
+ if (kinetic != null) {
+ return kinetic;
+ } else {
+ final Embedded finalEmbedded = embedded;
+ if (finalEmbedded != null) {
+ final Extension finalExtension = finalEmbedded.extension;
+ if (finalExtension != null) {
+ return finalExtension.kinetic;
+ }
+ }
+ return null;
+ }
+ }
+
+ public @Nullable Odometer getOdometer() {
+ if (odometer != null) {
+ return odometer;
+ } else {
+ Embedded finalEmbedded = embedded;
+ if (finalEmbedded != null) {
+ final Extension finalExtension = finalEmbedded.extension;
+ if (finalExtension != null) {
+ return finalExtension.odometer;
+ }
+ }
+ return null;
+ }
+ }
+
+ public @Nullable ZonedDateTime getUpdatedAt() {
+ return updatedAt;
+ }
+
+ public @Nullable Battery getBattery() {
+ return battery;
+ }
+
+ public @Nullable DoorsState getDoorsState() {
+ return doorsState;
+ }
+
+ public @Nullable List getEnergy() {
+ return energy;
+ }
+
+ public @Nullable Environment getEnvironment() {
+ return environment;
+ }
+
+ public @Nullable Ignition getIgnition() {
+ return ignition;
+ }
+
+ public @Nullable Position getLastPosition() {
+ return lastPosition;
+ }
+
+ public @Nullable Preconditionning getPreconditionning() {
+ return preconditionning;
+ }
+
+ public @Nullable Privacy getPrivacy() {
+ return privacy;
+ }
+
+ public @Nullable Safety getSafety() {
+ return safety;
+ }
+
+ public @Nullable Service getService() {
+ return service;
+ }
+
+ @Override
+ public String toString() {
+ return new ToStringBuilder(this).append("updatedAt", updatedAt).append("_embedded", embedded)
+ .append("battery", battery).append("doorsState", doorsState).append("energy", energy)
+ .append("environment", environment).append("ignition", ignition).append("kinetic", kinetic)
+ .append("odometer", odometer).append("lastPosition", lastPosition)
+ .append("preconditionning", preconditionning).append("privacy", privacy).append("safety", safety)
+ .append("service", service).toString();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/exceptions/GroupePSACommunicationException.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/exceptions/GroupePSACommunicationException.java
new file mode 100644
index 00000000000..8ea05e487a5
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/exceptions/GroupePSACommunicationException.java
@@ -0,0 +1,71 @@
+/**
+ * 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.groupepsa.internal.rest.exceptions;
+
+import java.io.IOException;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * An exception that occurred while communicating with an groupepsa or an groupepsa bridge
+ *
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class GroupePSACommunicationException extends IOException {
+ private static final long serialVersionUID = 1L;
+ private int statusCode = -1;
+
+ public GroupePSACommunicationException(Exception e) {
+ super(e);
+ }
+
+ public GroupePSACommunicationException(int statusCode, Exception e) {
+ super(e);
+ this.statusCode = statusCode;
+ }
+
+ public GroupePSACommunicationException(int statusCode) {
+ this.statusCode = statusCode;
+ }
+
+ public GroupePSACommunicationException(int statusCode, String message) {
+ super(message);
+ this.statusCode = statusCode;
+ }
+
+ public GroupePSACommunicationException(String message, Exception e) {
+ super(message, e);
+ }
+
+ public GroupePSACommunicationException(String message) {
+ super(message);
+ }
+
+ public int getStatusCode() {
+ return statusCode;
+ }
+
+ @Override
+ public @Nullable String getMessage() {
+ String message = super.getMessage();
+ return message == null ? null : "Rest call failed: statusCode=" + statusCode + ", message=" + message;
+ }
+
+ @Override
+ public String toString() {
+ return getClass().getSimpleName() + ": statusCode=" + statusCode + ", message=" + super.getMessage()
+ + ", cause: " + getCause();
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/exceptions/UnauthorizedException.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/exceptions/UnauthorizedException.java
new file mode 100644
index 00000000000..f1ad02c735b
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/exceptions/UnauthorizedException.java
@@ -0,0 +1,35 @@
+/**
+ * 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.groupepsa.internal.rest.exceptions;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class UnauthorizedException extends GroupePSACommunicationException {
+ private static final long serialVersionUID = 1L;
+
+ public UnauthorizedException(int statusCode, Exception e) {
+ super(statusCode, e);
+ }
+
+ public UnauthorizedException(int statusCode) {
+ super(statusCode);
+ }
+
+ public UnauthorizedException(int statusCode, String message) {
+ super(statusCode, message);
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/exceptions/UnavailableException.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/exceptions/UnavailableException.java
new file mode 100644
index 00000000000..c7bbf47b312
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/rest/exceptions/UnavailableException.java
@@ -0,0 +1,27 @@
+/**
+ * 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.groupepsa.internal.rest.exceptions;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class UnavailableException extends GroupePSACommunicationException {
+ private static final long serialVersionUID = 1L;
+
+ public UnavailableException(int statusCode, Exception e) {
+ super(statusCode, e);
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/things/GroupePSAConfiguration.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/things/GroupePSAConfiguration.java
new file mode 100644
index 00000000000..73bfdf15502
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/things/GroupePSAConfiguration.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.groupepsa.internal.things;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * The {@link GroupePSAConfiguration} class contains fields mapping thing configuration parameters.
+ *
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class GroupePSAConfiguration {
+
+ private @Nullable String id;
+ private @Nullable Integer pollingInterval;
+ private @Nullable Integer onlineInterval;
+
+ @Nullable
+ public String getId() {
+ return id;
+ }
+
+ public void setId(String id) {
+ this.id = id;
+ }
+
+ public @Nullable Integer getPollingInterval() {
+ return pollingInterval;
+ }
+
+ public void setPollingInterval(Integer pollingInterval) {
+ this.pollingInterval = pollingInterval;
+ }
+
+ public @Nullable Integer getOnlineInterval() {
+ return onlineInterval;
+ }
+
+ public void setOnlineInterval(Integer onlineInterval) {
+ this.onlineInterval = onlineInterval;
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/things/GroupePSAHandler.java b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/things/GroupePSAHandler.java
new file mode 100644
index 00000000000..30a1b706c85
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/java/org/openhab/binding/groupepsa/internal/things/GroupePSAHandler.java
@@ -0,0 +1,479 @@
+/**
+ * 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.groupepsa.internal.things;
+
+import static org.openhab.binding.groupepsa.internal.GroupePSABindingConstants.*;
+
+import java.math.BigDecimal;
+import java.text.MessageFormat;
+import java.time.ZonedDateTime;
+import java.util.List;
+import java.util.concurrent.ScheduledFuture;
+import java.util.concurrent.TimeUnit;
+import java.util.function.Function;
+
+import javax.measure.Quantity;
+import javax.measure.Unit;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.groupepsa.internal.bridge.GroupePSABridgeHandler;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Air;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.AirConditioning;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Battery;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.BatteryStatus;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Charging;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.DoorsState;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Energy;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Environment;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Health;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Ignition;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Kinetic;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Luminosity;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Odometer;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Opening;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Position;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Preconditionning;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Privacy;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Properties;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Safety;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.Service;
+import org.openhab.binding.groupepsa.internal.rest.api.dto.VehicleStatus;
+import org.openhab.binding.groupepsa.internal.rest.exceptions.GroupePSACommunicationException;
+import org.openhab.core.library.types.DateTimeType;
+import org.openhab.core.library.types.DecimalType;
+import org.openhab.core.library.types.OpenClosedType;
+import org.openhab.core.library.types.PointType;
+import org.openhab.core.library.types.QuantityType;
+import org.openhab.core.library.types.StringType;
+import org.openhab.core.library.unit.MetricPrefix;
+import org.openhab.core.library.unit.SIUnits;
+import org.openhab.core.library.unit.Units;
+import org.openhab.core.thing.Bridge;
+import org.openhab.core.thing.Channel;
+import org.openhab.core.thing.ChannelUID;
+import org.openhab.core.thing.Thing;
+import org.openhab.core.thing.ThingStatus;
+import org.openhab.core.thing.ThingStatusDetail;
+import org.openhab.core.thing.binding.BaseThingHandler;
+import org.openhab.core.thing.binding.ThingHandler;
+import org.openhab.core.thing.binding.ThingHandlerCallback;
+import org.openhab.core.thing.binding.builder.ThingBuilder;
+import org.openhab.core.thing.type.ChannelTypeUID;
+import org.openhab.core.types.Command;
+import org.openhab.core.types.RefreshType;
+import org.openhab.core.types.UnDefType;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import com.github.filosganga.geogson.model.Geometry;
+import com.github.filosganga.geogson.model.positions.SinglePosition;
+
+/**
+ * The {@link GroupePSAHandler} is responsible for handling commands, which are
+ * sent to one of the channels.
+ *
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class GroupePSAHandler extends BaseThingHandler {
+ private static final long DEFAULT_POLLING_INTERVAL_M = TimeUnit.MINUTES.toMinutes(1);
+ private static final long DEFAULT_ONLINE_INTERVAL_M = TimeUnit.MINUTES.toMinutes(60);
+
+ private final Logger logger = LoggerFactory.getLogger(GroupePSAHandler.class);
+
+ private @Nullable String id = null;
+ private long lastQueryTimeNs = 0L;
+
+ private @Nullable ScheduledFuture> groupepsaPollingJob;
+ private long maxQueryFrequencyNanos = TimeUnit.MINUTES.toNanos(1);
+ private long onlineIntervalM;
+
+ public GroupePSAHandler(Thing thing) {
+ super(thing);
+ }
+
+ @Override
+ protected @Nullable Bridge getBridge() {
+ return super.getBridge();
+ }
+
+ private void pollStatus() {
+ Bridge bridge = getBridge();
+ if (bridge != null && bridge.getStatus() == ThingStatus.ONLINE) {
+ updateGroupePSAState();
+ } else {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.BRIDGE_OFFLINE);
+ }
+ };
+
+ @Override
+ public void handleCommand(ChannelUID channelUID, Command command) {
+ if (command instanceof RefreshType) {
+ refreshChannels(channelUID);
+ }
+ }
+
+ private void refreshChannels(ChannelUID channelUID) {
+ updateGroupePSAState();
+ }
+
+ @Override
+ public void initialize() {
+ if (getBridgeHandler() != null) {
+ GroupePSAConfiguration currentConfig = getConfigAs(GroupePSAConfiguration.class);
+ final String id = currentConfig.getId();
+ final Integer pollingIntervalM = currentConfig.getPollingInterval();
+ final Integer onlineIntervalM = currentConfig.getOnlineInterval();
+
+ if (id == null) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
+ "@text/conf-error-no-vehicle-id");
+ } else if (pollingIntervalM != null && pollingIntervalM < 1) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
+ "@text/conf-error-invalid-polling-interval");
+ } else if (onlineIntervalM != null && onlineIntervalM < 1) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
+ "@text/conf-error-invalid-online-interval");
+ } else {
+ this.id = id;
+ this.onlineIntervalM = onlineIntervalM != null ? onlineIntervalM : DEFAULT_ONLINE_INTERVAL_M;
+ startGroupePSAPolling(pollingIntervalM);
+ }
+
+ } else {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.BRIDGE_UNINITIALIZED);
+ }
+ }
+
+ @Nullable
+ public GroupePSABridgeHandler getBridgeHandler() {
+ Bridge bridge = getBridge();
+ if (bridge != null) {
+ ThingHandler handler = bridge.getHandler();
+ if (handler instanceof GroupePSABridgeHandler) {
+ return (GroupePSABridgeHandler) handler;
+ }
+ }
+ return null;
+ }
+
+ @Override
+ public void dispose() {
+ stopGroupePSAPolling();
+ id = null;
+ }
+
+ private void startGroupePSAPolling(@Nullable Integer pollingIntervalM) {
+ if (groupepsaPollingJob == null) {
+ final long pollingIntervalToUse = pollingIntervalM == null ? DEFAULT_POLLING_INTERVAL_M : pollingIntervalM;
+ groupepsaPollingJob = scheduler.scheduleWithFixedDelay(() -> pollStatus(), 1, pollingIntervalToUse * 60,
+ TimeUnit.SECONDS);
+ }
+ }
+
+ private void stopGroupePSAPolling() {
+ ScheduledFuture> job = groupepsaPollingJob;
+ if (job != null) {
+ job.cancel(true);
+ groupepsaPollingJob = null;
+ }
+ }
+
+ private boolean isValidResult(VehicleStatus vehicle) {
+ return vehicle.getUpdatedAt() != null;
+ }
+
+ private boolean isConnected(VehicleStatus vehicle) {
+ ZonedDateTime updatedAt = vehicle.getUpdatedAt();
+ if (updatedAt == null) {
+ return false;
+ }
+
+ return updatedAt.isAfter(ZonedDateTime.now().minusMinutes(onlineIntervalM));
+ }
+
+ private synchronized void updateGroupePSAState() {
+ if (System.nanoTime() - lastQueryTimeNs <= maxQueryFrequencyNanos) {
+ return;
+ }
+
+ lastQueryTimeNs = System.nanoTime();
+
+ String id = this.id;
+ if (id == null) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR, "@text/conf-error-no-vehicle-id");
+ return;
+ }
+
+ GroupePSABridgeHandler groupepsaBridge = getBridgeHandler();
+ if (groupepsaBridge == null) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR, "@text/conf-error-no-bridge");
+ return;
+ }
+
+ try {
+ VehicleStatus vehicle = groupepsaBridge.getVehicleStatus(id);
+
+ if (vehicle != null && isValidResult(vehicle)) {
+ logger.trace("Vehicle: {}", vehicle.toString());
+
+ logger.debug("Update vehicle state now: {}, lastupdate: {}", ZonedDateTime.now(),
+ vehicle.getUpdatedAt());
+
+ updateChannelState(vehicle);
+
+ if (isConnected(vehicle)) {
+ updateStatus(ThingStatus.ONLINE);
+ } else {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.NONE,
+ "@text/comm-error-vehicle-not-connected-to-cloud");
+ }
+ } else {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR,
+ "@text/comm-error-query-vehicle-failed");
+ }
+ } catch (GroupePSACommunicationException e) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR,
+ MessageFormat.format("@text/comm-error-query-vehicle-failed", e.getMessage()));
+ }
+ }
+
+ private void updateChannelState(VehicleStatus vehicle) {
+ final DoorsState doorsState = vehicle.getDoorsState();
+ if (doorsState != null) {
+ buildDoorChannels(doorsState);
+
+ List openings = doorsState.getOpening();
+ if (openings != null) {
+ for (Opening opening : openings) {
+ String id = opening.getIdentifier();
+ if (id != null) {
+ ChannelUID channelUID = new ChannelUID(getThing().getUID(), CHANNEL_GROUP_DOORS,
+ id.toLowerCase());
+ updateState(channelUID, "open".equalsIgnoreCase(opening.getState()) ? OpenClosedType.OPEN
+ : OpenClosedType.CLOSED);
+ }
+ }
+ }
+
+ List lockedState = doorsState.getLockedState();
+ updateState(CHANNEL_DOORS_LOCK, lockedState, x -> x.get(0));
+ } else {
+ updateState(CHANNEL_DOORS_LOCK, UnDefType.UNDEF);
+ }
+
+ updateState(CHANNEL_BATTERY_CURRENT, vehicle.getBattery(), Battery::getCurrent, Units.AMPERE);
+ updateState(CHANNEL_BATTERY_VOLTAGE, vehicle.getBattery(), Battery::getVoltage, Units.VOLT);
+
+ updateState(CHANNEL_ENVIRONMENT_TEMPERATURE, vehicle.getEnvironment(), Environment::getAir, Air::getTemp,
+ SIUnits.CELSIUS);
+ updateStateBoolean(CHANNEL_ENVIRONMENT_DAYTIME, vehicle.getEnvironment(), Environment::getLuminosity,
+ Luminosity::isDay);
+
+ updateState(CHANNEL_MOTION_IGNITION, vehicle.getIgnition(), Ignition::getType);
+
+ updateStateBoolean(CHANNEL_MOTION_MOVING, vehicle.getKinetic(), Kinetic::isMoving);
+ updateState(CHANNEL_MOTION_ACCELERATION, vehicle.getKinetic(), Kinetic::getAcceleration,
+ Units.METRE_PER_SQUARE_SECOND);
+ updateState(CHANNEL_MOTION_SPEED, vehicle.getKinetic(), Kinetic::getSpeed, SIUnits.KILOMETRE_PER_HOUR);
+
+ updateState(CHANNEL_MOTION_MILEAGE, vehicle.getOdometer(), Odometer::getMileage,
+ MetricPrefix.KILO(SIUnits.METRE));
+
+ Position lastPosition = vehicle.getLastPosition();
+ if (lastPosition != null) {
+ Geometry geometry = lastPosition.getGeometry();
+ if (geometry != null) {
+ SinglePosition position = (SinglePosition) geometry.positions();
+ if (Double.isFinite(position.alt())) {
+ updateState(CHANNEL_POSITION_POSITION, new PointType(new DecimalType(position.lat()),
+ new DecimalType(position.lon()), new DecimalType(position.alt())));
+ } else {
+ updateState(CHANNEL_POSITION_POSITION,
+ new PointType(new DecimalType(position.lat()), new DecimalType(position.lon())));
+ }
+ } else {
+ updateState(CHANNEL_POSITION_POSITION, UnDefType.UNDEF);
+ }
+ updateState(CHANNEL_POSITION_HEADING, lastPosition.getProperties(), Properties::getHeading,
+ Units.DEGREE_ANGLE);
+ updateState(CHANNEL_POSITION_TYPE, lastPosition.getProperties(), Properties::getType);
+ updateState(CHANNEL_POSITION_SIGNALSTRENGTH, lastPosition.getProperties(), Properties::getSignalQuality,
+ Units.PERCENT);
+ }
+
+ updateState(CHANNEL_VARIOUS_LAST_UPDATED, vehicle.getUpdatedAt());
+ updateState(CHANNEL_VARIOUS_PRIVACY, vehicle.getPrivacy(), Privacy::getState);
+ updateState(CHANNEL_VARIOUS_BELT, vehicle.getSafety(), Safety::getBeltWarning);
+ updateState(CHANNEL_VARIOUS_EMERGENCY, vehicle.getSafety(), Safety::getECallTriggeringRequest);
+ updateState(CHANNEL_VARIOUS_SERVICE, vehicle.getService(), Service::getType);
+ updateState(CHANNEL_VARIOUS_PRECONDITINING, vehicle.getPreconditionning(), Preconditionning::getAirConditioning,
+ AirConditioning::getStatus);
+ updateState(CHANNEL_VARIOUS_PRECONDITINING_FAILURE, vehicle.getPreconditionning(),
+ Preconditionning::getAirConditioning, AirConditioning::getFailureCause);
+
+ List energies = vehicle.getEnergy();
+ if (energies != null) {
+ for (Energy energy : energies) {
+ if ("Fuel".equalsIgnoreCase(energy.getType())) {
+ updateState(CHANNEL_FUEL_AUTONOMY, energy, Energy::getAutonomy, MetricPrefix.KILO(SIUnits.METRE));
+ updateState(CHANNEL_FUEL_CONSUMPTION, energy, Energy::getConsumption,
+ Units.LITRE.divide(MetricPrefix.KILO(SIUnits.METRE)));
+ updateState(CHANNEL_FUEL_LEVEL, energy, Energy::getLevel, Units.PERCENT);
+ } else if ("Electric".equalsIgnoreCase(energy.getType())) {
+ updateState(CHANNEL_ELECTRIC_AUTONOMY, energy, Energy::getAutonomy,
+ MetricPrefix.KILO(SIUnits.METRE));
+ updateState(CHANNEL_ELECTRIC_RESIDUAL, energy, Energy::getResidual, Units.KILOWATT_HOUR);
+ updateState(CHANNEL_ELECTRIC_LEVEL, energy, Energy::getLevel, Units.PERCENT);
+
+ updateState(CHANNEL_ELECTRIC_BATTERY_CAPACITY, energy, Energy::getBattery,
+ BatteryStatus::getCapacity, Units.KILOWATT_HOUR);
+ updateState(CHANNEL_ELECTRIC_BATTERY_HEALTH_CAPACITY, energy, Energy::getBattery,
+ BatteryStatus::getHealth, Health::getCapacity, Units.PERCENT);
+ updateState(CHANNEL_ELECTRIC_BATTERY_HEALTH_RESISTANCE, energy, Energy::getBattery,
+ BatteryStatus::getHealth, Health::getResistance, Units.PERCENT);
+
+ updateState(CHANNEL_ELECTRIC_CHARGING_STATUS, energy, Energy::getCharging, Charging::getStatus);
+ updateState(CHANNEL_ELECTRIC_CHARGING_MODE, energy, Energy::getCharging, Charging::getChargingMode);
+ updateStateBoolean(CHANNEL_ELECTRIC_CHARGING_PLUGGED, energy, Energy::getCharging,
+ Charging::isPlugged);
+ updateState(CHANNEL_ELECTRIC_CHARGING_RATE, energy, Energy::getCharging, Charging::getChargingRate,
+ SIUnits.KILOMETRE_PER_HOUR);
+
+ updateState(CHANNEL_ELECTRIC_CHARGING_REMAININGTIME, energy, Energy::getCharging,
+ Charging::getRemainingTime, x -> new BigDecimal(x.getSeconds()), Units.SECOND);
+ updateState(CHANNEL_ELECTRIC_CHARGING_NEXTDELAYEDTIME, energy, Energy::getCharging,
+ Charging::getNextDelayedTime, x -> new BigDecimal(x.getSeconds()), Units.SECOND);
+
+ }
+ }
+ }
+ }
+
+ void buildDoorChannels(final DoorsState doorsState) {
+ ThingHandlerCallback callback = getCallback();
+ if (callback == null) {
+ return;
+ }
+
+ ThingBuilder thingBuilder = editThing();
+ List channels = getThing().getChannelsOfGroup(CHANNEL_GROUP_DOORS);
+ thingBuilder.withoutChannels(channels);
+
+ ChannelUID channelUID = new ChannelUID(getThing().getUID(), CHANNEL_DOORS_LOCK);
+ ChannelTypeUID channelTypeUID = new ChannelTypeUID(BINDING_ID, CHANNEL_TYPE_DOORLOCK);
+ thingBuilder.withChannel(callback.createChannelBuilder(channelUID, channelTypeUID).build());
+
+ List openings = doorsState.getOpening();
+ if (openings != null) {
+ for (Opening opening : openings) {
+ String id = opening.getIdentifier();
+ if (id != null) {
+ channelUID = new ChannelUID(getThing().getUID(), CHANNEL_GROUP_DOORS, id.toLowerCase());
+ channelTypeUID = new ChannelTypeUID(BINDING_ID, CHANNEL_TYPE_DOOROPEN);
+ thingBuilder.withChannel(callback.createChannelBuilder(channelUID, channelTypeUID).build());
+ }
+ }
+ }
+
+ updateThing(thingBuilder.build());
+ }
+
+ // Various update helper functions
+
+ protected > void updateState(String channelID, @Nullable BigDecimal number, Unit unit) {
+ if (number != null) {
+ updateState(channelID, new QuantityType(number, unit));
+ } else {
+ updateState(channelID, UnDefType.UNDEF);
+ }
+ }
+
+ protected > void updateState(String channelID, final @Nullable T1 object,
+ Function super T1, @Nullable BigDecimal> mapper, Unit unit) {
+ updateState(channelID, object != null ? mapper.apply(object) : null, unit);
+ }
+
+ protected > void updateState(String channelID, final @Nullable T1 object1,
+ Function super T1, @Nullable T2> mapper1, Function super T2, @Nullable BigDecimal> mapper2,
+ Unit unit) {
+ final @Nullable T2 object2 = object1 != null ? mapper1.apply(object1) : null;
+ updateState(channelID, object2 != null ? mapper2.apply(object2) : null, unit);
+ }
+
+ protected > void updateState(String channelID, final @Nullable T1 object1,
+ Function super T1, @Nullable T2> mapper1, Function super T2, @Nullable T3> mapper2,
+ Function super T3, @Nullable BigDecimal> mapper3, Unit unit) {
+ final @Nullable T2 object2 = object1 != null ? mapper1.apply(object1) : null;
+ final @Nullable T3 object3 = object2 != null ? mapper2.apply(object2) : null;
+ updateState(channelID, object3 != null ? mapper3.apply(object3) : null, unit);
+ }
+
+ protected void updateState(String channelID, @Nullable ZonedDateTime date) {
+ if (date != null) {
+ updateState(channelID, new DateTimeType(date));
+ } else {
+ updateState(channelID, UnDefType.UNDEF);
+ }
+ }
+
+ protected void updateStateDate(String channelID, @Nullable T1 object,
+ Function super T1, @Nullable ZonedDateTime> mapper) {
+ updateState(channelID, object != null ? mapper.apply(object) : null);
+ }
+
+ protected void updateStateDate(String channelID, @Nullable T1 object1,
+ Function super T1, @Nullable T2> mapper1, Function super T2, @Nullable ZonedDateTime> mapper2) {
+ final @Nullable T2 object2 = object1 != null ? mapper1.apply(object1) : null;
+ updateState(channelID, object2 != null ? mapper2.apply(object2) : null);
+ }
+
+ protected void updateState(String channelID, @Nullable String text) {
+ if (text != null) {
+ updateState(channelID, new StringType(text));
+ } else {
+ updateState(channelID, UnDefType.UNDEF);
+ }
+ }
+
+ protected void updateState(String channelID, @Nullable T1 object,
+ Function super T1, @Nullable String> mapper) {
+ updateState(channelID, object != null ? mapper.apply(object) : null);
+ }
+
+ protected void updateState(String channelID, @Nullable T1 object1,
+ Function super T1, @Nullable T2> mapper1, Function super T2, @Nullable String> mapper2) {
+ final @Nullable T2 object2 = object1 != null ? mapper1.apply(object1) : null;
+ updateState(channelID, object2 != null ? mapper2.apply(object2) : null);
+ }
+
+ protected void updateState(String channelID, @Nullable Boolean value) {
+ if (value != null) {
+ updateState(channelID, value ? OpenClosedType.OPEN : OpenClosedType.CLOSED);
+ } else {
+ updateState(channelID, UnDefType.UNDEF);
+ }
+ }
+
+ protected void updateStateBoolean(String channelID, @Nullable T1 object,
+ Function super T1, @Nullable Boolean> mapper) {
+ updateState(channelID, object != null ? mapper.apply(object) : null);
+ }
+
+ protected void updateStateBoolean(String channelID, final @Nullable T1 object1,
+ Function super T1, @Nullable T2> mapper1, Function super T2, @Nullable Boolean> mapper2) {
+ final @Nullable T2 object2 = object1 != null ? mapper1.apply(object1) : null;
+ updateState(channelID, object2 != null ? mapper2.apply(object2) : null);
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/resources/OH-INF/binding/binding.xml b/bundles/org.openhab.binding.groupepsa/src/main/resources/OH-INF/binding/binding.xml
new file mode 100644
index 00000000000..034f514b2f2
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/resources/OH-INF/binding/binding.xml
@@ -0,0 +1,9 @@
+
+
+
+ Groupe PSA Binding
+ Provides information and (limited) control of cars from the Groupe PSA (Peugeot, Citroen, Opel, etc.).
+
+
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/resources/OH-INF/i18n/groupepsa.properties b/bundles/org.openhab.binding.groupepsa/src/main/resources/OH-INF/i18n/groupepsa.properties
new file mode 100644
index 00000000000..e79057c77be
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/resources/OH-INF/i18n/groupepsa.properties
@@ -0,0 +1,21 @@
+# binding
+binding.groupepsa.name = Groupe PSA Binding
+binding.groupepsa.description = Provides information and (limited) control of cars from the Groupe PSA (Peugeot, Citroen, Opel, etc.).
+
+#configuration errors
+conf-error-no-vendor = Cannot connect to Groupe PSA bridge as no vendor is available in the configuration
+conf-error-unknown-vendor = Cannot connect to Groupe PSA bridge as the vendor is not valid
+conf-error-no-username = Cannot connect to Groupe PSA bridge as no username is available in the configuration
+conf-error-no-password = Cannot connect to Groupe PSA bridge as no password is available in the configuration
+conf-error-no-clientid = Cannot connect to Groupe PSA bridge as no client ID is available in the configuration
+conf-error-no-clientsecret = Cannot connect to Groupe PSA bridge as no client secret is available in the configuration
+conf-error-invalid-polling-interval = Invalid polling interval specified. The interval has to be >= 1
+conf-error-invalid-online-interval = Invalid online check interval specified. The interval has to be >= 1
+
+conf-error-no-vehicle-id = No VIN specified. Unable to communicate with the vehicle without an ID
+conf-error-no-bridge = No valid bridge for the vehicle is available
+
+#communication errors
+comm-error-vehicle-not-connected-to-cloud = Vehicle not connected to the cloud
+comm-error-query-vehicle-failed = Unable to query the vehicle status ({0})
+comm-error-query-vehicles-failed = Unable to get a list of vehicles ({0})
diff --git a/bundles/org.openhab.binding.groupepsa/src/main/resources/OH-INF/thing/thing-types.xml b/bundles/org.openhab.binding.groupepsa/src/main/resources/OH-INF/thing/thing-types.xml
new file mode 100644
index 00000000000..712f95ae920
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/main/resources/OH-INF/thing/thing-types.xml
@@ -0,0 +1,535 @@
+
+
+
+
+
+
+ The bridge to communicate with the PSA (Peugeot, Citroen, Vauxhall, Opel, DS) Web API for End-Users
+ WebService
+
+ vendor
+
+
+
+
+ The brand of the car
+
+
+
+
+
+
+
+
+
+
+ The user name for the mypeugot/mycitroen/myds/myopel/myvauxhall website or app
+
+
+ password
+ The password for the given user
+
+
+
+ 60
+ How often the available vehicles should be queried in minutes
+
+
+ true
+ 07364655-93cb-4194-8158-6b035ac2c24c
+
+ The client ID for API access: can normally left at the default value. (see:
+ https://developer.groupe-psa.io/webapi/b2c/quickstart/connect/#article)
+
+
+ true
+ F2kK7lC5kF5qN7tM0wT8kE3cW1dP0wC5pI6vC0sQ5iP5cN8cJ8
+
+ The client secret for API access: can normally left at the default value. (see:
+ https://developer.groupe-psa.io/webapi/b2c/quickstart/connect/#article)
+
+
+
+
+
+
+
+
+
+
+
+ Car connected via Groupe PSA (Peugeot, Citroen, Vauxhall, Opel, DS) Bridge
+
+
+
+
+
+
+
+
+
+
+
+
+
+ N/A
+ N/A
+ N/A
+
+
+ id
+
+
+
+
+ Vehicle API ID (obtained automatically during discovery, not equal to the VIN)
+
+
+
+ 5
+ How often this vehicle should be polled for updated information
+
+
+
+ 15
+ Timeout when the vehicle is considered offline
+
+
+
+
+
+
+
+ Main battery
+ Energy
+
+
+
+
+ Battery level
+
+
+
+
+
+
+ State of the doors
+ Door
+
+
+
+
+ Driver door open state
+
+
+
+
+
+
+ Environmental conditions
+
+
+
+ Outside Temperature
+
+
+
+
+
+
+
+ Motion information
+
+
+
+
+
+
+
+
+
+
+
+ Location information
+
+
+
+
+
+
+
+
+
+
+ Miscellaneous information
+
+
+
+
+
+
+
+
+
+
+
+
+ Number:ElectricCurrent
+
+ Electrical current
+ Energy
+
+
+
+
+ Number:ElectricPotential
+
+ Voltage
+ Energy
+
+
+
+
+ Number:Temperature
+
+ Temperature
+ Temperature
+
+
+
+
+ Contact
+
+ Enabled if it is daytime
+
+
+
+ String
+
+ Door lock state
+ Door
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Contact
+
+ Door is open
+ Door
+
+
+
+ String
+
+ Ignition state
+
+
+
+
+
+
+
+
+
+
+
+ Contact
+
+ Vehicle is moving
+
+
+
+ Number:Acceleration
+
+ Current acceleration
+
+
+
+
+ Number:Speed
+
+ Current speed
+
+
+
+
+ Number:Length
+
+ Total travelled distance
+
+
+
+
+ Location
+
+ Last known position
+
+
+
+
+ Number:Angle
+
+ Direction of travel
+
+
+
+
+ String
+
+ Position acquisition type
+
+
+
+
+
+
+
+
+
+ DateTime
+
+ Last time the results were updated on the server
+ Time
+
+
+
+ String
+
+ Privacy status
+
+
+
+
+
+
+
+
+
+
+ String
+
+ Seat belt status
+
+
+
+
+
+
+
+
+
+ String
+
+ Emergency call status
+
+
+
+
+
+
+
+
+
+
+ String
+
+ Service Type
+
+
+
+
+
+
+
+
+
+
+ String
+
+ Air conditioning status
+
+
+
+
+
+
+
+
+
+
+
+ String
+
+ Air conditioning failure cause
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Thermic Motor Fuel Status
+ Energy
+
+
+
+
+
+
+
+
+
+ Electric motor energy status
+
+
+
+ Remaining distance
+
+
+
+ Electricity level
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Number:Dimensionless
+
+ Fuel level
+
+
+
+
+ Number:Length
+
+ Remaining distance
+
+
+
+
+ Number:VolumetricFlowRate
+
+ Fuel consumption
+
+
+
+
+ Number:Energy
+
+ Remaining battery charge
+ BatteryLevel
+
+
+
+
+ Number:Energy
+
+ Battery capacity
+ BatteryLevel
+
+
+
+
+ Number:Dimensionless
+
+ Health of the battery capacity
+
+
+
+
+ Number:Dimensionless
+
+ Health of the battery resistance
+
+
+
+
+ String
+
+ Battery charging status
+
+
+
+
+
+
+
+
+
+
+
+
+ String
+
+ Battery charging mode
+
+
+
+
+
+
+
+
+
+
+ Contact
+
+ Vehicle plugged in to charger
+
+
+
+ Number:Speed
+
+ Battery Charging Rate
+ BatteryLevel
+
+
+
+
+ Number:Time
+
+ Time remaining till charged
+ Time
+
+
+
+
+ Number:Time
+
+ Time till the next charging starts
+ Time
+
+
+
+
diff --git a/bundles/org.openhab.binding.groupepsa/src/test/java/org/openhab/binding/groupepsa/internal/things/GroupePSAHandlerTest.java b/bundles/org.openhab.binding.groupepsa/src/test/java/org/openhab/binding/groupepsa/internal/things/GroupePSAHandlerTest.java
new file mode 100644
index 00000000000..bb672f80a22
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/test/java/org/openhab/binding/groupepsa/internal/things/GroupePSAHandlerTest.java
@@ -0,0 +1,159 @@
+/**
+ * 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.groupepsa.internal.things;
+
+import static org.mockito.ArgumentMatchers.*;
+import static org.mockito.Mockito.*;
+
+import java.io.BufferedReader;
+import java.io.InputStream;
+import java.io.InputStreamReader;
+import java.util.stream.Collectors;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.HttpClient;
+import org.eclipse.jetty.client.HttpContentResponse;
+import org.eclipse.jetty.client.HttpResponse;
+import org.junit.jupiter.api.AfterEach;
+import org.junit.jupiter.api.BeforeEach;
+import org.junit.jupiter.api.Test;
+import org.mockito.Mock;
+import org.mockito.MockitoAnnotations;
+import org.openhab.binding.groupepsa.internal.bridge.GroupePSABridgeHandler;
+import org.openhab.binding.groupepsa.internal.rest.api.GroupePSAConnectApi;
+import org.openhab.binding.groupepsa.internal.rest.exceptions.GroupePSACommunicationException;
+import org.openhab.core.auth.client.oauth2.OAuthFactory;
+import org.openhab.core.config.core.Configuration;
+import org.openhab.core.library.types.DateTimeType;
+import org.openhab.core.library.types.StringType;
+import org.openhab.core.thing.Bridge;
+import org.openhab.core.thing.ChannelUID;
+import org.openhab.core.thing.Thing;
+import org.openhab.core.thing.ThingStatus;
+import org.openhab.core.thing.ThingStatusDetail;
+import org.openhab.core.thing.ThingUID;
+import org.openhab.core.thing.binding.ThingHandlerCallback;
+import org.openhab.core.types.State;
+
+/**
+ * The {@link GroupePSAHandlerTest} is responsible for testing the binding
+ *
+ * @author Arjan Mels - Initial contribution
+ */
+@NonNullByDefault
+public class GroupePSAHandlerTest {
+ private @NonNullByDefault({}) AutoCloseable closeable;
+
+ private @NonNullByDefault({}) GroupePSAConnectApi api;
+ private @NonNullByDefault({}) GroupePSABridgeHandler bridgeHandler;
+ private @NonNullByDefault({}) GroupePSAHandler thingHandler;
+
+ private @NonNullByDefault({}) @Mock ThingHandlerCallback thingCallback;
+ private @NonNullByDefault({}) @Mock ThingHandlerCallback bridgeCallback;
+ private @NonNullByDefault({}) @Mock Thing thing;
+ private @NonNullByDefault({}) @Mock Bridge bridge;
+
+ private @NonNullByDefault({}) @Mock OAuthFactory oAuthFactory;
+ private @NonNullByDefault({}) @Mock HttpClient httpClient;
+
+ static String getResourceFileAsString(String fileName) throws GroupePSACommunicationException {
+ try (InputStream is = GroupePSAConnectApi.class.getResourceAsStream(fileName)) {
+ try (InputStreamReader isr = new InputStreamReader(is); BufferedReader reader = new BufferedReader(isr)) {
+ return reader.lines().collect(Collectors.joining(System.lineSeparator()));
+ }
+ } catch (Exception e) {
+ throw new GroupePSACommunicationException(e);
+ }
+ }
+
+ static HttpContentResponse createHttpResponse(String file) throws GroupePSACommunicationException {
+ return new HttpContentResponse(new HttpResponse(null, null).status(200),
+ getResourceFileAsString("/" + file).getBytes(), "json", "UTF-8");
+ }
+
+ @BeforeEach
+ @SuppressWarnings("null")
+ public void setUp() throws GroupePSACommunicationException {
+ closeable = MockitoAnnotations.openMocks(this);
+
+ // Create real objects
+ bridgeHandler = spy(new GroupePSABridgeHandler(bridge, oAuthFactory, httpClient));
+ thingHandler = spy(new GroupePSAHandler(thing));
+ api = spy(new GroupePSAConnectApi(httpClient, bridgeHandler, "clientId", "realm"));
+
+ // Setup API mock
+ doReturn(createHttpResponse("dummy_user.json")).when(api).executeRequest(contains("user"), anyString());
+ doReturn(createHttpResponse("dummy_vehiclestatus3.json")).when(api).executeRequest(contains("status"),
+ anyString());
+
+ // Setup bridge handler mock
+ bridgeHandler.setCallback(bridgeCallback);
+ doReturn(api).when(bridgeHandler).getAPI();
+
+ // Setup bridge mock
+ Configuration bridgeConfig = new Configuration();
+ bridgeConfig.put("vendor", "OPEL");
+ bridgeConfig.put("userName", "user");
+ bridgeConfig.put("password", "pwd");
+ bridgeConfig.put("clientId", "clientIdValue");
+ bridgeConfig.put("clientSecret", "clientSecretValue");
+ doReturn(bridgeConfig).when(bridge).getConfiguration();
+ doReturn(ThingStatus.ONLINE).when(bridge).getStatus();
+ doReturn(bridgeHandler).when(bridge).getHandler();
+ doReturn(new ThingUID("a:b:c")).when(bridge).getUID();
+
+ // Setup thing mock
+ Configuration thingConfig = new Configuration();
+ thingConfig.put("id", "mock_id");
+ doReturn(thingConfig).when(thing).getConfiguration();
+ doReturn(new ThingUID("a:b:c")).when(thing).getUID();
+
+ // Setup thing handler mock
+ thingHandler.setCallback(thingCallback);
+ doReturn(bridge).when(thingHandler).getBridge();
+ doNothing().when(thingHandler).buildDoorChannels(any());
+ }
+
+ @AfterEach
+ public void tearDown() throws Exception {
+ // Free any resources, like open database connections, files etc.
+ thingHandler.dispose();
+ bridgeHandler.dispose();
+ closeable.close();
+ }
+
+ @Test
+ public void intializeAndCheckChannels() throws InterruptedException {
+ // Initialize the bridge
+ bridgeHandler.initialize();
+
+ // check that the bridge is online
+ verify(bridgeCallback, timeout(2000)).statusUpdated(eq(bridge),
+ argThat(arg -> arg.getStatus().equals(ThingStatus.ONLINE)));
+
+ // Initialize the thing
+ thingHandler.initialize();
+
+ // check that the thing is offline without detail (last update time is not
+ // within 15 minutes)
+ verify(thingCallback, timeout(2000)).statusUpdated(eq(thing),
+ argThat(arg -> arg.getStatus().equals(ThingStatus.OFFLINE)
+ && arg.getStatusDetail().equals(ThingStatusDetail.NONE)));
+
+ // check that the channels are updated
+ verify(thingCallback, atLeast(30)).stateUpdated(any(ChannelUID.class), any(State.class));
+ verify(thingCallback).stateUpdated(eq(new ChannelUID("a:b:c:electric#chargingStatus")),
+ eq(new StringType("Disconnected")));
+ verify(thingCallback).stateUpdated(eq(new ChannelUID("a:b:c:various#lastUpdated")), any(DateTimeType.class));
+ }
+}
diff --git a/bundles/org.openhab.binding.groupepsa/src/test/resources/dummy_user.json b/bundles/org.openhab.binding.groupepsa/src/test/resources/dummy_user.json
new file mode 100644
index 00000000000..c2d683b26a5
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/test/resources/dummy_user.json
@@ -0,0 +1,71 @@
+{
+ "_links": {
+ "self": {
+ "href": "ivuvawoj"
+ },
+ "trips": {
+ "href": "ugihub"
+ },
+ "lastTrip": {
+ "href": "culugakin"
+ },
+ "lastEcodriving": {
+ "href": "bagnu"
+ }
+ },
+ "id": "7000650216898560",
+ "email": "iwni@georure.kr",
+ "firstName": "Floyd",
+ "lastName": "Gilbert",
+ "createdAt": "2019-08-24T14:15:22Z",
+ "updatedAt": "2019-08-24T14:15:22Z",
+ "_embedded": {
+ "vehicles": [
+ {
+ "_links": {
+ "self": {
+ "href": "ovoha"
+ },
+ "lastTrip": {
+ "href": "lotudwe"
+ },
+ "lastPosition": {
+ "href": "bobediirewelojbu"
+ },
+ "trips": {
+ "href": "vacofgutv"
+ },
+ "collisions": {
+ "href": "koafeu"
+ },
+ "telemetry": {
+ "href": "ewahzonug"
+ },
+ "maintenance": {
+ "href": "dovuzibwudmicel"
+ },
+ "alerts": {
+ "href": "izbisbiwpejunun"
+ },
+ "events": {
+ "href": "tece"
+ }
+ },
+ "id": "4336155157856256",
+ "vin": "mycar_vin1",
+ "brand": "opel",
+ "engine": [
+ {
+ "class": "Thermic",
+ "energy": "GPL"
+ },
+ {
+ "class": "Electric"
+ }
+ ],
+ "createdAt": "2019-08-24T14:15:22Z",
+ "updatedAt": "2019-08-24T14:15:22Z"
+ }
+ ]
+ }
+ }
\ No newline at end of file
diff --git a/bundles/org.openhab.binding.groupepsa/src/test/resources/dummy_vehiclestatus1.json b/bundles/org.openhab.binding.groupepsa/src/test/resources/dummy_vehiclestatus1.json
new file mode 100644
index 00000000000..476523ae082
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/test/resources/dummy_vehiclestatus1.json
@@ -0,0 +1,89 @@
+{
+ "lastPosition": {
+ "type": "Feature",
+ "geometry": {
+ "type": "Point",
+ "coordinates": [
+ 2.2221432,
+ 48.772476,
+ 172
+ ]
+ },
+ "properties": {
+ "updatedAt": "2019-08-24T14:15:22Z",
+ "type": "Aquire",
+ "signalQuality": 6
+ }
+ },
+ "preconditionning": {
+ "airConditioning": {
+ "updatedAt": "2019-08-24T14:15:22Z",
+ "status": "Disabled",
+ "programs": [
+ {
+ "enabled": false,
+ "recurrence": "Daily",
+ "start": "PT0S"
+ },
+ {
+ "enabled": false,
+ "recurrence": "Daily",
+ "start": "PT0S"
+ },
+ {
+ "enabled": false,
+ "recurrence": "Daily",
+ "start": "PT0S"
+ },
+ {
+ "enabled": false,
+ "recurrence": "Daily",
+ "start": "PT0S"
+ }
+ ]
+ }
+ },
+ "energy": [
+ {
+ "type": "Electric",
+ "level": 0,
+ "autonomy": 0,
+ "residual": 0
+ },
+ {
+ "type": "Fuel",
+ "level": 0,
+ "residual": 0
+ }
+ ],
+ "createdAt": "2019-08-24T14:15:22Z",
+ "autonomy": 2000,
+ "ignition": {
+ "createdAt": "2019-08-24T14:15:22Z"
+ },
+ "vin": "VF3A12BCDEF3456",
+ "privacy": {
+ "createdAt": "2019-08-24T14:15:22Z",
+ "state": "None"
+ },
+ "battery": {
+ "voltage": 0.55,
+ "current": 0.55,
+ "createdAt": "2019-08-24T14:15:22Z"
+ },
+ "kinetic": {
+ "createdAt": "2019-08-24T14:15:22Z",
+ "acceleration": 0.55,
+ "speed": 0.55,
+ "pace": 0.55,
+ "moving": false
+ },
+ "_links": {
+ "self": {
+ "href": "https://api.groupe-psa.com/connectedcar/v4/user/vehicles/{id}/status"
+ },
+ "vehicles": {
+ "href": "https://api.groupe-psa.com/connectedcar/v4/user/vehicles/{id}"
+ }
+ }
+}
\ No newline at end of file
diff --git a/bundles/org.openhab.binding.groupepsa/src/test/resources/dummy_vehiclestatus2.json b/bundles/org.openhab.binding.groupepsa/src/test/resources/dummy_vehiclestatus2.json
new file mode 100644
index 00000000000..e111ee10127
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/test/resources/dummy_vehiclestatus2.json
@@ -0,0 +1,180 @@
+{
+ "createdAt": "2019-08-24T14:15:22Z",
+ "_embedded": {
+ "extension": {
+ "kinetic": {
+ "createdAt": "2019-08-24T14:15:22Z",
+ "acceleration": 0,
+ "moving": true,
+ "pace": 0,
+ "speed": 0
+ },
+ "odometer": {
+ "createdAt": "2019-08-24T14:15:22Z",
+ "mileage": 0
+ }
+ }
+ },
+ "_links": {
+ "self": {
+ "deprecation": "string",
+ "href": "string",
+ "hreflang": "string",
+ "name": "string",
+ "profile": "string",
+ "templated": true,
+ "title": "string",
+ "type": "string"
+ },
+ "vehicle": {
+ "deprecation": "string",
+ "href": "string",
+ "hreflang": "string",
+ "name": "string",
+ "profile": "string",
+ "templated": true,
+ "title": "string",
+ "type": "string"
+ }
+ },
+ "battery": {
+ "createdAt": "2019-08-24T14:15:22Z",
+ "current": 0,
+ "voltage": 0
+ },
+ "doorsState": {
+ "lockedState": [
+ "Unlocked"
+ ],
+ "opening": [
+ {
+ "identifier": "Driver",
+ "state": "Open"
+ }
+ ],
+ "updatedAt": "2019-08-24T14:15:22Z"
+ },
+ "energy": [
+ {
+ "updatedAt": "2019-08-24T14:15:22Z",
+ "createdAt": "2019-08-24T14:15:22Z",
+ "autonomy": 1,
+ "battery": {
+ "capacity": 0,
+ "health": {
+ "capacity": 0,
+ "resistance": 0
+ }
+ },
+ "charging": {
+ "chargingMode": "No",
+ "chargingRate": 0,
+ "nextDelayedTime": "PT15M",
+ "plugged": true,
+ "remainingTime": "PT15M",
+ "status": "Disconnected"
+ },
+ "consumption": 0,
+ "level": 0,
+ "residual": 0,
+ "type": "Fuel"
+ },
+ {
+ "updatedAt": "2019-08-24T14:15:22Z",
+ "createdAt": "2019-08-24T14:15:22Z",
+ "autonomy": 2,
+ "battery": {
+ "capacity": 0,
+ "health": {
+ "capacity": 0,
+ "resistance": 0
+ }
+ },
+ "charging": {
+ "chargingMode": "No",
+ "chargingRate": 0,
+ "nextDelayedTime": "PT15M",
+ "plugged": true,
+ "remainingTime": "PT15M",
+ "status": "Disconnected"
+ },
+ "consumption": 0,
+ "level": 0,
+ "residual": 0,
+ "type": "Electric"
+ }
+ ],
+ "environment": {
+ "updatedAt": "2019-08-24T14:15:22Z",
+ "createdAt": "2019-08-24T14:15:22Z",
+ "air": {
+ "temp": 0
+ },
+ "luminosity": {
+ "day": true
+ }
+ },
+ "ignition": {
+ "updatedAt": "2019-08-24T14:15:22Z",
+ "type": "Stop"
+ },
+ "kinetic": {
+ "createdAt": "2019-08-24T14:15:22Z",
+ "acceleration": 0,
+ "moving": true,
+ "pace": 0,
+ "speed": 0
+ },
+ "lastPosition": {
+ "updatedAt": "2019-08-24T14:15:22Z",
+ "createdAt": "2019-08-24T14:15:22Z",
+ "geometry": {
+ "coordinates": [
+ 0, 0, 0
+ ],
+ "type": "Point"
+ },
+ "properties": {
+ "heading": 0,
+ "signalQuality": 0,
+ "type": "Estimate"
+ },
+ "type": "Feature"
+ },
+ "preconditionning": {
+ "airConditioning": {
+ "failureCause": "Defect",
+ "programs": [
+ {
+ "occurence": {
+ "day": [
+ "Mon"
+ ],
+ "week": [
+ "string"
+ ]
+ },
+ "recurrence": "Daily",
+ "start": "string",
+ "enabled": true,
+ "slot": 0
+ }
+ ],
+ "status": "Enabled",
+ "updatedAt": "2019-08-24T14:15:22Z"
+ }
+ },
+ "privacy": {
+ "createdAt": "2019-08-24T14:15:22Z",
+ "state": "None"
+ },
+ "safety": {
+ "createdAt": "2019-08-24T14:15:22Z",
+ "beltWarning": "Normal",
+ "eCallTriggeringRequest": "AirbagUnabled"
+ },
+ "service": {
+ "type": "Electric",
+ "updatedAt": "2019-08-24T14:15:22Z"
+ }
+ }
\ No newline at end of file
diff --git a/bundles/org.openhab.binding.groupepsa/src/test/resources/dummy_vehiclestatus3.json b/bundles/org.openhab.binding.groupepsa/src/test/resources/dummy_vehiclestatus3.json
new file mode 100644
index 00000000000..4b92f1c7198
--- /dev/null
+++ b/bundles/org.openhab.binding.groupepsa/src/test/resources/dummy_vehiclestatus3.json
@@ -0,0 +1,102 @@
+{
+ "lastPosition": {
+ "type": "Feature",
+ "geometry": {
+ "type": "Point",
+ "coordinates": [
+ 22.1111111,
+ 11.2222222
+ ]
+ },
+ "properties": {
+ "updatedAt": "2020-12-25T15:16:48Z",
+ "heading": 331,
+ "type": "Estimated"
+ }
+ },
+ "preconditionning": {
+ "airConditioning": {
+ "updatedAt": "2020-12-23T16:54:14Z",
+ "status": "Disabled",
+ "programs": [
+ {
+ "enabled": false,
+ "slot": 1,
+ "recurrence": "Daily",
+ "start": "PT6H15M",
+ "occurence": {
+ "day": [
+ "Mon",
+ "Tue",
+ "Wed",
+ "Thu",
+ "Fri"
+ ]
+ }
+ },
+ {
+ "enabled": false,
+ "slot": 2,
+ "recurrence": "Daily",
+ "start": "PT8H",
+ "occurence": {
+ "day": [
+ "Mon",
+ "Tue",
+ "Wed",
+ "Thu",
+ "Fri"
+ ]
+ }
+ }
+ ]
+ }
+ },
+ "energy": [
+ {
+ "updatedAt": "2020-12-23T16:54:13Z",
+ "type": "Electric",
+ "level": 77,
+ "autonomy": 200,
+ "charging": {
+ "plugged": false,
+ "status": "Disconnected",
+ "remainingTime": "PT0S",
+ "chargingRate": 0,
+ "chargingMode": "No",
+ "nextDelayedTime": "PT5H"
+ }
+ }
+ ],
+ "createdAt": "2020-12-25T15:16:48Z",
+ "battery": {
+ "voltage": 87,
+ "current": 0,
+ "createdAt": "2020-12-23T16:54:13Z"
+ },
+ "kinetic": {
+ "createdAt": "2020-12-25T15:16:48Z",
+ "moving": false
+ },
+ "privacy": {
+ "createdAt": "2020-12-25T15:16:43Z",
+ "state": "None"
+ },
+ "service": {
+ "type": "Electric",
+ "updatedAt": "2020-11-06T19:56:10Z"
+ },
+ "_links": {
+ "self": {
+ "href": "dummy_link/status"
+ },
+ "vehicles": {
+ "href": "dummy_link/vehicles"
+ }
+ },
+ "timed.odometer": {
+ "createdAt": null,
+ "mileage": 1539
+ },
+ "updatedAt": "2020-12-25T15:16:48Z"
+}
\ No newline at end of file
diff --git a/bundles/pom.xml b/bundles/pom.xml
index 66d4eccca40..52b37b21894 100644
--- a/bundles/pom.xml
+++ b/bundles/pom.xml
@@ -142,6 +142,7 @@
org.openhab.binding.globalcache
org.openhab.binding.gpstracker
org.openhab.binding.gree
+ org.openhab.binding.groupepsa
org.openhab.binding.groheondus
org.openhab.binding.guntamatic
org.openhab.binding.haassohnpelletstove