Skip to content

Commit e871942

Browse files
authored
Rework the Bitbucket Server getUser request (#790) (#792)
Since Bitbucket Server version 8.19.14, the /plugins/servlet/applinks/whoami request does not return username, if PAT is used. Change the getUser() request to /rest/api/1.0/application-properties instead and extract the username from the response headers.
1 parent edb9aff commit e871942

File tree

6 files changed

+200
-47
lines changed

6 files changed

+200
-47
lines changed

assembly/assembly-wsmaster-war/src/main/webapp/WEB-INF/classes/che/che.properties

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
#
2-
# Copyright (c) 2012-2024 Red Hat, Inc.
2+
# Copyright (c) 2012-2025 Red Hat, Inc.
33
# This program and the accompanying materials are made
44
# available under the terms of the Eclipse Public License 2.0
55
# which is available at https://www.eclipse.org/legal/epl-2.0/

wsmaster/che-core-api-factory-bitbucket-server/src/main/java/org/eclipse/che/api/factory/server/bitbucket/HttpBitbucketServerApiClient.java

+43-35
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2012-2024 Red Hat, Inc.
2+
* Copyright (c) 2012-2025 Red Hat, Inc.
33
* This program and the accompanying materials are made
44
* available under the terms of the Eclipse Public License 2.0
55
* which is available at https://www.eclipse.org/legal/epl-2.0/
@@ -48,6 +48,7 @@
4848
import org.eclipse.che.api.core.NotFoundException;
4949
import org.eclipse.che.api.core.ServerException;
5050
import org.eclipse.che.api.core.UnauthorizedException;
51+
import org.eclipse.che.api.factory.server.bitbucket.server.BitbucketApplicationProperties;
5152
import org.eclipse.che.api.factory.server.bitbucket.server.BitbucketPersonalAccessToken;
5253
import org.eclipse.che.api.factory.server.bitbucket.server.BitbucketServerApiClient;
5354
import org.eclipse.che.api.factory.server.bitbucket.server.BitbucketUser;
@@ -77,6 +78,8 @@ public class HttpBitbucketServerApiClient implements BitbucketServerApiClient {
7778

7879
private static final Logger LOG = LoggerFactory.getLogger(HttpBitbucketServerApiClient.class);
7980
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";
8083
private final URI serverUri;
8184
private final OAuthAuthenticator authenticator;
8285
private final OAuthAPI oAuthAPI;
@@ -166,10 +169,10 @@ public void deletePersonalAccessTokens(String tokenId)
166169
executeRequest(
167170
httpClient,
168171
request,
169-
inputStream -> {
172+
response -> {
170173
try {
171174
String result =
172-
CharStreams.toString(new InputStreamReader(inputStream, Charsets.UTF_8));
175+
CharStreams.toString(new InputStreamReader(response.body(), Charsets.UTF_8));
173176
return OM.readValue(result, String.class);
174177
} catch (IOException e) {
175178
throw new UncheckedIOException(e);
@@ -210,10 +213,10 @@ public BitbucketPersonalAccessToken createPersonalAccessTokens(
210213
return executeRequest(
211214
httpClient,
212215
request,
213-
inputStream -> {
216+
response -> {
214217
try {
215218
String result =
216-
CharStreams.toString(new InputStreamReader(inputStream, Charsets.UTF_8));
219+
CharStreams.toString(new InputStreamReader(response.body(), Charsets.UTF_8));
217220
return OM.readValue(result, BitbucketPersonalAccessToken.class);
218221
} catch (IOException e) {
219222
throw new UncheckedIOException(e);
@@ -262,10 +265,10 @@ public BitbucketPersonalAccessToken getPersonalAccessToken(String tokenId, Strin
262265
return executeRequest(
263266
httpClient,
264267
request,
265-
inputStream -> {
268+
response -> {
266269
try {
267270
String result =
268-
CharStreams.toString(new InputStreamReader(inputStream, Charsets.UTF_8));
271+
CharStreams.toString(new InputStreamReader(response.body(), Charsets.UTF_8));
269272
return OM.readValue(result, BitbucketPersonalAccessToken.class);
270273
} catch (IOException e) {
271274
throw new UncheckedIOException(e);
@@ -283,13 +286,10 @@ private String getUserSlug(Optional<String> token)
283286

284287
private BitbucketUser getUser(Optional<String> token)
285288
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+
// We use the application-properties request to obtain the authenticated username from the
290+
// response headers. The request does not fail if no authentication is passed, see:
291+
// https://developer.atlassian.com/server/bitbucket/rest/v906/api-group-system-maintenance/#api-api-latest-application-properties-get
292+
URI uri = serverUri.resolve("/rest/api/1.0/application-properties");
293293

294294
HttpRequest request =
295295
HttpRequest.newBuilder(uri)
@@ -301,29 +301,15 @@ private BitbucketUser getUser(Optional<String> token)
301301
.timeout(DEFAULT_HTTP_TIMEOUT)
302302
.build();
303303

304-
String username;
304+
HttpResponse<InputStream> response;
305305
try {
306306
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-
});
307+
response = executeRequest(httpClient, request, r -> r);
318308
} catch (ScmBadRequestException e) {
319309
throw new ScmCommunicationException(e.getMessage(), e);
320310
}
321311

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-
}
312+
String username = getUsername(response);
327313

328314
try {
329315
List<BitbucketUser> users =
@@ -340,6 +326,26 @@ private BitbucketUser getUser(Optional<String> token)
340326
}
341327
}
342328

329+
private String getUsername(HttpResponse<InputStream> response)
330+
throws ScmCommunicationException, ScmUnauthorizedException {
331+
try {
332+
// Try to get the username from the response header.
333+
if (response.headers().firstValue(USERNAME_HEADER).isPresent()) {
334+
return response.headers().firstValue(USERNAME_HEADER).get();
335+
} else {
336+
String result =
337+
CharStreams.toString(new InputStreamReader(response.body(), Charsets.UTF_8));
338+
// Convert the response data to the Bitbucket Server info object.
339+
OBJECT_MAPPER.readValue(result, BitbucketApplicationProperties.class);
340+
// Throw the unauthorized exception if the response contains the Bitbucket info.
341+
throw buildScmUnauthorizedException();
342+
}
343+
} catch (IOException e) {
344+
// The response does not contain the Bitbucket Server info
345+
throw new ScmCommunicationException("Bad request");
346+
}
347+
}
348+
343349
private <T> List<T> doGetItems(Optional<String> token, Class<T> tClass, String api, String filter)
344350
throws ScmUnauthorizedException, ScmCommunicationException, ScmBadRequestException,
345351
ScmItemNotFoundException {
@@ -377,10 +383,10 @@ private <T> Page<T> doGetPage(
377383
return executeRequest(
378384
httpClient,
379385
request,
380-
inputStream -> {
386+
response -> {
381387
try {
382388
String result =
383-
CharStreams.toString(new InputStreamReader(inputStream, Charsets.UTF_8));
389+
CharStreams.toString(new InputStreamReader(response.body(), Charsets.UTF_8));
384390
return OM.readValue(result, typeReference);
385391
} catch (IOException e) {
386392
throw new UncheckedIOException(e);
@@ -389,15 +395,17 @@ private <T> Page<T> doGetPage(
389395
}
390396

391397
private <T> T executeRequest(
392-
HttpClient httpClient, HttpRequest request, Function<InputStream, T> bodyConverter)
398+
HttpClient httpClient,
399+
HttpRequest request,
400+
Function<HttpResponse<InputStream>, T> bodyConverter)
393401
throws ScmBadRequestException, ScmItemNotFoundException, ScmCommunicationException,
394402
ScmUnauthorizedException {
395403
try {
396404
HttpResponse<InputStream> response =
397405
httpClient.send(request, HttpResponse.BodyHandlers.ofInputStream());
398406
LOG.trace("executeRequest={} response {}", request, response.statusCode());
399407
if (response.statusCode() == 200) {
400-
return bodyConverter.apply(response.body());
408+
return bodyConverter.apply(response);
401409
} else if (response.statusCode() == 204) {
402410
return null;
403411
} else {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,83 @@
1+
/*
2+
* Copyright (c) 2012-2025 Red Hat, Inc.
3+
* This program and the accompanying materials are made
4+
* available under the terms of the Eclipse Public License 2.0
5+
* which is available at https://www.eclipse.org/legal/epl-2.0/
6+
*
7+
* SPDX-License-Identifier: EPL-2.0
8+
*
9+
* Contributors:
10+
* Red Hat, Inc. - initial API and implementation
11+
*/
12+
package org.eclipse.che.api.factory.server.bitbucket.server;
13+
14+
import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
15+
import java.util.Objects;
16+
17+
@JsonIgnoreProperties(ignoreUnknown = true)
18+
public class BitbucketApplicationProperties {
19+
20+
private String version;
21+
private String buildNumber;
22+
private String buildDate;
23+
private String displayName;
24+
25+
public BitbucketApplicationProperties() {}
26+
27+
public BitbucketApplicationProperties(
28+
String version, String buildNumber, String buildDate, String displayName) {
29+
this.version = version;
30+
this.buildNumber = buildNumber;
31+
this.buildDate = buildDate;
32+
this.displayName = displayName;
33+
}
34+
35+
public String getVersion() {
36+
return version;
37+
}
38+
39+
public String getBuildNumber() {
40+
return buildNumber;
41+
}
42+
43+
public String getBuildDate() {
44+
return buildDate;
45+
}
46+
47+
public String getDisplayName() {
48+
return displayName;
49+
}
50+
51+
@Override
52+
public String toString() {
53+
return "BitbucketApplicationProperties{"
54+
+ "version='"
55+
+ version
56+
+ '\''
57+
+ ", buildNumber="
58+
+ buildNumber
59+
+ ", buildDate="
60+
+ buildDate
61+
+ ", displayName='"
62+
+ displayName
63+
+ '\''
64+
+ '}';
65+
}
66+
67+
@Override
68+
public boolean equals(Object o) {
69+
if (o == null || getClass() != o.getClass()) {
70+
return false;
71+
}
72+
BitbucketApplicationProperties that = (BitbucketApplicationProperties) o;
73+
return buildNumber == that.buildNumber
74+
&& buildDate == that.buildDate
75+
&& Objects.equals(version, that.version)
76+
&& Objects.equals(displayName, that.displayName);
77+
}
78+
79+
@Override
80+
public int hashCode() {
81+
return Objects.hash(version, buildNumber, buildDate, displayName);
82+
}
83+
}

wsmaster/che-core-api-factory-bitbucket-server/src/test/java/org/eclipse/che/api/factory/server/bitbucket/BitbucketServerURLParserTest.java

+50-7
Original file line numberDiff line numberDiff line change
@@ -117,8 +117,10 @@ public void shouldValidateUrlByApiRequest() {
117117
null, devfileFilenamesProvider, oAuthAPI, mock(PersonalAccessTokenManager.class));
118118
String url = wireMockServer.url("/users/user/repos/repo");
119119
stubFor(
120-
get(urlEqualTo("/plugins/servlet/applinks/whoami"))
121-
.willReturn(aResponse().withStatus(200)));
120+
get(urlEqualTo("/rest/api/1.0/application-properties"))
121+
.willReturn(
122+
aResponse()
123+
.withBodyFile("bitbucket/rest/api.1.0.application-properties/response.json")));
122124

123125
// when
124126
boolean result = bitbucketURLParser.isValid(url);
@@ -135,8 +137,46 @@ public void shouldValidateUrlByApiRequestButFailOnPatternCheck() {
135137
null, devfileFilenamesProvider, oAuthAPI, mock(PersonalAccessTokenManager.class));
136138
String url = wireMockServer.url("/user/repo");
137139
stubFor(
138-
get(urlEqualTo("/plugins/servlet/applinks/whoami"))
139-
.willReturn(aResponse().withStatus(200)));
140+
get(urlEqualTo("/rest/api/1.0/application-properties"))
141+
.willReturn(
142+
aResponse()
143+
.withBodyFile("bitbucket/rest/api.1.0.application-properties/response.json")));
144+
145+
// when
146+
boolean result = bitbucketURLParser.isValid(url);
147+
148+
// then
149+
assertFalse(result);
150+
}
151+
152+
@Test
153+
public void shouldNotValidateUrlByApiRequestWithBadRequest() {
154+
// given
155+
bitbucketURLParser =
156+
new BitbucketServerURLParser(
157+
null, devfileFilenamesProvider, oAuthAPI, mock(PersonalAccessTokenManager.class));
158+
String url = wireMockServer.url("/users/user/repos/repo");
159+
stubFor(
160+
get(urlEqualTo("/rest/api/1.0/application-properties"))
161+
.willReturn(aResponse().withStatus(400)));
162+
163+
// when
164+
boolean result = bitbucketURLParser.isValid(url);
165+
166+
// then
167+
assertFalse(result);
168+
}
169+
170+
@Test
171+
public void shouldNotValidateUrlByApiRequestWithEmptyData() {
172+
// given
173+
bitbucketURLParser =
174+
new BitbucketServerURLParser(
175+
null, devfileFilenamesProvider, oAuthAPI, mock(PersonalAccessTokenManager.class));
176+
String url = wireMockServer.url("/users/user/repos/repo");
177+
stubFor(
178+
get(urlEqualTo("/rest/api/1.0/application-properties"))
179+
.willReturn(aResponse().withBody("")));
140180

141181
// when
142182
boolean result = bitbucketURLParser.isValid(url);
@@ -146,12 +186,15 @@ public void shouldValidateUrlByApiRequestButFailOnPatternCheck() {
146186
}
147187

148188
@Test
149-
public void shouldNotValidateUrlByApiRequest() {
189+
public void shouldNotValidateUrlByApiRequestWithEmptyHeader() {
150190
// given
191+
bitbucketURLParser =
192+
new BitbucketServerURLParser(
193+
null, devfileFilenamesProvider, oAuthAPI, mock(PersonalAccessTokenManager.class));
151194
String url = wireMockServer.url("/users/user/repos/repo");
152195
stubFor(
153-
get(urlEqualTo("/plugins/servlet/applinks/whoami"))
154-
.willReturn(aResponse().withStatus(500)));
196+
get(urlEqualTo("/rest/api/1.0/application-properties"))
197+
.willReturn(aResponse().withStatus(200)));
155198

156199
// when
157200
boolean result = bitbucketURLParser.isValid(url);

wsmaster/che-core-api-factory-bitbucket-server/src/test/java/org/eclipse/che/api/factory/server/bitbucket/HttpBitbucketServerApiClientTest.java

+17-4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2012-2024 Red Hat, Inc.
2+
* Copyright (c) 2012-2025 Red Hat, Inc.
33
* This program and the accompanying materials are made
44
* available under the terms of the Eclipse Public License 2.0
55
* which is available at https://www.eclipse.org/legal/epl-2.0/
@@ -102,8 +102,8 @@ public String computeAuthorizationHeader(
102102
oAuthAPI,
103103
apiEndpoint);
104104
stubFor(
105-
get(urlEqualTo("/plugins/servlet/applinks/whoami"))
106-
.willReturn(aResponse().withBody("ksmster")));
105+
get(urlEqualTo("/rest/api/1.0/application-properties"))
106+
.willReturn(aResponse().withHeader("x-ausername", "ksmster")));
107107
}
108108

109109
@AfterMethod
@@ -354,7 +354,20 @@ public void shouldBeAbleToThrowScmUnauthorizedExceptionOnGetUser()
354354
throws ScmCommunicationException, ScmUnauthorizedException, ScmItemNotFoundException {
355355
// given
356356
stubFor(
357-
get(urlEqualTo("/plugins/servlet/applinks/whoami")).willReturn(aResponse().withBody("")));
357+
get(urlEqualTo("/rest/api/1.0/application-properties"))
358+
.willReturn(
359+
aResponse()
360+
.withBodyFile("bitbucket/rest/api.1.0.application-properties/response.json")));
361+
362+
// when
363+
bitbucketServer.getUser();
364+
}
365+
366+
@Test(expectedExceptions = ScmCommunicationException.class)
367+
public void shouldBeAbleToThrowScmCommunicationExceptionOnGetUser()
368+
throws ScmCommunicationException, ScmUnauthorizedException, ScmItemNotFoundException {
369+
// given
370+
stubFor(get(urlEqualTo("/rest/api/1.0/application-properties")).willReturn(aResponse()));
358371

359372
// when
360373
bitbucketServer.getUser();
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
{
2+
"version": "8.7.0",
3+
"buildNumber": 8007000,
4+
"buildDate": 1672975062662,
5+
"displayName": "Bitbucket"
6+
}

0 commit comments

Comments
 (0)