mirror of
https://repository.entgra.net/community/device-mgt-core.git
synced 2025-10-06 02:01:45 +00:00
Add scope publishing to sub tenants
This commit is contained in:
commit
e017425539
@ -84,6 +84,10 @@
|
|||||||
<artifactId>okhttp</artifactId>
|
<artifactId>okhttp</artifactId>
|
||||||
<scope>compile</scope>
|
<scope>compile</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.wso2.carbon</groupId>
|
||||||
|
<artifactId>org.wso2.carbon.user.api</artifactId>
|
||||||
|
</dependency>
|
||||||
</dependencies>
|
</dependencies>
|
||||||
|
|
||||||
<build>
|
<build>
|
||||||
@ -121,7 +125,10 @@
|
|||||||
org.wso2.carbon.apimgt.impl;version="${carbon.api.mgt.version.range}",
|
org.wso2.carbon.apimgt.impl;version="${carbon.api.mgt.version.range}",
|
||||||
org.wso2.carbon.apimgt.impl.utils;version="${carbon.api.mgt.version.range}",
|
org.wso2.carbon.apimgt.impl.utils;version="${carbon.api.mgt.version.range}",
|
||||||
org.wso2.carbon.apimgt.impl.internal;version="${carbon.api.mgt.version.range}",
|
org.wso2.carbon.apimgt.impl.internal;version="${carbon.api.mgt.version.range}",
|
||||||
org.json
|
org.json,
|
||||||
|
org.wso2.carbon.user.api,
|
||||||
|
org.wso2.carbon.context;version="4.6",
|
||||||
|
org.wso2.carbon.utils.*
|
||||||
</Import-Package>
|
</Import-Package>
|
||||||
</instructions>
|
</instructions>
|
||||||
</configuration>
|
</configuration>
|
||||||
|
|||||||
@ -33,8 +33,11 @@ import okhttp3.RequestBody;
|
|||||||
import okhttp3.Credentials;
|
import okhttp3.Credentials;
|
||||||
import org.apache.commons.logging.Log;
|
import org.apache.commons.logging.Log;
|
||||||
import org.apache.commons.logging.LogFactory;
|
import org.apache.commons.logging.LogFactory;
|
||||||
|
import org.wso2.carbon.apimgt.impl.APIConstants;
|
||||||
import org.wso2.carbon.apimgt.impl.APIManagerConfiguration;
|
import org.wso2.carbon.apimgt.impl.APIManagerConfiguration;
|
||||||
import org.wso2.carbon.apimgt.impl.internal.ServiceReferenceHolder;
|
import org.wso2.carbon.apimgt.impl.internal.ServiceReferenceHolder;
|
||||||
|
import org.wso2.carbon.context.PrivilegedCarbonContext;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
|
||||||
public class APIApplicationServicesImpl implements APIApplicationServices {
|
public class APIApplicationServicesImpl implements APIApplicationServices {
|
||||||
@ -48,12 +51,12 @@ public class APIApplicationServicesImpl implements APIApplicationServices {
|
|||||||
getAPIManagerConfigurationService().getAPIManagerConfiguration();
|
getAPIManagerConfigurationService().getAPIManagerConfiguration();
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public APIApplicationKey createAndRetrieveApplicationCredentials()
|
public APIApplicationKey createAndRetrieveApplicationCredentials() throws APIServicesException {
|
||||||
throws APIServicesException {
|
String tenantDomain = PrivilegedCarbonContext.getThreadLocalCarbonContext().getTenantDomain();
|
||||||
|
String serverUser = getScopePublishUserName(tenantDomain);
|
||||||
|
String serverPassword = getScopePublishUserPassword(tenantDomain);
|
||||||
|
|
||||||
String applicationEndpoint = config.getFirstProperty(Constants.DCR_END_POINT);
|
String applicationEndpoint = config.getFirstProperty(Constants.DCR_END_POINT);
|
||||||
String serverUser = config.getFirstProperty(Constants.SERVER_USER);
|
|
||||||
String serverPassword = config.getFirstProperty(Constants.SERVER_PASSWORD);
|
|
||||||
|
|
||||||
JSONObject jsonObject = new JSONObject();
|
JSONObject jsonObject = new JSONObject();
|
||||||
jsonObject.put("callbackUrl", Constants.EMPTY_STRING);
|
jsonObject.put("callbackUrl", Constants.EMPTY_STRING);
|
||||||
@ -68,9 +71,11 @@ public class APIApplicationServicesImpl implements APIApplicationServices {
|
|||||||
.addHeader(Constants.AUTHORIZATION_HEADER_NAME, Credentials.basic(serverUser, serverPassword))
|
.addHeader(Constants.AUTHORIZATION_HEADER_NAME, Credentials.basic(serverUser, serverPassword))
|
||||||
.post(requestBody)
|
.post(requestBody)
|
||||||
.build();
|
.build();
|
||||||
|
|
||||||
try {
|
try {
|
||||||
Response response = client.newCall(request).execute();
|
try (Response response = client.newCall(request).execute()) {
|
||||||
return gson.fromJson(response.body().string(), APIApplicationKey.class);
|
return gson.fromJson(response.body().string(), APIApplicationKey.class);
|
||||||
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
msg = "Error occurred while processing the response";
|
msg = "Error occurred while processing the response";
|
||||||
log.error(msg, e);
|
log.error(msg, e);
|
||||||
@ -82,8 +87,9 @@ public class APIApplicationServicesImpl implements APIApplicationServices {
|
|||||||
public AccessTokenInfo generateAccessTokenFromRegisteredApplication(String consumerKey, String consumerSecret)
|
public AccessTokenInfo generateAccessTokenFromRegisteredApplication(String consumerKey, String consumerSecret)
|
||||||
throws APIServicesException {
|
throws APIServicesException {
|
||||||
|
|
||||||
String userName = config.getFirstProperty(Constants.SERVER_USER);
|
String tenantDomain = PrivilegedCarbonContext.getThreadLocalCarbonContext().getTenantDomain();
|
||||||
String userPassword = config.getFirstProperty(Constants.SERVER_PASSWORD);
|
String userName = getScopePublishUserName(tenantDomain);
|
||||||
|
String userPassword = getScopePublishUserPassword(tenantDomain);
|
||||||
|
|
||||||
JSONObject params = new JSONObject();
|
JSONObject params = new JSONObject();
|
||||||
params.put(Constants.GRANT_TYPE_PARAM_NAME, Constants.PASSWORD_GRANT_TYPE);
|
params.put(Constants.GRANT_TYPE_PARAM_NAME, Constants.PASSWORD_GRANT_TYPE);
|
||||||
@ -94,9 +100,8 @@ public class APIApplicationServicesImpl implements APIApplicationServices {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public AccessTokenInfo generateAccessTokenFromRefreshToken(String refreshToken, String consumerKey, String consumerSecret)
|
public AccessTokenInfo generateAccessTokenFromRefreshToken(String refreshToken, String consumerKey,
|
||||||
throws APIServicesException {
|
String consumerSecret) throws APIServicesException {
|
||||||
|
|
||||||
JSONObject params = new JSONObject();
|
JSONObject params = new JSONObject();
|
||||||
params.put(Constants.GRANT_TYPE_PARAM_NAME, Constants.REFRESH_TOKEN_GRANT_TYPE);
|
params.put(Constants.GRANT_TYPE_PARAM_NAME, Constants.REFRESH_TOKEN_GRANT_TYPE);
|
||||||
params.put(Constants.REFRESH_TOKEN_GRANT_TYPE_PARAM_NAME, refreshToken);
|
params.put(Constants.REFRESH_TOKEN_GRANT_TYPE_PARAM_NAME, refreshToken);
|
||||||
@ -125,4 +130,20 @@ public class APIApplicationServicesImpl implements APIApplicationServices {
|
|||||||
throw new APIServicesException(e);
|
throw new APIServicesException(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private String getScopePublishUserName(String tenantDomain) {
|
||||||
|
if(APIConstants.SUPER_TENANT_DOMAIN.equals(tenantDomain)) {
|
||||||
|
return config.getFirstProperty(Constants.SERVER_USER);
|
||||||
|
} else {
|
||||||
|
return Constants.SCOPE_PUBLISH_RESERVED_USER_NAME + "@" + tenantDomain;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private String getScopePublishUserPassword(String tenantDomain) {
|
||||||
|
if(APIConstants.SUPER_TENANT_DOMAIN.equals(tenantDomain)) {
|
||||||
|
return config.getFirstProperty(Constants.SERVER_PASSWORD);
|
||||||
|
} else {
|
||||||
|
return Constants.SCOPE_PUBLISH_RESERVED_USER_PASSWORD;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -42,6 +42,9 @@ public interface PublisherRESTAPIServices {
|
|||||||
boolean updateSharedScope(APIApplicationKey apiApplicationKey, AccessTokenInfo accessTokenInfo, Scope scope)
|
boolean updateSharedScope(APIApplicationKey apiApplicationKey, AccessTokenInfo accessTokenInfo, Scope scope)
|
||||||
throws APIServicesException, BadRequestException, UnexpectedResponseException;
|
throws APIServicesException, BadRequestException, UnexpectedResponseException;
|
||||||
|
|
||||||
|
boolean deleteSharedScope(APIApplicationKey apiApplicationKey, AccessTokenInfo accessTokenInfo, Scope scope)
|
||||||
|
throws APIServicesException, BadRequestException, UnexpectedResponseException;
|
||||||
|
|
||||||
APIInfo getApi(APIApplicationKey apiApplicationKey, AccessTokenInfo accessTokenInfo, String apiUuid)
|
APIInfo getApi(APIApplicationKey apiApplicationKey, AccessTokenInfo accessTokenInfo, String apiUuid)
|
||||||
throws APIServicesException, BadRequestException, UnexpectedResponseException;
|
throws APIServicesException, BadRequestException, UnexpectedResponseException;
|
||||||
|
|
||||||
|
|||||||
@ -79,6 +79,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -123,6 +124,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
return false;
|
return false;
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -177,6 +179,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.message();
|
String msg = "Response : " + response.code() + response.message();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -231,6 +234,61 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
|
throw new UnexpectedResponseException(msg);
|
||||||
|
}
|
||||||
|
} catch (IOException e) {
|
||||||
|
String msg = "Error occurred while processing the response";
|
||||||
|
log.error(msg, e);
|
||||||
|
throw new APIServicesException(msg, e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean deleteSharedScope(APIApplicationKey apiApplicationKey, AccessTokenInfo accessTokenInfo, Scope scope)
|
||||||
|
throws APIServicesException, BadRequestException, UnexpectedResponseException {
|
||||||
|
String updateScopeUrl = endPointPrefix + Constants.SCOPE_API_ENDPOINT + scope.getId();
|
||||||
|
|
||||||
|
JSONArray bindings = new JSONArray();
|
||||||
|
if (scope.getBindings() != null) {
|
||||||
|
for (String str : scope.getBindings()) {
|
||||||
|
bindings.put(str);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
JSONObject payload = new JSONObject();
|
||||||
|
payload.put("name", (scope.getName() != null ? scope.getName() : ""));
|
||||||
|
payload.put("displayName", (scope.getDisplayName() != null ? scope.getDisplayName() : ""));
|
||||||
|
payload.put("description", (scope.getDescription() != null ? scope.getDescription() : ""));
|
||||||
|
payload.put("bindings", (bindings != null ? bindings : ""));
|
||||||
|
payload.put("usageCount", (scope.getUsageCount() != 0 ? scope.getUsageCount() : 0));
|
||||||
|
|
||||||
|
RequestBody requestBody = RequestBody.create(JSON, payload.toString());
|
||||||
|
Request request = new Request.Builder()
|
||||||
|
.url(updateScopeUrl)
|
||||||
|
.addHeader(Constants.AUTHORIZATION_HEADER_NAME, Constants.AUTHORIZATION_HEADER_PREFIX_BEARER
|
||||||
|
+ accessTokenInfo.getAccess_token())
|
||||||
|
.delete(requestBody)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
try {
|
||||||
|
Response response = client.newCall(request).execute();
|
||||||
|
if (HttpStatus.SC_OK == response.code()) {
|
||||||
|
return true;
|
||||||
|
} else if (HttpStatus.SC_UNAUTHORIZED == response.code()) {
|
||||||
|
APIApplicationServices apiApplicationServices = new APIApplicationServicesImpl();
|
||||||
|
AccessTokenInfo refreshedAccessToken = apiApplicationServices.
|
||||||
|
generateAccessTokenFromRefreshToken(accessTokenInfo.getRefresh_token(),
|
||||||
|
apiApplicationKey.getClientId(), apiApplicationKey.getClientSecret());
|
||||||
|
//TODO: max attempt count
|
||||||
|
return deleteSharedScope(apiApplicationKey, refreshedAccessToken, scope);
|
||||||
|
} else if (HttpStatus.SC_BAD_REQUEST == response.code()) {
|
||||||
|
String msg = "Bad Request, Invalid scope object";
|
||||||
|
log.error(msg);
|
||||||
|
throw new BadRequestException(msg);
|
||||||
|
} else {
|
||||||
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -269,6 +327,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -308,6 +367,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -447,6 +507,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response status : " + response.code() + " Response message : " + response.message();
|
String msg = "Response status : " + response.code() + " Response message : " + response.message();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -586,6 +647,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -632,6 +694,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -673,6 +736,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -720,6 +784,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -761,6 +826,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -804,6 +870,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -846,6 +913,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -889,6 +957,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -939,6 +1008,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -990,6 +1060,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -1031,6 +1102,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -1071,6 +1143,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -1111,6 +1184,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -1162,6 +1236,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -1208,6 +1283,7 @@ public class PublisherRESTAPIServicesImpl implements PublisherRESTAPIServices {
|
|||||||
throw new BadRequestException(msg);
|
throw new BadRequestException(msg);
|
||||||
} else {
|
} else {
|
||||||
String msg = "Response : " + response.code() + response.body();
|
String msg = "Response : " + response.code() + response.body();
|
||||||
|
log.error(msg);
|
||||||
throw new UnexpectedResponseException(msg);
|
throw new UnexpectedResponseException(msg);
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
|
|||||||
@ -65,6 +65,10 @@ public final class Constants {
|
|||||||
public static final String SCOPE_API_ENDPOINT = "/api/am/publisher/v2/scopes/";
|
public static final String SCOPE_API_ENDPOINT = "/api/am/publisher/v2/scopes/";
|
||||||
public static final String API_ENDPOINT = "/api/am/publisher/v2/apis/";
|
public static final String API_ENDPOINT = "/api/am/publisher/v2/apis/";
|
||||||
public static final String GET_ALL_APIS = "/api/am/publisher/v2/apis?limit=1000";
|
public static final String GET_ALL_APIS = "/api/am/publisher/v2/apis?limit=1000";
|
||||||
|
public static final String SCOPE_PUBLISH_RESERVED_USER_NAME = "scope_publish_reserved_user";
|
||||||
|
public static final String SCOPE_PUBLISH_RESERVED_USER_PASSWORD = "&gKfyE8E4rUY4Q";
|
||||||
|
public static final String ADMIN_ROLE_KEY = "admin";
|
||||||
|
public static final String PERM_SCOPE_MAPPING_META_KEY = "perm-scope-mapping";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@ -21,12 +21,16 @@ package io.entgra.device.mgt.core.apimgt.extension.rest.api.internal;
|
|||||||
import io.entgra.device.mgt.core.apimgt.extension.rest.api.APIApplicationServices;
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.APIApplicationServices;
|
||||||
import io.entgra.device.mgt.core.apimgt.extension.rest.api.PublisherRESTAPIServices;
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.PublisherRESTAPIServices;
|
||||||
import org.wso2.carbon.apimgt.impl.APIManagerConfigurationService;
|
import org.wso2.carbon.apimgt.impl.APIManagerConfigurationService;
|
||||||
|
import org.wso2.carbon.user.core.service.RealmService;
|
||||||
|
import org.wso2.carbon.user.core.tenant.TenantManager;
|
||||||
|
|
||||||
public class APIManagerServiceDataHolder {
|
public class APIManagerServiceDataHolder {
|
||||||
|
|
||||||
private APIApplicationServices apiApplicationServices;
|
private APIApplicationServices apiApplicationServices;
|
||||||
private APIManagerConfigurationService apiManagerConfigurationService;
|
private APIManagerConfigurationService apiManagerConfigurationService;
|
||||||
private PublisherRESTAPIServices publisherRESTAPIServices;
|
private PublisherRESTAPIServices publisherRESTAPIServices;
|
||||||
|
private RealmService realmService;
|
||||||
|
private TenantManager tenantManager;
|
||||||
|
|
||||||
private static APIManagerServiceDataHolder thisInstance = new APIManagerServiceDataHolder();
|
private static APIManagerServiceDataHolder thisInstance = new APIManagerServiceDataHolder();
|
||||||
|
|
||||||
@ -63,4 +67,27 @@ public class APIManagerServiceDataHolder {
|
|||||||
public void setPublisherRESTAPIServices(PublisherRESTAPIServices publisherRESTAPIServices) {
|
public void setPublisherRESTAPIServices(PublisherRESTAPIServices publisherRESTAPIServices) {
|
||||||
this.publisherRESTAPIServices = publisherRESTAPIServices;
|
this.publisherRESTAPIServices = publisherRESTAPIServices;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public RealmService getRealmService() {
|
||||||
|
if (realmService == null) {
|
||||||
|
throw new IllegalStateException("Realm service is not initialized properly");
|
||||||
|
}
|
||||||
|
return realmService;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setRealmService(RealmService realmService) {
|
||||||
|
this.realmService = realmService;
|
||||||
|
this.setTenantManager(realmService);
|
||||||
|
}
|
||||||
|
|
||||||
|
public TenantManager getTenantManager() {
|
||||||
|
return tenantManager;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setTenantManager(RealmService realmService) {
|
||||||
|
if (realmService == null) {
|
||||||
|
throw new IllegalStateException("Realm service is not initialized properly");
|
||||||
|
}
|
||||||
|
this.tenantManager = realmService.getTenantManager();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -0,0 +1,91 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2018 - 2024, Entgra (Pvt) Ltd. (http://www.entgra.io) All Rights Reserved.
|
||||||
|
*
|
||||||
|
* Entgra (Pvt) Ltd. licenses this file to you under the Apache License,
|
||||||
|
* Version 2.0 (the "License"); you may not use this file except
|
||||||
|
* in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing,
|
||||||
|
* software distributed under the License is distributed on an
|
||||||
|
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
||||||
|
* KIND, either express or implied. See the License for the
|
||||||
|
* specific language governing permissions and limitations
|
||||||
|
* under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package io.entgra.device.mgt.core.apimgt.extension.rest.api.util;
|
||||||
|
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.constants.Constants;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.exceptions.APIServicesException;
|
||||||
|
import org.apache.commons.logging.Log;
|
||||||
|
import org.apache.commons.logging.LogFactory;
|
||||||
|
import org.wso2.carbon.context.PrivilegedCarbonContext;
|
||||||
|
import org.wso2.carbon.user.api.UserStoreException;
|
||||||
|
import org.wso2.carbon.user.api.UserStoreManager;
|
||||||
|
import org.wso2.carbon.utils.multitenancy.MultitenantConstants;
|
||||||
|
import org.wso2.carbon.utils.multitenancy.MultitenantUtils;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class contains utility methods needed for API publishing
|
||||||
|
*/
|
||||||
|
public class APIPublisherUtils {
|
||||||
|
private static final Log log = LogFactory.getLog(APIPublisherUtils.class);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method will create the temporary user created to publish scopes to the sub tenant space.
|
||||||
|
* @param tenantDomain sub tenant domain from which the user will be created
|
||||||
|
* @throws APIServicesException if the user was unable to be created
|
||||||
|
*/
|
||||||
|
public static void createScopePublishUserIfNotExists(String tenantDomain) throws APIServicesException {
|
||||||
|
if (!MultitenantConstants.SUPER_TENANT_DOMAIN_NAME.equals(tenantDomain)) {
|
||||||
|
try {
|
||||||
|
UserStoreManager userStoreManager =
|
||||||
|
PrivilegedCarbonContext.getThreadLocalCarbonContext().getUserRealm().getUserStoreManager();
|
||||||
|
if (!userStoreManager.isExistingUser(MultitenantUtils.getTenantAwareUsername(Constants.SCOPE_PUBLISH_RESERVED_USER_NAME))) {
|
||||||
|
if (log.isDebugEnabled()) {
|
||||||
|
log.debug("Creating scope publish user '" + Constants.SCOPE_PUBLISH_RESERVED_USER_NAME + "' in '" +
|
||||||
|
tenantDomain + "' tenant domain.");
|
||||||
|
}
|
||||||
|
String[] roles = {Constants.ADMIN_ROLE_KEY};
|
||||||
|
userStoreManager.addUser(
|
||||||
|
MultitenantUtils.getTenantAwareUsername(Constants.SCOPE_PUBLISH_RESERVED_USER_NAME),
|
||||||
|
Constants.SCOPE_PUBLISH_RESERVED_USER_PASSWORD,
|
||||||
|
roles,
|
||||||
|
null,
|
||||||
|
""
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} catch (UserStoreException e) {
|
||||||
|
String msg = "Error occurred while creating scope publishing user in tenant: '" + tenantDomain + "'.";
|
||||||
|
log.error(msg);
|
||||||
|
throw new APIServicesException(msg, e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method will delete the temporary user created to publish scopes to the sub tenant space.
|
||||||
|
* @param tenantDomain sub tenant domain from which the scope publish user will be removed from
|
||||||
|
*/
|
||||||
|
public static void removeScopePublishUserIfExists(String tenantDomain) {
|
||||||
|
if (!MultitenantConstants.SUPER_TENANT_DOMAIN_NAME.equals(tenantDomain)) {
|
||||||
|
try {
|
||||||
|
UserStoreManager userStoreManager =
|
||||||
|
PrivilegedCarbonContext.getThreadLocalCarbonContext().getUserRealm().getUserStoreManager();
|
||||||
|
if (userStoreManager.isExistingUser(MultitenantUtils.getTenantAwareUsername(Constants.SCOPE_PUBLISH_RESERVED_USER_NAME))) {
|
||||||
|
if (log.isDebugEnabled()) {
|
||||||
|
log.debug("Deleting scope publish user '" + Constants.SCOPE_PUBLISH_RESERVED_USER_NAME + "' from '" +
|
||||||
|
tenantDomain + "' tenant domain.");
|
||||||
|
}
|
||||||
|
userStoreManager.deleteUser(MultitenantUtils.getTenantAwareUsername(Constants.SCOPE_PUBLISH_RESERVED_USER_NAME));
|
||||||
|
}
|
||||||
|
} catch(UserStoreException e){
|
||||||
|
String msg = "Error occurred while deleting scope publishing user from tenant: '" + tenantDomain + "'.";
|
||||||
|
log.error(msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -39,7 +39,7 @@ public interface APIPublisherService {
|
|||||||
/**
|
/**
|
||||||
* Add default scopes defined in the cdm-config.xml
|
* Add default scopes defined in the cdm-config.xml
|
||||||
*/
|
*/
|
||||||
void addDefaultScopesIfNotExist();
|
void addDefaultScopesIfNotExist() throws APIManagerPublisherException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* If the permissions are in the permission list, identify the relevant scopes of the supplied permission list
|
* If the permissions are in the permission list, identify the relevant scopes of the supplied permission list
|
||||||
|
|||||||
@ -19,16 +19,23 @@ package io.entgra.device.mgt.core.apimgt.webapp.publisher;
|
|||||||
|
|
||||||
import com.google.gson.Gson;
|
import com.google.gson.Gson;
|
||||||
import io.entgra.device.mgt.core.apimgt.extension.rest.api.APIApplicationServices;
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.APIApplicationServices;
|
||||||
import io.entgra.device.mgt.core.apimgt.extension.rest.api.APIApplicationServicesImpl;
|
|
||||||
import io.entgra.device.mgt.core.apimgt.extension.rest.api.PublisherRESTAPIServices;
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.PublisherRESTAPIServices;
|
||||||
import io.entgra.device.mgt.core.apimgt.extension.rest.api.PublisherRESTAPIServicesImpl;
|
|
||||||
import io.entgra.device.mgt.core.apimgt.extension.rest.api.constants.Constants;
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.constants.Constants;
|
||||||
import io.entgra.device.mgt.core.apimgt.extension.rest.api.dto.APIApplicationKey;
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.dto.APIApplicationKey;
|
||||||
import io.entgra.device.mgt.core.apimgt.extension.rest.api.dto.APIInfo.*;
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.dto.APIInfo.APIInfo;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.dto.APIInfo.APIRevision;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.dto.APIInfo.APIRevisionDeployment;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.dto.APIInfo.CORSConfiguration;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.dto.APIInfo.Documentation;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.dto.APIInfo.Mediation;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.dto.APIInfo.MediationPolicy;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.dto.APIInfo.Operations;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.dto.APIInfo.Scope;
|
||||||
import io.entgra.device.mgt.core.apimgt.extension.rest.api.dto.AccessTokenInfo;
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.dto.AccessTokenInfo;
|
||||||
import io.entgra.device.mgt.core.apimgt.extension.rest.api.exceptions.APIServicesException;
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.exceptions.APIServicesException;
|
||||||
import io.entgra.device.mgt.core.apimgt.extension.rest.api.exceptions.BadRequestException;
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.exceptions.BadRequestException;
|
||||||
import io.entgra.device.mgt.core.apimgt.extension.rest.api.exceptions.UnexpectedResponseException;
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.exceptions.UnexpectedResponseException;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.util.APIPublisherUtils;
|
||||||
import io.entgra.device.mgt.core.apimgt.webapp.publisher.config.WebappPublisherConfig;
|
import io.entgra.device.mgt.core.apimgt.webapp.publisher.config.WebappPublisherConfig;
|
||||||
import io.entgra.device.mgt.core.apimgt.webapp.publisher.dto.ApiScope;
|
import io.entgra.device.mgt.core.apimgt.webapp.publisher.dto.ApiScope;
|
||||||
import io.entgra.device.mgt.core.apimgt.webapp.publisher.dto.ApiUriTemplate;
|
import io.entgra.device.mgt.core.apimgt.webapp.publisher.dto.ApiUriTemplate;
|
||||||
@ -60,6 +67,7 @@ import org.wso2.carbon.user.core.tenant.Tenant;
|
|||||||
import org.wso2.carbon.user.core.tenant.TenantSearchResult;
|
import org.wso2.carbon.user.core.tenant.TenantSearchResult;
|
||||||
import org.wso2.carbon.utils.CarbonUtils;
|
import org.wso2.carbon.utils.CarbonUtils;
|
||||||
import org.wso2.carbon.utils.multitenancy.MultitenantConstants;
|
import org.wso2.carbon.utils.multitenancy.MultitenantConstants;
|
||||||
|
import org.wso2.carbon.utils.multitenancy.MultitenantUtils;
|
||||||
|
|
||||||
import java.io.BufferedReader;
|
import java.io.BufferedReader;
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
@ -111,15 +119,6 @@ public class APIPublisherServiceImpl implements APIPublisherService {
|
|||||||
PublisherRESTAPIServices publisherRESTAPIServices = APIPublisherDataHolder.getInstance().getPublisherRESTAPIServices();
|
PublisherRESTAPIServices publisherRESTAPIServices = APIPublisherDataHolder.getInstance().getPublisherRESTAPIServices();
|
||||||
APIApplicationKey apiApplicationKey;
|
APIApplicationKey apiApplicationKey;
|
||||||
AccessTokenInfo accessTokenInfo;
|
AccessTokenInfo accessTokenInfo;
|
||||||
try {
|
|
||||||
apiApplicationKey = apiApplicationServices.createAndRetrieveApplicationCredentials();
|
|
||||||
accessTokenInfo = apiApplicationServices.generateAccessTokenFromRegisteredApplication(
|
|
||||||
apiApplicationKey.getClientId(), apiApplicationKey.getClientSecret());
|
|
||||||
} catch (APIServicesException e) {
|
|
||||||
String errorMsg = "Error occurred while generating the API application";
|
|
||||||
log.error(errorMsg, e);
|
|
||||||
throw new APIManagerPublisherException(e);
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
boolean tenantFound = false;
|
boolean tenantFound = false;
|
||||||
@ -154,6 +153,17 @@ public class APIPublisherServiceImpl implements APIPublisherService {
|
|||||||
PrivilegedCarbonContext.getThreadLocalCarbonContext().setUsername(apiConfig.getOwner());
|
PrivilegedCarbonContext.getThreadLocalCarbonContext().setUsername(apiConfig.getOwner());
|
||||||
int tenantId = PrivilegedCarbonContext.getThreadLocalCarbonContext().getTenantId();
|
int tenantId = PrivilegedCarbonContext.getThreadLocalCarbonContext().getTenantId();
|
||||||
|
|
||||||
|
try {
|
||||||
|
APIPublisherUtils.createScopePublishUserIfNotExists(tenantDomain);
|
||||||
|
apiApplicationKey = apiApplicationServices.createAndRetrieveApplicationCredentials();
|
||||||
|
accessTokenInfo = apiApplicationServices.generateAccessTokenFromRegisteredApplication(
|
||||||
|
apiApplicationKey.getClientId(), apiApplicationKey.getClientSecret());
|
||||||
|
} catch (APIServicesException e) {
|
||||||
|
String errorMsg = "Error occurred while generating the API application";
|
||||||
|
log.error(errorMsg, e);
|
||||||
|
throw new APIManagerPublisherException(e);
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
apiConfig.setOwner(APIUtil.getTenantAdminUserName(tenantDomain));
|
apiConfig.setOwner(APIUtil.getTenantAdminUserName(tenantDomain));
|
||||||
apiConfig.setTenantDomain(tenantDomain);
|
apiConfig.setTenantDomain(tenantDomain);
|
||||||
@ -428,6 +438,7 @@ public class APIPublisherServiceImpl implements APIPublisherService {
|
|||||||
log.error(msg, e);
|
log.error(msg, e);
|
||||||
throw new APIManagerPublisherException(e);
|
throw new APIManagerPublisherException(e);
|
||||||
} finally {
|
} finally {
|
||||||
|
APIPublisherUtils.removeScopePublishUserIfExists(tenantDomain);
|
||||||
PrivilegedCarbonContext.endTenantFlow();
|
PrivilegedCarbonContext.endTenantFlow();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -439,13 +450,23 @@ public class APIPublisherServiceImpl implements APIPublisherService {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void addDefaultScopesIfNotExist() {
|
@Override
|
||||||
|
public void addDefaultScopesIfNotExist() throws APIManagerPublisherException {
|
||||||
|
WebappPublisherConfig config = WebappPublisherConfig.getInstance();
|
||||||
|
List<String> tenants = new ArrayList<>(Collections.singletonList(APIConstants.SUPER_TENANT_DOMAIN));
|
||||||
|
tenants.addAll(config.getTenants().getTenant());
|
||||||
|
|
||||||
DeviceManagementConfig deviceManagementConfig = DeviceConfigurationManager.getInstance().getDeviceManagementConfig();
|
DeviceManagementConfig deviceManagementConfig = DeviceConfigurationManager.getInstance().getDeviceManagementConfig();
|
||||||
DefaultPermissions defaultPermissions = deviceManagementConfig.getDefaultPermissions();
|
DefaultPermissions defaultPermissions = deviceManagementConfig.getDefaultPermissions();
|
||||||
APIApplicationServices apiApplicationServices = APIPublisherDataHolder.getInstance().getApiApplicationServices();
|
APIApplicationServices apiApplicationServices = APIPublisherDataHolder.getInstance().getApiApplicationServices();
|
||||||
PublisherRESTAPIServices publisherRESTAPIServices = APIPublisherDataHolder.getInstance().getPublisherRESTAPIServices();
|
PublisherRESTAPIServices publisherRESTAPIServices = APIPublisherDataHolder.getInstance().getPublisherRESTAPIServices();
|
||||||
|
|
||||||
|
for (String tenantDomain : tenants) {
|
||||||
try {
|
try {
|
||||||
|
PrivilegedCarbonContext.startTenantFlow();
|
||||||
|
PrivilegedCarbonContext.getThreadLocalCarbonContext().setTenantDomain(tenantDomain, true);
|
||||||
|
|
||||||
|
APIPublisherUtils.createScopePublishUserIfNotExists(tenantDomain);
|
||||||
APIApplicationKey apiApplicationKey =
|
APIApplicationKey apiApplicationKey =
|
||||||
apiApplicationServices.createAndRetrieveApplicationCredentials();
|
apiApplicationServices.createAndRetrieveApplicationCredentials();
|
||||||
AccessTokenInfo accessTokenInfo =
|
AccessTokenInfo accessTokenInfo =
|
||||||
@ -453,7 +474,7 @@ public class APIPublisherServiceImpl implements APIPublisherService {
|
|||||||
apiApplicationKey.getClientId(), apiApplicationKey.getClientSecret());
|
apiApplicationKey.getClientId(), apiApplicationKey.getClientSecret());
|
||||||
|
|
||||||
Scope scope = new Scope();
|
Scope scope = new Scope();
|
||||||
for (DefaultPermission defaultPermission: defaultPermissions.getDefaultPermissions()) {
|
for (DefaultPermission defaultPermission : defaultPermissions.getDefaultPermissions()) {
|
||||||
if (!publisherRESTAPIServices.isSharedScopeNameExists(apiApplicationKey, accessTokenInfo,
|
if (!publisherRESTAPIServices.isSharedScopeNameExists(apiApplicationKey, accessTokenInfo,
|
||||||
defaultPermission.getScopeMapping().getKey())) {
|
defaultPermission.getScopeMapping().getKey())) {
|
||||||
ScopeMapping scopeMapping = defaultPermission.getScopeMapping();
|
ScopeMapping scopeMapping = defaultPermission.getScopeMapping();
|
||||||
@ -469,7 +490,13 @@ public class APIPublisherServiceImpl implements APIPublisherService {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (BadRequestException | UnexpectedResponseException | APIServicesException e) {
|
} catch (BadRequestException | UnexpectedResponseException | APIServicesException e) {
|
||||||
log.error("Error occurred while adding default scopes");
|
String errorMsg = "Error occurred while adding default scopes";
|
||||||
|
log.error(errorMsg, e);
|
||||||
|
throw new APIManagerPublisherException(e);
|
||||||
|
} finally {
|
||||||
|
APIPublisherUtils.removeScopePublishUserIfExists(tenantDomain);
|
||||||
|
PrivilegedCarbonContext.endTenantFlow();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -487,7 +514,15 @@ public class APIPublisherServiceImpl implements APIPublisherService {
|
|||||||
|
|
||||||
APIApplicationKey apiApplicationKey;
|
APIApplicationKey apiApplicationKey;
|
||||||
AccessTokenInfo accessTokenInfo;
|
AccessTokenInfo accessTokenInfo;
|
||||||
|
UserStoreManager userStoreManager;
|
||||||
|
String fileName = null;
|
||||||
|
|
||||||
|
for (String tenantDomain : tenants) {
|
||||||
try {
|
try {
|
||||||
|
PrivilegedCarbonContext.startTenantFlow();
|
||||||
|
PrivilegedCarbonContext.getThreadLocalCarbonContext().setTenantDomain(tenantDomain, true);
|
||||||
|
try {
|
||||||
|
APIPublisherUtils.createScopePublishUserIfNotExists(tenantDomain);
|
||||||
apiApplicationKey = apiApplicationServices.createAndRetrieveApplicationCredentials();
|
apiApplicationKey = apiApplicationServices.createAndRetrieveApplicationCredentials();
|
||||||
accessTokenInfo = apiApplicationServices.generateAccessTokenFromRegisteredApplication(
|
accessTokenInfo = apiApplicationServices.generateAccessTokenFromRegisteredApplication(
|
||||||
apiApplicationKey.getClientId(), apiApplicationKey.getClientSecret());
|
apiApplicationKey.getClientId(), apiApplicationKey.getClientSecret());
|
||||||
@ -496,15 +531,9 @@ public class APIPublisherServiceImpl implements APIPublisherService {
|
|||||||
log.error(errorMsg, e);
|
log.error(errorMsg, e);
|
||||||
throw new APIManagerPublisherException(e);
|
throw new APIManagerPublisherException(e);
|
||||||
}
|
}
|
||||||
UserStoreManager userStoreManager;
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
for (String tenantDomain : tenants) {
|
fileName =
|
||||||
PrivilegedCarbonContext.startTenantFlow();
|
|
||||||
PrivilegedCarbonContext.getThreadLocalCarbonContext().setTenantDomain(tenantDomain, true);
|
|
||||||
|
|
||||||
try {
|
|
||||||
String fileName =
|
|
||||||
CarbonUtils.getCarbonConfigDirPath() + File.separator + "etc"
|
CarbonUtils.getCarbonConfigDirPath() + File.separator + "etc"
|
||||||
+ File.separator + tenantDomain + ".csv";
|
+ File.separator + tenantDomain + ".csv";
|
||||||
try {
|
try {
|
||||||
@ -594,9 +623,10 @@ public class APIPublisherServiceImpl implements APIPublisherService {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (IOException | DirectoryIteratorException ex) {
|
} catch (IOException | DirectoryIteratorException e) {
|
||||||
log.error("failed to read scopes from file.", ex);
|
String errorMsg = "Failed to read scopes from file: '" + fileName + "'.";
|
||||||
}
|
log.error(errorMsg, e);
|
||||||
|
throw new APIManagerPublisherException(e);
|
||||||
}
|
}
|
||||||
} catch (APIServicesException e) {
|
} catch (APIServicesException e) {
|
||||||
String errorMsg = "Error while processing Publisher REST API response";
|
String errorMsg = "Error while processing Publisher REST API response";
|
||||||
@ -611,17 +641,21 @@ public class APIPublisherServiceImpl implements APIPublisherService {
|
|||||||
log.error(errorMsg, e);
|
log.error(errorMsg, e);
|
||||||
throw new APIManagerPublisherException(e);
|
throw new APIManagerPublisherException(e);
|
||||||
} finally {
|
} finally {
|
||||||
|
APIPublisherUtils.removeScopePublishUserIfExists(tenantDomain);
|
||||||
PrivilegedCarbonContext.endTenantFlow();
|
PrivilegedCarbonContext.endTenantFlow();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void updateScopeRoleMapping(String roleName, String[] permissions, String[] removedPermissions) throws APIManagerPublisherException {
|
public void updateScopeRoleMapping(String roleName, String[] permissions, String[] removedPermissions) throws APIManagerPublisherException {
|
||||||
|
String tenantDomain = PrivilegedCarbonContext.getThreadLocalCarbonContext().getTenantDomain();
|
||||||
APIApplicationServices apiApplicationServices = APIPublisherDataHolder.getInstance().getApiApplicationServices();
|
APIApplicationServices apiApplicationServices = APIPublisherDataHolder.getInstance().getApiApplicationServices();
|
||||||
PublisherRESTAPIServices publisherRESTAPIServices = APIPublisherDataHolder.getInstance().getPublisherRESTAPIServices();
|
PublisherRESTAPIServices publisherRESTAPIServices = APIPublisherDataHolder.getInstance().getPublisherRESTAPIServices();
|
||||||
APIApplicationKey apiApplicationKey;
|
APIApplicationKey apiApplicationKey;
|
||||||
AccessTokenInfo accessTokenInfo;
|
AccessTokenInfo accessTokenInfo;
|
||||||
try {
|
try {
|
||||||
|
APIPublisherUtils.createScopePublishUserIfNotExists(tenantDomain);
|
||||||
apiApplicationKey = apiApplicationServices.createAndRetrieveApplicationCredentials();
|
apiApplicationKey = apiApplicationServices.createAndRetrieveApplicationCredentials();
|
||||||
accessTokenInfo = apiApplicationServices.generateAccessTokenFromRegisteredApplication(
|
accessTokenInfo = apiApplicationServices.generateAccessTokenFromRegisteredApplication(
|
||||||
apiApplicationKey.getClientId(), apiApplicationKey.getClientSecret());
|
apiApplicationKey.getClientId(), apiApplicationKey.getClientSecret());
|
||||||
@ -662,6 +696,8 @@ public class APIPublisherServiceImpl implements APIPublisherService {
|
|||||||
String errorMsg = "Unexpected response from the server";
|
String errorMsg = "Unexpected response from the server";
|
||||||
log.error(errorMsg, e);
|
log.error(errorMsg, e);
|
||||||
throw new APIManagerPublisherException(errorMsg, e);
|
throw new APIManagerPublisherException(errorMsg, e);
|
||||||
|
} finally {
|
||||||
|
APIPublisherUtils.removeScopePublishUserIfExists(tenantDomain);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -419,7 +419,8 @@ public class UserManagementServiceImpl implements UserManagementService {
|
|||||||
userList = new ArrayList<>(users.size());
|
userList = new ArrayList<>(users.size());
|
||||||
BasicUserInfo user;
|
BasicUserInfo user;
|
||||||
for (String username : users) {
|
for (String username : users) {
|
||||||
if (Constants.APIM_RESERVED_USER.equals(username) || Constants.RESERVED_USER.equals(username)) {
|
if (Constants.APIM_RESERVED_USER.equals(username) || Constants.RESERVED_USER.equals(username) ||
|
||||||
|
Constants.SCOPE_PUBLISH_RESERVED_USER.equals(username)) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
user = getBasicUserInfo(username);
|
user = getBasicUserInfo(username);
|
||||||
@ -485,6 +486,7 @@ public class UserManagementServiceImpl implements UserManagementService {
|
|||||||
if (commonUsers != null) {
|
if (commonUsers != null) {
|
||||||
commonUsers.remove(Constants.APIM_RESERVED_USER);
|
commonUsers.remove(Constants.APIM_RESERVED_USER);
|
||||||
commonUsers.remove(Constants.RESERVED_USER);
|
commonUsers.remove(Constants.RESERVED_USER);
|
||||||
|
commonUsers.remove(Constants.SCOPE_PUBLISH_RESERVED_USER);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!skipSearch(commonUsers) && StringUtils.isNotEmpty(firstName)) {
|
if (!skipSearch(commonUsers) && StringUtils.isNotEmpty(firstName)) {
|
||||||
@ -660,7 +662,8 @@ public class UserManagementServiceImpl implements UserManagementService {
|
|||||||
userList = new ArrayList<>();
|
userList = new ArrayList<>();
|
||||||
UserInfo user;
|
UserInfo user;
|
||||||
for (String username : users) {
|
for (String username : users) {
|
||||||
if (Constants.APIM_RESERVED_USER.equals(username) || Constants.RESERVED_USER.equals(username)) {
|
if (Constants.APIM_RESERVED_USER.equals(username) || Constants.RESERVED_USER.equals(username) ||
|
||||||
|
Constants.SCOPE_PUBLISH_RESERVED_USER.equals(username)) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
user = new UserInfo();
|
user = new UserInfo();
|
||||||
|
|||||||
@ -31,6 +31,7 @@ public class Constants {
|
|||||||
public static final String USER_CLAIM_DEVICES = "http://wso2.org/claims/devices";
|
public static final String USER_CLAIM_DEVICES = "http://wso2.org/claims/devices";
|
||||||
public static final String PRIMARY_USER_STORE = "PRIMARY";
|
public static final String PRIMARY_USER_STORE = "PRIMARY";
|
||||||
public static final String APIM_RESERVED_USER = "apim_reserved_user";
|
public static final String APIM_RESERVED_USER = "apim_reserved_user";
|
||||||
|
public static final String SCOPE_PUBLISH_RESERVED_USER = "scope_publish_reserved_user";
|
||||||
public static final String RESERVED_USER = "reserved_user";
|
public static final String RESERVED_USER = "reserved_user";
|
||||||
public static final String DEFAULT_STREAM_VERSION = "1.0.0";
|
public static final String DEFAULT_STREAM_VERSION = "1.0.0";
|
||||||
public static final String SCOPE = "scope";
|
public static final String SCOPE = "scope";
|
||||||
|
|||||||
@ -374,6 +374,10 @@
|
|||||||
<version>2.3.1.wso2v1</version>
|
<version>2.3.1.wso2v1</version>
|
||||||
<scope>compile</scope>
|
<scope>compile</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.entgra.device.mgt.core</groupId>
|
||||||
|
<artifactId>io.entgra.device.mgt.core.apimgt.extension.rest.api</artifactId>
|
||||||
|
</dependency>
|
||||||
</dependencies>
|
</dependencies>
|
||||||
|
|
||||||
</project>
|
</project>
|
||||||
|
|||||||
@ -18,6 +18,8 @@
|
|||||||
|
|
||||||
package io.entgra.device.mgt.core.device.mgt.core.internal;
|
package io.entgra.device.mgt.core.device.mgt.core.internal;
|
||||||
|
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.APIApplicationServices;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.PublisherRESTAPIServices;
|
||||||
import io.entgra.device.mgt.core.device.mgt.common.metadata.mgt.DeviceStatusManagementService;
|
import io.entgra.device.mgt.core.device.mgt.common.metadata.mgt.DeviceStatusManagementService;
|
||||||
import io.entgra.device.mgt.core.server.bootup.heartbeat.beacon.service.HeartBeatManagementService;
|
import io.entgra.device.mgt.core.server.bootup.heartbeat.beacon.service.HeartBeatManagementService;
|
||||||
import org.wso2.carbon.context.PrivilegedCarbonContext;
|
import org.wso2.carbon.context.PrivilegedCarbonContext;
|
||||||
@ -93,8 +95,9 @@ public class DeviceManagementDataHolder {
|
|||||||
private MetadataManagementService metadataManagementService;
|
private MetadataManagementService metadataManagementService;
|
||||||
private WhiteLabelManagementService whiteLabelManagementService;
|
private WhiteLabelManagementService whiteLabelManagementService;
|
||||||
private TraccarManagementService traccarManagementService;
|
private TraccarManagementService traccarManagementService;
|
||||||
|
|
||||||
private DeviceStatusManagementService deviceStatusManagementService;
|
private DeviceStatusManagementService deviceStatusManagementService;
|
||||||
|
private APIApplicationServices apiApplicationServices;
|
||||||
|
private PublisherRESTAPIServices publisherRESTAPIServices;
|
||||||
|
|
||||||
private final Map<DeviceType, DeviceStatusTaskPluginConfig> deviceStatusTaskPluginConfigs = Collections.synchronizedMap(
|
private final Map<DeviceType, DeviceStatusTaskPluginConfig> deviceStatusTaskPluginConfigs = Collections.synchronizedMap(
|
||||||
new HashMap<>());
|
new HashMap<>());
|
||||||
@ -410,4 +413,38 @@ public class DeviceManagementDataHolder {
|
|||||||
public void setTraccarManagementService(TraccarManagementService traccarManagementService) {
|
public void setTraccarManagementService(TraccarManagementService traccarManagementService) {
|
||||||
this.traccarManagementService = traccarManagementService;
|
this.traccarManagementService = traccarManagementService;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the Dynamic Client Registration REST API Service instance from OSGI service context.
|
||||||
|
* @return {@link APIApplicationServices} Dynamic Client Registration REST API Service
|
||||||
|
*/
|
||||||
|
public APIApplicationServices getApiApplicationServices() {
|
||||||
|
PrivilegedCarbonContext ctx = PrivilegedCarbonContext.getThreadLocalCarbonContext();
|
||||||
|
apiApplicationServices = (APIApplicationServices) ctx.getOSGiService(APIApplicationServices.class, null);
|
||||||
|
if (apiApplicationServices == null) {
|
||||||
|
throw new IllegalStateException("Dynamic Client Registration REST API Service was not initialized.");
|
||||||
|
}
|
||||||
|
return apiApplicationServices;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setApiApplicationServices(APIApplicationServices apiApplicationServices) {
|
||||||
|
this.apiApplicationServices = apiApplicationServices;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the API Manager Publisher REST API Service instance from OSGI service context.
|
||||||
|
* @return {@link PublisherRESTAPIServices} API Manager Publisher REST API Service
|
||||||
|
*/
|
||||||
|
public PublisherRESTAPIServices getPublisherRESTAPIServices() {
|
||||||
|
PrivilegedCarbonContext ctx = PrivilegedCarbonContext.getThreadLocalCarbonContext();
|
||||||
|
publisherRESTAPIServices = (PublisherRESTAPIServices) ctx.getOSGiService(PublisherRESTAPIServices.class, null);
|
||||||
|
if (publisherRESTAPIServices == null) {
|
||||||
|
throw new IllegalStateException("API Manager Publisher REST API Service was not initialized.");
|
||||||
|
}
|
||||||
|
return publisherRESTAPIServices;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setPublisherRESTAPIServices(PublisherRESTAPIServices publisherRESTAPIServices) {
|
||||||
|
this.publisherRESTAPIServices = publisherRESTAPIServices;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -17,12 +17,28 @@
|
|||||||
*/
|
*/
|
||||||
package io.entgra.device.mgt.core.device.mgt.core.internal;
|
package io.entgra.device.mgt.core.device.mgt.core.internal;
|
||||||
|
|
||||||
|
import com.google.gson.Gson;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.APIApplicationServices;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.PublisherRESTAPIServices;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.constants.Constants;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.dto.APIApplicationKey;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.dto.APIInfo.Scope;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.dto.AccessTokenInfo;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.exceptions.APIServicesException;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.exceptions.BadRequestException;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.exceptions.UnexpectedResponseException;
|
||||||
|
import io.entgra.device.mgt.core.apimgt.extension.rest.api.util.APIPublisherUtils;
|
||||||
|
import io.entgra.device.mgt.core.device.mgt.common.exceptions.MetadataKeyAlreadyExistsException;
|
||||||
|
import io.entgra.device.mgt.core.device.mgt.common.exceptions.MetadataManagementException;
|
||||||
|
import io.entgra.device.mgt.core.device.mgt.common.metadata.mgt.Metadata;
|
||||||
|
import io.entgra.device.mgt.core.device.mgt.common.metadata.mgt.MetadataManagementService;
|
||||||
import org.apache.axis2.context.ConfigurationContext;
|
import org.apache.axis2.context.ConfigurationContext;
|
||||||
import org.apache.commons.logging.Log;
|
import org.apache.commons.logging.Log;
|
||||||
import org.apache.commons.logging.LogFactory;
|
import org.apache.commons.logging.LogFactory;
|
||||||
import org.wso2.carbon.context.PrivilegedCarbonContext;
|
import org.wso2.carbon.context.PrivilegedCarbonContext;
|
||||||
import io.entgra.device.mgt.core.device.mgt.core.DeviceManagementConstants;
|
import io.entgra.device.mgt.core.device.mgt.core.DeviceManagementConstants;
|
||||||
import io.entgra.device.mgt.core.device.mgt.core.DeviceManagementConstants.User;
|
import io.entgra.device.mgt.core.device.mgt.core.DeviceManagementConstants.User;
|
||||||
|
import org.wso2.carbon.tenant.mgt.exception.TenantManagementException;
|
||||||
import org.wso2.carbon.user.api.AuthorizationManager;
|
import org.wso2.carbon.user.api.AuthorizationManager;
|
||||||
import org.wso2.carbon.user.api.Permission;
|
import org.wso2.carbon.user.api.Permission;
|
||||||
import org.wso2.carbon.user.api.UserRealm;
|
import org.wso2.carbon.user.api.UserRealm;
|
||||||
@ -30,12 +46,20 @@ import org.wso2.carbon.user.api.UserStoreException;
|
|||||||
import org.wso2.carbon.user.api.UserStoreManager;
|
import org.wso2.carbon.user.api.UserStoreManager;
|
||||||
import org.wso2.carbon.utils.AbstractAxis2ConfigurationContextObserver;
|
import org.wso2.carbon.utils.AbstractAxis2ConfigurationContextObserver;
|
||||||
import org.wso2.carbon.utils.multitenancy.MultitenantConstants;
|
import org.wso2.carbon.utils.multitenancy.MultitenantConstants;
|
||||||
|
import org.wso2.carbon.utils.multitenancy.MultitenantUtils;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Load configuration files to tenant's registry.
|
* Load configuration files to tenant's registry.
|
||||||
*/
|
*/
|
||||||
public class TenantCreateObserver extends AbstractAxis2ConfigurationContextObserver {
|
public class TenantCreateObserver extends AbstractAxis2ConfigurationContextObserver {
|
||||||
private static final Log log = LogFactory.getLog(TenantCreateObserver.class);
|
private static final Log log = LogFactory.getLog(TenantCreateObserver.class);
|
||||||
|
private String msg = null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create configuration context.
|
* Create configuration context.
|
||||||
@ -82,6 +106,19 @@ public class TenantCreateObserver extends AbstractAxis2ConfigurationContextObser
|
|||||||
userStoreManager.updateRoleListOfUser(tenantAdminName, null,
|
userStoreManager.updateRoleListOfUser(tenantAdminName, null,
|
||||||
new String[] {DeviceManagementConstants.User.DEFAULT_DEVICE_ADMIN,
|
new String[] {DeviceManagementConstants.User.DEFAULT_DEVICE_ADMIN,
|
||||||
DeviceManagementConstants.User.DEFAULT_DEVICE_USER});
|
DeviceManagementConstants.User.DEFAULT_DEVICE_USER});
|
||||||
|
|
||||||
|
Thread thread = new Thread(new Runnable() {
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
try {
|
||||||
|
publishScopesToTenant(tenantDomain);
|
||||||
|
} catch (TenantManagementException e) {
|
||||||
|
log.error("Error occurred while generating API application for the tenant: " + tenantDomain + ".");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
thread.start();
|
||||||
|
|
||||||
if (log.isDebugEnabled()) {
|
if (log.isDebugEnabled()) {
|
||||||
log.debug("Device management roles: " + User.DEFAULT_DEVICE_USER + ", " + User.DEFAULT_DEVICE_ADMIN +
|
log.debug("Device management roles: " + User.DEFAULT_DEVICE_USER + ", " + User.DEFAULT_DEVICE_ADMIN +
|
||||||
" created for the tenant:" + tenantDomain + "."
|
" created for the tenant:" + tenantDomain + "."
|
||||||
@ -94,4 +131,308 @@ public class TenantCreateObserver extends AbstractAxis2ConfigurationContextObser
|
|||||||
log.error("Error occurred while creating roles for the tenant: " + tenantDomain + ".");
|
log.error("Error occurred while creating roles for the tenant: " + tenantDomain + ".");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method will create OAuth application under the given tenant domain and generate an access token against the
|
||||||
|
* client credentials. Once this access token is generated it will then be used to retrieve all the scopes that are already
|
||||||
|
* published to that tenant space. The scopes of the super tenant will also be retrieved in order to compare which scopes were added
|
||||||
|
* or removed. (A temporary admin user will be created in the sub tenant space to publish the scopes and will be deleted once
|
||||||
|
* the scope publishing task is done)
|
||||||
|
* @param tenantDomain tenant domain that the scopes will be published to.
|
||||||
|
* @throws TenantManagementException if there are any errors when publishing scopes to a tenant
|
||||||
|
*/
|
||||||
|
private void publishScopesToTenant(String tenantDomain) throws TenantManagementException {
|
||||||
|
if (!MultitenantConstants.SUPER_TENANT_DOMAIN_NAME.equals(tenantDomain)) {
|
||||||
|
|
||||||
|
MetadataManagementService metadataManagementService = DeviceManagementDataHolder.getInstance().getMetadataManagementService();
|
||||||
|
|
||||||
|
Map<String, String> superTenantPermScopeMapping = getPermScopeMapping(MultitenantConstants.SUPER_TENANT_DOMAIN_NAME);
|
||||||
|
Map<String, String> subTenantPermScopeMapping = getPermScopeMapping(tenantDomain);
|
||||||
|
|
||||||
|
if (superTenantPermScopeMapping == null) {
|
||||||
|
msg = "Error occurred while retrieving meta key '" + Constants.PERM_SCOPE_MAPPING_META_KEY + "' for tenant '" +
|
||||||
|
MultitenantConstants.SUPER_TENANT_DOMAIN_NAME + "'. Hence aborting publishing scopes to tenant: '" +
|
||||||
|
tenantDomain + "'.";
|
||||||
|
log.error(msg);
|
||||||
|
throw new TenantManagementException(msg);
|
||||||
|
}
|
||||||
|
if (superTenantPermScopeMapping.equals(subTenantPermScopeMapping)) {
|
||||||
|
if (log.isDebugEnabled()) {
|
||||||
|
log.debug( "Scopes in '" + tenantDomain + "' are up to date with super tenant scopes.");
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
APIApplicationServices apiApplicationServices = DeviceManagementDataHolder.getInstance().getApiApplicationServices();
|
||||||
|
APIApplicationKey apiApplicationKey;
|
||||||
|
AccessTokenInfo accessTokenInfo;
|
||||||
|
|
||||||
|
try {
|
||||||
|
PrivilegedCarbonContext.startTenantFlow();
|
||||||
|
PrivilegedCarbonContext.getThreadLocalCarbonContext().setTenantDomain(tenantDomain, true);
|
||||||
|
|
||||||
|
APIPublisherUtils.createScopePublishUserIfNotExists(tenantDomain);
|
||||||
|
apiApplicationKey = apiApplicationServices.createAndRetrieveApplicationCredentials();
|
||||||
|
accessTokenInfo = apiApplicationServices.generateAccessTokenFromRegisteredApplication(
|
||||||
|
apiApplicationKey.getClientId(), apiApplicationKey.getClientSecret());
|
||||||
|
} catch (APIServicesException e) {
|
||||||
|
msg = "Error occurred while generating the API application for tenant: '" + tenantDomain + "'.";
|
||||||
|
log.error(msg, e);
|
||||||
|
throw new TenantManagementException(msg, e);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
PublisherRESTAPIServices publisherRESTAPIServices = DeviceManagementDataHolder.getInstance().getPublisherRESTAPIServices();
|
||||||
|
Scope[] superTenantScopes = getAllScopesFromSuperTenant(apiApplicationServices, publisherRESTAPIServices);
|
||||||
|
|
||||||
|
if (superTenantScopes != null) {
|
||||||
|
if (log.isDebugEnabled()) {
|
||||||
|
log.debug("Number of super tenant scopes already published - " + superTenantScopes.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
Scope[] subTenantScopes = publisherRESTAPIServices.getScopes(apiApplicationKey, accessTokenInfo);
|
||||||
|
|
||||||
|
if (subTenantScopes.length > 0) {
|
||||||
|
// If there is already existing scopes on the sub tenant space then do a comparison with the
|
||||||
|
// super tenant scopes to add those new scopes to sub tenant space or to delete them from
|
||||||
|
// sub tenant space if it is not existing on the super tenant scope list.
|
||||||
|
|
||||||
|
if (log.isDebugEnabled()) {
|
||||||
|
log.debug("Number of sub tenant scopes already published - " + subTenantScopes.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
List<Scope> missingScopes = new ArrayList<>();
|
||||||
|
List<Scope> deletedScopes = new ArrayList<>();
|
||||||
|
|
||||||
|
for (Scope superTenantScope : superTenantScopes) {
|
||||||
|
boolean isMatchingScope = false;
|
||||||
|
for (Scope subTenantScope : subTenantScopes) {
|
||||||
|
if (superTenantScope.getName().equals(subTenantScope.getName())) {
|
||||||
|
isMatchingScope = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!isMatchingScope) {
|
||||||
|
if (log.isDebugEnabled()) {
|
||||||
|
log.debug("Missing scope found in sub tenant space - " +
|
||||||
|
superTenantScope.getName());
|
||||||
|
}
|
||||||
|
missingScopes.add(superTenantScope);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (log.isDebugEnabled()) {
|
||||||
|
log.debug("Total number of missing scopes found in sub tenant space - " +
|
||||||
|
missingScopes.size());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (missingScopes.size() > 0) {
|
||||||
|
if (log.isDebugEnabled()) {
|
||||||
|
log.debug("Starting to add new/updated shared scopes to the tenant: '" + tenantDomain + "'.");
|
||||||
|
}
|
||||||
|
publishSharedScopes(missingScopes, publisherRESTAPIServices, apiApplicationKey,
|
||||||
|
accessTokenInfo);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (Scope subTenantScope : subTenantScopes) {
|
||||||
|
boolean isMatchingScope = false;
|
||||||
|
for (Scope superTenantScope : superTenantScopes) {
|
||||||
|
if (superTenantScope.getName().equals(subTenantScope.getName())) {
|
||||||
|
isMatchingScope = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!isMatchingScope) {
|
||||||
|
if (log.isDebugEnabled()) {
|
||||||
|
log.debug("Deleted scope found in sub tenant space - " +
|
||||||
|
subTenantScope.getName());
|
||||||
|
}
|
||||||
|
deletedScopes.add(subTenantScope);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (log.isDebugEnabled()) {
|
||||||
|
log.debug("Total number of deleted scopes found in sub tenant space - " +
|
||||||
|
deletedScopes.size());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (deletedScopes.size() > 0) {
|
||||||
|
if (log.isDebugEnabled()) {
|
||||||
|
log.debug("Starting to delete shared scopes from the tenant: '" + tenantDomain + "'.");
|
||||||
|
}
|
||||||
|
for (Scope deletedScope : deletedScopes) {
|
||||||
|
if (publisherRESTAPIServices.isSharedScopeNameExists(apiApplicationKey, accessTokenInfo,
|
||||||
|
deletedScope.getName())) {
|
||||||
|
Scope scope = createScopeObject(deletedScope);
|
||||||
|
publisherRESTAPIServices.deleteSharedScope(apiApplicationKey, accessTokenInfo, scope);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (missingScopes.size() > 0 || deletedScopes.size() > 0) {
|
||||||
|
updatePermScopeMetaData(superTenantPermScopeMapping, metadataManagementService);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (log.isDebugEnabled()) {
|
||||||
|
log.debug("Starting to publish shared scopes to newly created tenant: '" + tenantDomain + "'.");
|
||||||
|
}
|
||||||
|
|
||||||
|
publishSharedScopes(Arrays.asList(superTenantScopes), publisherRESTAPIServices,
|
||||||
|
apiApplicationKey, accessTokenInfo);
|
||||||
|
updatePermScopeMetaData(superTenantPermScopeMapping, metadataManagementService);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
msg = "Unable to publish scopes to sub tenants due to super tenant scopes list being empty.";
|
||||||
|
log.error(msg);
|
||||||
|
throw new TenantManagementException(msg);
|
||||||
|
}
|
||||||
|
} catch (BadRequestException e) {
|
||||||
|
msg = "Invalid request sent when publishing scopes to '" + tenantDomain + "' tenant space.";
|
||||||
|
log.error(msg, e);
|
||||||
|
throw new TenantManagementException(msg, e);
|
||||||
|
} catch (UnexpectedResponseException e) {
|
||||||
|
msg = "Unexpected response received when publishing scopes to '" + tenantDomain + "' tenant space.";
|
||||||
|
log.error(msg, e);
|
||||||
|
throw new TenantManagementException(msg, e);
|
||||||
|
} catch (APIServicesException e) {
|
||||||
|
msg = "Error occurred while publishing scopes to '" + tenantDomain + "' tenant space.";
|
||||||
|
log.error(msg, e);
|
||||||
|
throw new TenantManagementException(msg, e);
|
||||||
|
} catch (MetadataManagementException e) {
|
||||||
|
msg = "Error occurred trying to create metadata entry '" + Constants.PERM_SCOPE_MAPPING_META_KEY + "'.";
|
||||||
|
log.error(msg);
|
||||||
|
throw new TenantManagementException(msg);
|
||||||
|
} catch (MetadataKeyAlreadyExistsException e) {
|
||||||
|
msg = "Error occurred trying to create metadata entry '" + Constants.PERM_SCOPE_MAPPING_META_KEY + "'. The meta key " +
|
||||||
|
"already exists.";
|
||||||
|
log.error(msg);
|
||||||
|
throw new TenantManagementException(msg);
|
||||||
|
} finally {
|
||||||
|
APIPublisherUtils.removeScopePublishUserIfExists(tenantDomain);
|
||||||
|
PrivilegedCarbonContext.endTenantFlow();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method will retrieve the value of the permission scope mapping meta key stored in each tenant's metadata
|
||||||
|
* @param tenantDomain the tenant domain that the permission scope mapping meta value retrieved from
|
||||||
|
* @return {@link Map} containing the permission key and the scope value
|
||||||
|
* @throws TenantManagementException if there is an error while retrieving permission scope metadata
|
||||||
|
*/
|
||||||
|
private Map<String, String> getPermScopeMapping(String tenantDomain) throws TenantManagementException {
|
||||||
|
if (log.isDebugEnabled()) {
|
||||||
|
log.debug("Retrieving permission scope mapping from metadata from the tenant: '" + tenantDomain + "'.");
|
||||||
|
}
|
||||||
|
Map<String, String> permScopeMapping = null;
|
||||||
|
try {
|
||||||
|
PrivilegedCarbonContext.startTenantFlow();
|
||||||
|
PrivilegedCarbonContext.getThreadLocalCarbonContext().setTenantDomain(tenantDomain, true);
|
||||||
|
MetadataManagementService metadataManagementService = DeviceManagementDataHolder.getInstance().getMetadataManagementService();
|
||||||
|
Metadata metadata = metadataManagementService.retrieveMetadata(Constants.PERM_SCOPE_MAPPING_META_KEY);
|
||||||
|
if (metadata != null) {
|
||||||
|
permScopeMapping = new Gson().fromJson(metadata.getMetaValue().toString(), HashMap.class);
|
||||||
|
}
|
||||||
|
} catch (MetadataManagementException e) {
|
||||||
|
msg = "Error occurred while retrieving permission scope mapping from metadata for tenant: '" + tenantDomain + "'.";
|
||||||
|
log.error(msg, e);
|
||||||
|
throw new TenantManagementException(msg, e);
|
||||||
|
} finally {
|
||||||
|
PrivilegedCarbonContext.endTenantFlow();
|
||||||
|
}
|
||||||
|
return permScopeMapping;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method will create a new metadata entry or update the existing metadata entry in the sub tenant metadata repository which is
|
||||||
|
* taken from the super tenant metadata
|
||||||
|
* @param superTenantPermScopeMapping {@link Map} containing the permission key and the scope value of the super tenant
|
||||||
|
* @param metadataManagementService {@link MetadataManagementService} instance
|
||||||
|
* @throws MetadataManagementException if there is an error while creating or updating the metadata entry
|
||||||
|
* @throws MetadataKeyAlreadyExistsException if the metadata key already exists while trying to create a new metadata entry
|
||||||
|
*/
|
||||||
|
private void updatePermScopeMetaData(Map<String, String> superTenantPermScopeMapping,
|
||||||
|
MetadataManagementService metadataManagementService) throws MetadataManagementException,
|
||||||
|
MetadataKeyAlreadyExistsException {
|
||||||
|
|
||||||
|
Metadata newMetaData = new Metadata();
|
||||||
|
newMetaData.setMetaKey(Constants.PERM_SCOPE_MAPPING_META_KEY);
|
||||||
|
newMetaData.setMetaValue(new Gson().toJson(superTenantPermScopeMapping));
|
||||||
|
if (metadataManagementService.retrieveMetadata(Constants.PERM_SCOPE_MAPPING_META_KEY) == null) {
|
||||||
|
metadataManagementService.createMetadata(newMetaData);
|
||||||
|
} else {
|
||||||
|
metadataManagementService.updateMetadata(newMetaData);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get all the scopes from the super tenant space
|
||||||
|
* @param apiApplicationServices {@link APIApplicationServices} is used to create an OAuth application and retrieve client ID and secret
|
||||||
|
* @param publisherRESTAPIServices {@link PublisherRESTAPIServices} is used to get all scopes under a given tenant using client credentials
|
||||||
|
* @return array of {@link Scope}
|
||||||
|
* @throws BadRequestException if an invalid request is sent to the API Manager Publisher REST API Service
|
||||||
|
* @throws UnexpectedResponseException if an unexpected response is received from the API Manager Publisher REST API Service
|
||||||
|
* @throws TenantManagementException if an error occurred while processing the request sent to API Manager Publisher REST API Service
|
||||||
|
*/
|
||||||
|
private Scope[] getAllScopesFromSuperTenant(APIApplicationServices apiApplicationServices,
|
||||||
|
PublisherRESTAPIServices publisherRESTAPIServices) throws BadRequestException,
|
||||||
|
UnexpectedResponseException, TenantManagementException {
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Get all scopes of super tenant to compare later with the sub tenant scopes. This is done
|
||||||
|
// in order to see if any new scopes were added or deleted
|
||||||
|
PrivilegedCarbonContext.startTenantFlow();
|
||||||
|
PrivilegedCarbonContext.getThreadLocalCarbonContext().setTenantDomain(MultitenantConstants.SUPER_TENANT_DOMAIN_NAME, true);
|
||||||
|
APIApplicationKey superTenantApiApplicationKey = apiApplicationServices.createAndRetrieveApplicationCredentials();
|
||||||
|
AccessTokenInfo superTenantAccessToken = apiApplicationServices.generateAccessTokenFromRegisteredApplication(
|
||||||
|
superTenantApiApplicationKey.getClientId(), superTenantApiApplicationKey.getClientSecret());
|
||||||
|
return publisherRESTAPIServices.getScopes(superTenantApiApplicationKey, superTenantAccessToken);
|
||||||
|
} catch (APIServicesException e) {
|
||||||
|
msg = "Error occurred while retrieving access token from super tenant";
|
||||||
|
log.error(msg, e);
|
||||||
|
throw new TenantManagementException(msg, e);
|
||||||
|
} finally {
|
||||||
|
PrivilegedCarbonContext.endTenantFlow();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add shared scopes to the tenant space.
|
||||||
|
* @param scopeList {@link List} of {@link Scope}
|
||||||
|
* @param publisherRESTAPIServices {@link PublisherRESTAPIServices} is used to add shared scopes to a given tenant using client credentials
|
||||||
|
* @param apiApplicationKey {@link APIApplicationKey} contains client credentials of the OAuth application
|
||||||
|
* @param accessTokenInfo {@link AccessTokenInfo} contains token information generated from the client credentials
|
||||||
|
* @throws BadRequestException if an invalid request is sent to the API Manager Publisher REST API Service
|
||||||
|
* @throws UnexpectedResponseException if an unexpected response is received from the API Manager Publisher REST API Service
|
||||||
|
* @throws APIServicesException if an error occurred while processing the request sent to API Manager Publisher REST API Service
|
||||||
|
*/
|
||||||
|
private void publishSharedScopes (List<Scope> scopeList, PublisherRESTAPIServices publisherRESTAPIServices,
|
||||||
|
APIApplicationKey apiApplicationKey, AccessTokenInfo accessTokenInfo)
|
||||||
|
throws BadRequestException, UnexpectedResponseException, APIServicesException {
|
||||||
|
|
||||||
|
for (Scope tenantScope : scopeList) {
|
||||||
|
if (!publisherRESTAPIServices.isSharedScopeNameExists(apiApplicationKey, accessTokenInfo,
|
||||||
|
tenantScope.getName())) {
|
||||||
|
Scope scope = createScopeObject(tenantScope);
|
||||||
|
publisherRESTAPIServices.addNewSharedScope(apiApplicationKey, accessTokenInfo, scope);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a new scope object from the passed scope which includes the id, display name, description, name and bindings.
|
||||||
|
* @param tenantScope existing {@link Scope} from a tenant
|
||||||
|
* @return {@link Scope}
|
||||||
|
*/
|
||||||
|
private Scope createScopeObject (Scope tenantScope) {
|
||||||
|
Scope scope = new Scope();
|
||||||
|
scope.setId(tenantScope.getId());
|
||||||
|
scope.setDisplayName(tenantScope.getDisplayName());
|
||||||
|
scope.setDescription(tenantScope.getDescription());
|
||||||
|
scope.setName(tenantScope.getName());
|
||||||
|
List<String> bindings = new ArrayList<>();
|
||||||
|
bindings.add(Constants.ADMIN_ROLE_KEY);
|
||||||
|
scope.setBindings(bindings);
|
||||||
|
return scope;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
Loading…
Reference in New Issue
Block a user