2 * Copyright (c) 2010-2022 Contributors to the openHAB project
4 * See the NOTICE file(s) distributed with this work for additional
7 * This program and the accompanying materials are made available under the
8 * terms of the Eclipse Public License 2.0 which is available at
9 * http://www.eclipse.org/legal/epl-2.0
11 * SPDX-License-Identifier: EPL-2.0
13 package org.openhab.binding.netatmo.internal.api.data;
15 import static org.openhab.binding.netatmo.internal.NetatmoBindingConstants.*;
16 import static org.openhab.core.library.CoreItemFactory.*;
17 import static org.openhab.core.library.unit.MetricPrefix.*;
20 import java.util.EnumSet;
21 import java.util.HashMap;
24 import java.util.stream.Collectors;
25 import java.util.stream.Stream;
27 import javax.measure.Unit;
29 import org.eclipse.jdt.annotation.NonNullByDefault;
30 import org.openhab.core.library.unit.SIUnits;
31 import org.openhab.core.library.unit.Units;
32 import org.openhab.core.types.StateDescriptionFragment;
33 import org.openhab.core.types.StateDescriptionFragmentBuilder;
34 import org.openhab.core.types.util.UnitUtils;
36 import com.google.gson.annotations.SerializedName;
39 * This class holds various definitions and settings provided by the Netatmo
42 * @author Gaƫl L'hopital - Initial contribution
45 public class NetatmoConstants {
46 public static class Measure {
47 public final double minValue;
48 public final double maxValue;
49 public final int scale;
50 public final Unit<?> unit;
52 private Measure(double minValue, double maxValue, double precision, Unit<?> unit) {
53 this.minValue = minValue;
54 this.maxValue = maxValue;
56 String[] splitter = Double.valueOf(precision).toString().split("\\.");
57 if (splitter.length > 1) {
58 int dec = Integer.parseInt(splitter[1]);
59 this.scale = dec > 0 ? Integer.toString(dec).length() : 0;
66 public static class MeasureChannelDetails {
67 private static final StateDescriptionFragmentBuilder BUILDER = StateDescriptionFragmentBuilder.create();
68 public final URI configURI;
69 public final String itemType;
70 public final StateDescriptionFragment stateDescriptionFragment;
72 private MeasureChannelDetails(String measureType, String itemType, String pattern) {
73 this.configURI = URI.create(String.join(":", BINDING_ID, measureType, "config"));
74 this.itemType = itemType;
75 this.stateDescriptionFragment = BUILDER.withReadOnly(true).withPattern(pattern).build();
79 public enum MeasureClass {
80 INSIDE_TEMPERATURE(0, 50, 0.3, SIUnits.CELSIUS, "temp", "measure", true),
81 OUTSIDE_TEMPERATURE(-40, 65, 0.3, SIUnits.CELSIUS, "temp", "measure", true),
82 HEAT_INDEX(-40, 65, 1, SIUnits.CELSIUS, "", "", false),
83 PRESSURE(260, 1260, 1, HECTO(SIUnits.PASCAL), "pressure", "measure", true),
84 CO2(0, 5000, 50, Units.PARTS_PER_MILLION, "co2", "measure", true),
85 NOISE(35, 120, 1, Units.DECIBEL, "noise", "measure", true),
86 RAIN_QUANTITY(0, 150, 0.1, MILLI(SIUnits.METRE), "sum_rain", "sum_rain", false),
87 RAIN_INTENSITY(0, 150, 0.1, Units.MILLIMETRE_PER_HOUR, "", "", false),
88 WIND_SPEED(0, 160, 1.8, SIUnits.KILOMETRE_PER_HOUR, "", "", false),
89 WIND_ANGLE(0, 360, 5, Units.DEGREE_ANGLE, "", "", false),
90 HUMIDITY(0, 100, 3, Units.PERCENT, "hum", "measure", true);
92 public static final EnumSet<MeasureClass> AS_SET = EnumSet.allOf(MeasureClass.class);
94 public final Measure measureDefinition;
95 public final String apiDescriptor;
96 public final Map<String, MeasureChannelDetails> channels = new HashMap<>(2);
98 MeasureClass(double min, double max, double precision, Unit<?> unit, String apiDescriptor, String confFragment,
100 this.measureDefinition = new Measure(min, max, precision, unit);
101 this.apiDescriptor = apiDescriptor;
102 if (!apiDescriptor.isBlank()) {
103 String dimension = UnitUtils.getDimensionName(unit);
105 channels.put(String.join("-", apiDescriptor, "measurement"),
106 new MeasureChannelDetails(confFragment, String.join(":", NUMBER, dimension),
107 String.format("%%.%df %s", measureDefinition.scale, UnitUtils.UNIT_PLACEHOLDER)));
109 channels.put(String.join("-", apiDescriptor, GROUP_TIMESTAMP),
110 new MeasureChannelDetails(GROUP_TIMESTAMP, DATETIME, "%1$tA, %1$td.%1$tm. %1$tH:%1$tM"));
117 public static final String URL_API = "https://api.netatmo.com/";
118 public static final String URL_APP = "https://app.netatmo.net/";
119 public static final String PATH_OAUTH = "oauth2";
120 public static final String SUB_PATH_TOKEN = "token";
121 public static final String SUB_PATH_AUTHORIZE = "authorize";
122 public static final String PATH_API = "api";
123 public static final String PATH_COMMAND = "command";
124 public static final String PATH_STATE = "setstate";
125 public static final String SUB_PATH_PERSON_AWAY = "setpersonsaway";
126 public static final String SUB_PATH_PERSON_HOME = "setpersonshome";
127 public static final String SUB_PATH_HOMES_DATA = "homesdata";
128 public static final String SUB_PATH_ADDWEBHOOK = "addwebhook";
129 public static final String SUB_PATH_DROPWEBHOOK = "dropwebhook";
130 public static final String SUB_PATH_SETROOMTHERMPOINT = "setroomthermpoint";
131 public static final String SUB_PATH_SETTHERMMODE = "setthermmode";
132 public static final String SUB_PATH_SWITCHSCHEDULE = "switchschedule";
133 public static final String SUB_PATH_GETSTATION = "getstationsdata";
134 public static final String SUB_PATH_GETMEASURE = "getmeasure";
135 public static final String SUB_PATH_HOMESTATUS = "homestatus";
136 public static final String SUB_PATH_HOMECOACH = "gethomecoachsdata";
137 public static final String SUB_PATH_GETEVENTS = "getevents";
138 public static final String SUB_PATH_PING = "ping";
139 public static final String SUB_PATH_CHANGESTATUS = "changestatus";
140 public static final String PARAM_DEVICEID = "device_id";
141 public static final String PARAM_MODULEID = "module_id";
142 public static final String PARAM_HOMEID = "home_id";
143 public static final String PARAM_ROOMID = "room_id";
144 public static final String PARAM_PERSONID = "person_id";
145 public static final String PARAM_SCHEDULEID = "schedule_id";
146 public static final String PARAM_OFFSET = "offset";
147 public static final String PARAM_GATEWAYTYPE = "gateway_types";
148 public static final String PARAM_MODE = "mode";
149 public static final String PARAM_URL = "url";
150 public static final String PARAM_FAVORITES = "get_favorites";
151 public static final String PARAM_STATUS = "status";
153 // Autentication process params
154 public static final String PARAM_ERROR = "error";
157 public static final int THERM_MAX_SETPOINT = 30;
160 public static enum Scope {
161 @SerializedName("read_station")
163 @SerializedName("read_thermostat")
165 @SerializedName("write_thermostat")
167 @SerializedName("read_camera")
169 @SerializedName("write_camera")
171 @SerializedName("access_camera")
173 @SerializedName("read_presence")
175 @SerializedName("write_presence")
177 @SerializedName("access_presence")
179 @SerializedName("read_smokedetector")
181 @SerializedName("read_homecoach")
183 @SerializedName("read_doorbell")
185 @SerializedName("write_doorbell")
187 @SerializedName("access_doorbell")
192 private static final Set<Scope> SMOKE = Set.of(Scope.READ_SMOKEDETECTOR);
193 private static final Set<Scope> WELCOME = Set.of(Scope.READ_CAMERA, Scope.WRITE_CAMERA, Scope.ACCESS_CAMERA);
194 private static final Set<Scope> DOORBELL = Set.of(Scope.READ_DOORBELL, Scope.WRITE_DOORBELL, Scope.ACCESS_DOORBELL);
195 private static final Set<Scope> PRESENCE = Set.of(Scope.READ_PRESENCE, Scope.WRITE_PRESENCE, Scope.ACCESS_PRESENCE);
197 // Radio signal quality thresholds
198 static final int[] WIFI_SIGNAL_LEVELS = new int[] { 99, 84, 69, 54 }; // Resp : bad, average, good, full
199 static final int[] RADIO_SIGNAL_LEVELS = new int[] { 90, 80, 70, 60 }; // Resp : low, medium, high, full
201 public static enum FeatureArea {
202 AIR_CARE(Scope.READ_HOMECOACH),
203 WEATHER(Scope.READ_STATION),
204 ENERGY(Scope.READ_THERMOSTAT, Scope.WRITE_THERMOSTAT),
205 SECURITY(Stream.of(WELCOME, PRESENCE, SMOKE, DOORBELL).flatMap(Set::stream).toArray(Scope[]::new)),
208 public static final Set<FeatureArea> AS_SET = EnumSet.allOf(FeatureArea.class);
210 public static String toScopeString(Set<FeatureArea> featureSet) {
211 return featureSet.stream().map(fa -> fa.scopes).flatMap(Set::stream).map(s -> s.name().toLowerCase())
212 .collect(Collectors.joining(" "));
215 public final Set<Scope> scopes;
217 FeatureArea(Scope... scopes) {
218 this.scopes = Set.of(scopes);
222 // Thermostat definitions
223 public static enum SetpointMode {
224 @SerializedName("program")
226 @SerializedName("away")
228 @SerializedName("hg")
230 @SerializedName("manual")
232 @SerializedName("off")
234 @SerializedName("max")
236 @SerializedName("schedule")
237 SCHEDULE("schedule"),
241 public final String apiDescriptor;
243 SetpointMode(String descriptor) {
244 this.apiDescriptor = descriptor;
248 public static enum ThermostatZoneType {
265 public final String zoneId;
267 private ThermostatZoneType(String id) {
272 public enum FloodLightMode {
273 @SerializedName("on")
275 @SerializedName("off")
277 @SerializedName("auto")
282 public enum EventCategory {
283 @SerializedName("human")
285 @SerializedName("animal")
287 @SerializedName("vehicle")
292 public enum TrendDescription {
293 @SerializedName("up")
295 @SerializedName("stable")
297 @SerializedName("down")
302 public enum VideoStatus {
303 @SerializedName("recording")
305 @SerializedName("available")
307 @SerializedName("deleted")
312 public enum SdCardStatus {
324 SD_CARD_INCOMPATIBLE_SPEED,
326 SD_CARD_INSUFFICIENT_SPACE,
330 public enum AlimentationStatus {
332 ALIM_INCORRECT_POWER,
338 public enum BatteryState {
339 @SerializedName("full")
341 @SerializedName("high")
343 @SerializedName("medium")
345 @SerializedName("low")
349 public final int level;
351 BatteryState(int i) {
356 public enum ServiceError {
357 @SerializedName("99")
359 @SerializedName("-2")
360 UNKNOWN_ERROR_IN_OAUTH,
361 @SerializedName("-1")
364 ACCESS_TOKEN_MISSING,
366 INVALID_TOKEN_MISSING,
368 ACCESS_TOKEN_EXPIRED,
370 APPLICATION_DEACTIVATED,
375 @SerializedName("10")
377 @SerializedName("13")
379 @SerializedName("19")
381 @SerializedName("21")
383 @SerializedName("22")
384 APPLICATION_NOT_FOUND,
385 @SerializedName("23")
387 @SerializedName("25")
389 @SerializedName("26")
390 MAXIMUM_USAGE_REACHED,
391 @SerializedName("30")
392 INVALID_REFRESH_TOKEN,
393 @SerializedName("31")
395 @SerializedName("35")
397 @SerializedName("36")
399 @SerializedName("37")
400 NO_MORE_SPACE_AVAILABLE_ON_THE_CAMERA,
401 @SerializedName("40")
402 JSON_GIVEN_HAS_AN_INVALID_ENCODING,
403 @SerializedName("41")
404 DEVICE_IS_UNREACHABLE;