18
18
import java .security .cert .CertificateException ;
19
19
import java .util .Optional ;
20
20
import java .util .concurrent .ExecutionException ;
21
+ import java .util .concurrent .TimeUnit ;
21
22
import java .util .concurrent .TimeoutException ;
22
23
23
24
import org .eclipse .jdt .annotation .NonNullByDefault ;
26
27
import org .eclipse .jetty .client .HttpResponseException ;
27
28
import org .eclipse .jetty .client .api .AuthenticationStore ;
28
29
import org .eclipse .jetty .client .api .ContentResponse ;
29
- import org .eclipse .jetty .client .api .Request ;
30
- import org .eclipse .jetty .client .api .Response ;
31
30
import org .eclipse .jetty .client .util .StringContentProvider ;
32
31
import org .eclipse .jetty .http .HttpHeader ;
33
32
import org .eclipse .jetty .http .HttpMethod ;
@@ -55,8 +54,10 @@ public class HueSyncConnection {
55
54
public static final ObjectMapper OBJECT_MAPPER = new ObjectMapper ()
56
55
.configure (DeserializationFeature .FAIL_ON_UNKNOWN_PROPERTIES , false );
57
56
/**
58
- * Request format: The Sync Box API can be accessed locally via HTTPS on root level (port 443,
59
- * /api/v1), resource level /api/v1/<resource> and in some cases sub-resource level
57
+ * Request format: The Sync Box API can be accessed locally via HTTPS on root
58
+ * level (port 443,
59
+ * /api/v1), resource level /api/v1/<resource> and in some cases sub-resource
60
+ * level
60
61
* /api/v1/<resource>/<sub-resource>.
61
62
*/
62
63
private static final String REQUEST_FORMAT = "https://%s:%s/%s/%s" ;
@@ -72,6 +73,41 @@ public class HueSyncConnection {
72
73
73
74
private Optional <HueSyncAuthenticationResult > authentication = Optional .empty ();
74
75
76
+ private class Request {
77
+
78
+ private final String endpoint ;
79
+
80
+ private HttpMethod method = HttpMethod .GET ;
81
+ private String payload = "" ;
82
+
83
+ private Request (HttpMethod httpMethod , String endpoint , String payload ) {
84
+ this .method = httpMethod ;
85
+ this .endpoint = endpoint ;
86
+ this .payload = payload ;
87
+ }
88
+
89
+ protected Request (String endpoint ) {
90
+ this .endpoint = endpoint ;
91
+ }
92
+
93
+ private Request (HttpMethod httpMethod , String endpoint ) {
94
+ this .method = httpMethod ;
95
+ this .endpoint = endpoint ;
96
+ }
97
+
98
+ protected ContentResponse execute () throws InterruptedException , ExecutionException , TimeoutException {
99
+ String uri = String .format (REQUEST_FORMAT , host , port , API , endpoint );
100
+
101
+ var request = httpClient .newRequest (uri ).method (method ).timeout (1 , TimeUnit .SECONDS );
102
+ if (!payload .isBlank ()) {
103
+ request .header (HttpHeader .CONTENT_TYPE , MimeTypes .Type .APPLICATION_JSON_UTF_8 .toString ())
104
+ .content (new StringContentProvider (payload ));
105
+ }
106
+
107
+ return request .send ();
108
+ }
109
+ }
110
+
75
111
protected String registrationId = "" ;
76
112
77
113
public HueSyncConnection (HttpClient httpClient , String host , Integer port )
@@ -102,46 +138,30 @@ public void updateAuthentication(String id, String token) {
102
138
103
139
// #region protected
104
140
protected @ Nullable <T > T executeRequest (HttpMethod method , String endpoint , String payload ,
105
- @ Nullable Class <T > type ) {
106
- try {
107
- return this .processedResponse (this .executeRequest (method , endpoint , payload ), type );
108
- } catch (ExecutionException e ) {
109
- this .handleExecutionException (e );
110
- } catch (InterruptedException | TimeoutException e ) {
111
- this .logger .warn ("{}" , e .getMessage ());
112
- }
141
+ @ Nullable Class <T > type ) throws HueSyncConnectionException {
113
142
114
- return null ;
143
+ return this . executeRequest ( new Request ( method , endpoint , payload ), type ) ;
115
144
}
116
145
117
- protected @ Nullable <T > T executeGetRequest (String endpoint , Class <T > type ) {
118
- try {
119
- return this .processedResponse (this .executeGetRequest (endpoint ), type );
120
- } catch (ExecutionException e ) {
121
- this .handleExecutionException (e );
122
- } catch (InterruptedException | TimeoutException e ) {
123
- this .logger .warn ("{}" , e .getMessage ());
124
- }
146
+ protected @ Nullable <T > T executeRequest (HttpMethod httpMethod , String endpoint , @ Nullable Class <T > type )
147
+ throws HueSyncConnectionException {
148
+ return this .executeRequest (new Request (httpMethod , endpoint ), type );
149
+ }
125
150
126
- return null ;
151
+ protected @ Nullable <T > T executeGetRequest (String endpoint , Class <T > type ) throws HueSyncConnectionException {
152
+ return this .executeRequest (new Request (endpoint ), type );
127
153
}
128
154
129
155
protected boolean isRegistered () {
130
156
return this .authentication .isPresent ();
131
157
}
132
158
133
- protected void unregisterDevice () {
159
+ protected void unregisterDevice () throws HueSyncConnectionException {
134
160
if (this .isRegistered ()) {
135
- try {
136
- String endpoint = ENDPOINTS .REGISTRATIONS + "/" + this .registrationId ;
137
- ContentResponse response = this .executeRequest (HttpMethod .DELETE , endpoint );
161
+ String endpoint = ENDPOINTS .REGISTRATIONS + "/" + this .registrationId ;
138
162
139
- if (response .getStatus () == HttpStatus .OK_200 ) {
140
- this .removeAuthentication ();
141
- }
142
- } catch (InterruptedException | TimeoutException | ExecutionException e ) {
143
- this .logger .warn ("{}" , e .getMessage ());
144
- }
163
+ this .executeRequest (HttpMethod .DELETE , endpoint , null );
164
+ this .removeAuthentication ();
145
165
}
146
166
}
147
167
@@ -151,93 +171,55 @@ protected void dispose() {
151
171
// #endregion
152
172
153
173
// #region private
154
- private @ Nullable <T > T processedResponse (Response response , @ Nullable Class <T > type ) {
155
- int status = response .getStatus ();
174
+
175
+ private @ Nullable <T > T executeRequest (Request request , @ Nullable Class <T > type ) throws HueSyncConnectionException {
176
+ String message = "@text/connection.generic-error" ;
177
+
156
178
try {
179
+ ContentResponse response = request .execute ();
180
+
157
181
/*
158
182
* 400 Invalid State: Registration in progress
159
183
*
160
- * 401 Authentication failed: If credentials are missing or invalid, errors out. If
161
- * credentials are missing, continues on to GET only the Configuration state when
184
+ * 401 Authentication failed: If credentials are missing or invalid, errors out.
185
+ * If
186
+ * credentials are missing, continues on to GET only the Configuration state
187
+ * when
162
188
* unauthenticated, to allow for device identification.
163
189
*
164
190
* 404 Invalid URI Path: Accessing URI path which is not supported
165
191
*
166
192
* 500 Internal: Internal errors like out of memory
167
193
*/
168
- switch (status ) {
194
+ switch (response . getStatus () ) {
169
195
case HttpStatus .OK_200 -> {
170
- return (type != null && (response instanceof ContentResponse ))
171
- ? this .deserialize (((ContentResponse ) response ).getContentAsString (), type )
172
- : null ;
196
+ return this .deserialize (response .getContentAsString (), type );
173
197
}
174
- case HttpStatus .BAD_REQUEST_400 -> this .logger .debug ("registration in progress: no token received yet" );
175
- case HttpStatus .UNAUTHORIZED_401 -> {
176
- this .authentication = Optional .empty ();
177
- throw new HueSyncConnectionException ("@text/connection.invalid-login" );
198
+ case HttpStatus .BAD_REQUEST_400 -> {
199
+ logger .debug ("registration in progress: no token received yet" );
200
+ return null ;
178
201
}
179
- case HttpStatus .NOT_FOUND_404 -> this .logger .warn ("invalid device URI or API endpoint" );
180
- case HttpStatus .INTERNAL_SERVER_ERROR_500 -> this .logger .warn ("hue sync box server problem" );
181
- default -> this .logger .warn ("unexpected HTTP status: {}" , status );
202
+ case HttpStatus .UNAUTHORIZED_401 -> message = "@text/connection.invalid-login" ;
203
+ case HttpStatus .NOT_FOUND_404 -> message = "@text/connection.generic-error" ;
182
204
}
183
- } catch (HueSyncConnectionException e ) {
184
- this .logger .warn ("{}" , e .getMessage ());
185
- }
186
- return null ;
187
- }
205
+ throw new HueSyncConnectionException (message , new HttpResponseException (message , response ));
206
+ } catch (JsonProcessingException | InterruptedException | ExecutionException | TimeoutException e ) {
188
207
189
- private @ Nullable <T > T deserialize (String json , Class <T > type ) {
190
- try {
191
- return OBJECT_MAPPER .readValue (json , type );
192
- } catch (JsonProcessingException | NoClassDefFoundError e ) {
193
- this .logger .error ("{}" , e .getMessage ());
208
+ var logMessage = message + " {}" ;
209
+ this .logger .warn (logMessage , e .toString ());
194
210
195
- return null ;
211
+ throw new HueSyncConnectionException ( message , e ) ;
196
212
}
197
213
}
198
214
199
- private ContentResponse executeRequest (HttpMethod method , String endpoint )
200
- throws InterruptedException , TimeoutException , ExecutionException {
201
- return this .executeRequest (method , endpoint , "" );
202
- }
203
-
204
- private ContentResponse executeGetRequest (String endpoint )
205
- throws InterruptedException , ExecutionException , TimeoutException {
206
- String uri = String .format (REQUEST_FORMAT , this .host , this .port , API , endpoint );
207
-
208
- return httpClient .GET (uri );
209
- }
210
-
211
- private ContentResponse executeRequest (HttpMethod method , String endpoint , String payload )
212
- throws InterruptedException , TimeoutException , ExecutionException {
213
- String uri = String .format (REQUEST_FORMAT , this .host , this .port , API , endpoint );
214
-
215
- Request request = this .httpClient .newRequest (uri ).method (method );
216
-
217
- this .logger .trace ("uri: {}" , uri );
218
- this .logger .trace ("method: {}" , method );
219
- this .logger .trace ("payload: {}" , payload );
220
-
221
- if (!payload .isBlank ()) {
222
- request .header (HttpHeader .CONTENT_TYPE , MimeTypes .Type .APPLICATION_JSON_UTF_8 .toString ())
223
- .content (new StringContentProvider (payload ));
224
- }
225
-
226
- return request .send ();
227
- }
228
-
229
- private void handleExecutionException (ExecutionException e ) {
230
- this .logger .warn ("{}" , e .getMessage ());
231
-
232
- Throwable cause = e .getCause ();
233
- if (cause != null && cause instanceof HttpResponseException ) {
234
- processedResponse (((HttpResponseException ) cause ).getResponse (), null );
235
- }
215
+ private @ Nullable <T > T deserialize (String json , @ Nullable Class <T > type ) throws JsonProcessingException {
216
+ return type == null ? null : OBJECT_MAPPER .readValue (json , type );
236
217
}
237
218
238
219
private void removeAuthentication () {
239
220
AuthenticationStore store = this .httpClient .getAuthenticationStore ();
240
221
store .clearAuthenticationResults ();
222
+
241
223
this .httpClient .setAuthenticationStore (store );
242
224
243
225
this .registrationId = "" ;
0 commit comments