1
1
/*
2
- * Copyright (c) 2012-2024 Red Hat, Inc.
2
+ * Copyright (c) 2012-2025 Red Hat, Inc.
3
3
* This program and the accompanying materials are made
4
4
* available under the terms of the Eclipse Public License 2.0
5
5
* which is available at https://www.eclipse.org/legal/epl-2.0/
48
48
import org .eclipse .che .api .core .NotFoundException ;
49
49
import org .eclipse .che .api .core .ServerException ;
50
50
import org .eclipse .che .api .core .UnauthorizedException ;
51
+ import org .eclipse .che .api .factory .server .bitbucket .server .BitbucketApplicationProperties ;
51
52
import org .eclipse .che .api .factory .server .bitbucket .server .BitbucketPersonalAccessToken ;
52
53
import org .eclipse .che .api .factory .server .bitbucket .server .BitbucketServerApiClient ;
53
54
import org .eclipse .che .api .factory .server .bitbucket .server .BitbucketUser ;
@@ -77,6 +78,8 @@ public class HttpBitbucketServerApiClient implements BitbucketServerApiClient {
77
78
78
79
private static final Logger LOG = LoggerFactory .getLogger (HttpBitbucketServerApiClient .class );
79
80
private static final Duration DEFAULT_HTTP_TIMEOUT = ofSeconds (10 );
81
+ private static final ObjectMapper OBJECT_MAPPER = new ObjectMapper ();
82
+ public static final String USERNAME_HEADER = "x-ausername" ;
80
83
private final URI serverUri ;
81
84
private final OAuthAuthenticator authenticator ;
82
85
private final OAuthAPI oAuthAPI ;
@@ -166,10 +169,10 @@ public void deletePersonalAccessTokens(String tokenId)
166
169
executeRequest (
167
170
httpClient ,
168
171
request ,
169
- inputStream -> {
172
+ response -> {
170
173
try {
171
174
String result =
172
- CharStreams .toString (new InputStreamReader (inputStream , Charsets .UTF_8 ));
175
+ CharStreams .toString (new InputStreamReader (response . body () , Charsets .UTF_8 ));
173
176
return OM .readValue (result , String .class );
174
177
} catch (IOException e ) {
175
178
throw new UncheckedIOException (e );
@@ -210,10 +213,10 @@ public BitbucketPersonalAccessToken createPersonalAccessTokens(
210
213
return executeRequest (
211
214
httpClient ,
212
215
request ,
213
- inputStream -> {
216
+ response -> {
214
217
try {
215
218
String result =
216
- CharStreams .toString (new InputStreamReader (inputStream , Charsets .UTF_8 ));
219
+ CharStreams .toString (new InputStreamReader (response . body () , Charsets .UTF_8 ));
217
220
return OM .readValue (result , BitbucketPersonalAccessToken .class );
218
221
} catch (IOException e ) {
219
222
throw new UncheckedIOException (e );
@@ -262,10 +265,10 @@ public BitbucketPersonalAccessToken getPersonalAccessToken(String tokenId, Strin
262
265
return executeRequest (
263
266
httpClient ,
264
267
request ,
265
- inputStream -> {
268
+ response -> {
266
269
try {
267
270
String result =
268
- CharStreams .toString (new InputStreamReader (inputStream , Charsets .UTF_8 ));
271
+ CharStreams .toString (new InputStreamReader (response . body () , Charsets .UTF_8 ));
269
272
return OM .readValue (result , BitbucketPersonalAccessToken .class );
270
273
} catch (IOException e ) {
271
274
throw new UncheckedIOException (e );
@@ -283,13 +286,7 @@ private String getUserSlug(Optional<String> token)
283
286
284
287
private BitbucketUser getUser (Optional <String > token )
285
288
throws ScmCommunicationException , ScmUnauthorizedException , ScmItemNotFoundException {
286
- URI uri ;
287
- try {
288
- uri = serverUri .resolve ("./plugins/servlet/applinks/whoami" );
289
- } catch (IllegalArgumentException e ) {
290
- // if the slug contains invalid characters (space for example) then the URI will be invalid
291
- throw new ScmCommunicationException (e .getMessage (), e );
292
- }
289
+ URI uri = serverUri .resolve ("/rest/api/1.0/application-properties" );
293
290
294
291
HttpRequest request =
295
292
HttpRequest .newBuilder (uri )
@@ -301,29 +298,15 @@ private BitbucketUser getUser(Optional<String> token)
301
298
.timeout (DEFAULT_HTTP_TIMEOUT )
302
299
.build ();
303
300
304
- String username ;
301
+ HttpResponse < InputStream > response ;
305
302
try {
306
303
LOG .trace ("executeRequest={}" , request );
307
- username =
308
- executeRequest (
309
- httpClient ,
310
- request ,
311
- inputStream -> {
312
- try {
313
- return CharStreams .toString (new InputStreamReader (inputStream , Charsets .UTF_8 ));
314
- } catch (IOException e ) {
315
- throw new UncheckedIOException (e );
316
- }
317
- });
304
+ response = executeRequest (httpClient , request , r -> r );
318
305
} catch (ScmBadRequestException e ) {
319
306
throw new ScmCommunicationException (e .getMessage (), e );
320
307
}
321
308
322
- // Only authenticated users can do the request below, so we must ensure that the username is not
323
- // empty
324
- if (isNullOrEmpty (username )) {
325
- throw buildScmUnauthorizedException ();
326
- }
309
+ String username = getUsername (response );
327
310
328
311
try {
329
312
List <BitbucketUser > users =
@@ -340,6 +323,26 @@ private BitbucketUser getUser(Optional<String> token)
340
323
}
341
324
}
342
325
326
+ private String getUsername (HttpResponse <InputStream > response )
327
+ throws ScmCommunicationException , ScmUnauthorizedException {
328
+ try {
329
+ // Try to get the username from the response header.
330
+ if (response .headers ().firstValue (USERNAME_HEADER ).isPresent ()) {
331
+ return response .headers ().firstValue (USERNAME_HEADER ).get ();
332
+ } else {
333
+ String result =
334
+ CharStreams .toString (new InputStreamReader (response .body (), Charsets .UTF_8 ));
335
+ // Convert the response data to the Bitbucket Server info object.
336
+ OBJECT_MAPPER .readValue (result , BitbucketApplicationProperties .class );
337
+ // Throw the unauthorized exception if the response contains the Bitbucket info.
338
+ throw buildScmUnauthorizedException ();
339
+ }
340
+ } catch (IOException e ) {
341
+ // The response does not contain the Bitbucket Server info
342
+ throw new ScmCommunicationException ("Bad request" );
343
+ }
344
+ }
345
+
343
346
private <T > List <T > doGetItems (Optional <String > token , Class <T > tClass , String api , String filter )
344
347
throws ScmUnauthorizedException , ScmCommunicationException , ScmBadRequestException ,
345
348
ScmItemNotFoundException {
@@ -377,10 +380,10 @@ private <T> Page<T> doGetPage(
377
380
return executeRequest (
378
381
httpClient ,
379
382
request ,
380
- inputStream -> {
383
+ response -> {
381
384
try {
382
385
String result =
383
- CharStreams .toString (new InputStreamReader (inputStream , Charsets .UTF_8 ));
386
+ CharStreams .toString (new InputStreamReader (response . body () , Charsets .UTF_8 ));
384
387
return OM .readValue (result , typeReference );
385
388
} catch (IOException e ) {
386
389
throw new UncheckedIOException (e );
@@ -389,15 +392,17 @@ private <T> Page<T> doGetPage(
389
392
}
390
393
391
394
private <T > T executeRequest (
392
- HttpClient httpClient , HttpRequest request , Function <InputStream , T > bodyConverter )
395
+ HttpClient httpClient ,
396
+ HttpRequest request ,
397
+ Function <HttpResponse <InputStream >, T > bodyConverter )
393
398
throws ScmBadRequestException , ScmItemNotFoundException , ScmCommunicationException ,
394
399
ScmUnauthorizedException {
395
400
try {
396
401
HttpResponse <InputStream > response =
397
402
httpClient .send (request , HttpResponse .BodyHandlers .ofInputStream ());
398
403
LOG .trace ("executeRequest={} response {}" , request , response .statusCode ());
399
404
if (response .statusCode () == 200 ) {
400
- return bodyConverter .apply (response . body () );
405
+ return bodyConverter .apply (response );
401
406
} else if (response .statusCode () == 204 ) {
402
407
return null ;
403
408
} else {
0 commit comments