From ea285be1a2573dd95583c08c77271e687b81e753 Mon Sep 17 00:00:00 2001 From: Sahan Randika Date: Fri, 10 May 2024 14:52:39 +0530 Subject: [PATCH] Add changes to add subscription status to application.yaml when exporting and to support ignoreTier query param when importing --- .../swagger.json | 4 ++-- .../src/main/resources/devportal-api.yaml | 6 ++++++ .../rest/api/store/v1/ApplicationsApi.java | 4 ++-- .../api/store/v1/ApplicationsApiService.java | 2 +- .../v1/impl/ApplicationsApiServiceImpl.java | 7 ++++--- .../v1/models/ExportedSubscribedAPI.java | 12 +++++++++++- .../rest/api/store/v1/utils/ExportUtils.java | 2 +- .../rest/api/store/v1/utils/ImportUtils.java | 19 ++++++++++++++++--- .../src/main/resources/devportal-api.yaml | 6 ++++++ 9 files changed, 49 insertions(+), 13 deletions(-) diff --git a/components/apimgt/org.wso2.carbon.apimgt.internal.service/swagger.json b/components/apimgt/org.wso2.carbon.apimgt.internal.service/swagger.json index a7fa59630d63..8040e385bd35 100644 --- a/components/apimgt/org.wso2.carbon.apimgt.internal.service/swagger.json +++ b/components/apimgt/org.wso2.carbon.apimgt.internal.service/swagger.json @@ -1354,8 +1354,8 @@ "type" : "string", "example" : "EXCHANGED", "description" : "The type of the tokens to be used (exchanged or without exchanged). Accepted values are EXCHANGED, DIRECT or BOTH.", - "enum" : [ "EXCHANGED", "DIRECT", "BOTH" ], - "default" : "DIRECT" + "default" : "DIRECT", + "enum" : [ "EXCHANGED", "DIRECT", "BOTH" ] } } }, diff --git a/components/apimgt/org.wso2.carbon.apimgt.rest.api.common/src/main/resources/devportal-api.yaml b/components/apimgt/org.wso2.carbon.apimgt.rest.api.common/src/main/resources/devportal-api.yaml index 3fef7471ffff..f892f555a3ef 100644 --- a/components/apimgt/org.wso2.carbon.apimgt.rest.api.common/src/main/resources/devportal-api.yaml +++ b/components/apimgt/org.wso2.carbon.apimgt.rest.api.common/src/main/resources/devportal-api.yaml @@ -2534,6 +2534,12 @@ paths: Update if application exists schema: type: boolean + - name: ignoreTier + in: query + description: | + Ignore tier and proceed with subscribed APIs + schema: + type: boolean requestBody: content: multipart/form-data: diff --git a/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/gen/java/org/wso2/carbon/apimgt/rest/api/store/v1/ApplicationsApi.java b/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/gen/java/org/wso2/carbon/apimgt/rest/api/store/v1/ApplicationsApi.java index a6150706f789..4806f0af0946 100644 --- a/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/gen/java/org/wso2/carbon/apimgt/rest/api/store/v1/ApplicationsApi.java +++ b/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/gen/java/org/wso2/carbon/apimgt/rest/api/store/v1/ApplicationsApi.java @@ -485,8 +485,8 @@ public Response applicationsExportGet( @NotNull @ApiParam(value = "Application N @ApiResponse(code = 207, message = "Multi Status. Partially successful response with skipped APIs information object as entity in the body. ", response = APIInfoListDTO.class), @ApiResponse(code = 400, message = "Bad Request. Invalid request or validation error.", response = ErrorDTO.class), @ApiResponse(code = 406, message = "Not Acceptable. The requested media type is not supported.", response = ErrorDTO.class) }) - public Response applicationsImportPost( @Multipart(value = "file") InputStream fileInputStream, @Multipart(value = "file" ) Attachment fileDetail, @ApiParam(value = "Preserve Original Creator of the Application ") @QueryParam("preserveOwner") Boolean preserveOwner, @ApiParam(value = "Skip importing Subscriptions of the Application ") @QueryParam("skipSubscriptions") Boolean skipSubscriptions, @ApiParam(value = "Expected Owner of the Application in the Import Environment ") @QueryParam("appOwner") String appOwner, @ApiParam(value = "Skip importing Keys of the Application ") @QueryParam("skipApplicationKeys") Boolean skipApplicationKeys, @ApiParam(value = "Update if application exists ") @QueryParam("update") Boolean update) throws APIManagementException{ - return delegate.applicationsImportPost(fileInputStream, fileDetail, preserveOwner, skipSubscriptions, appOwner, skipApplicationKeys, update, securityContext); + public Response applicationsImportPost( @Multipart(value = "file") InputStream fileInputStream, @Multipart(value = "file" ) Attachment fileDetail, @ApiParam(value = "Preserve Original Creator of the Application ") @QueryParam("preserveOwner") Boolean preserveOwner, @ApiParam(value = "Skip importing Subscriptions of the Application ") @QueryParam("skipSubscriptions") Boolean skipSubscriptions, @ApiParam(value = "Expected Owner of the Application in the Import Environment ") @QueryParam("appOwner") String appOwner, @ApiParam(value = "Skip importing Keys of the Application ") @QueryParam("skipApplicationKeys") Boolean skipApplicationKeys, @ApiParam(value = "Update if application exists ") @QueryParam("update") Boolean update, @ApiParam(value = "Ignore tier and proceed with subscribed APIs ") @QueryParam("ignoreTier") Boolean ignoreTier) throws APIManagementException{ + return delegate.applicationsImportPost(fileInputStream, fileDetail, preserveOwner, skipSubscriptions, appOwner, skipApplicationKeys, update, ignoreTier, securityContext); } @POST diff --git a/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/gen/java/org/wso2/carbon/apimgt/rest/api/store/v1/ApplicationsApiService.java b/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/gen/java/org/wso2/carbon/apimgt/rest/api/store/v1/ApplicationsApiService.java index 965e308306f4..0b386551ef3a 100644 --- a/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/gen/java/org/wso2/carbon/apimgt/rest/api/store/v1/ApplicationsApiService.java +++ b/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/gen/java/org/wso2/carbon/apimgt/rest/api/store/v1/ApplicationsApiService.java @@ -58,6 +58,6 @@ public interface ApplicationsApiService { public Response applicationsApplicationIdPut(String applicationId, ApplicationDTO applicationDTO, String ifMatch, MessageContext messageContext) throws APIManagementException; public Response applicationsExportGet(String appName, String appOwner, Boolean withKeys, String format, MessageContext messageContext) throws APIManagementException; public Response applicationsGet(String groupId, String query, String sortBy, String sortOrder, Integer limit, Integer offset, String ifNoneMatch, MessageContext messageContext) throws APIManagementException; - public Response applicationsImportPost(InputStream fileInputStream, Attachment fileDetail, Boolean preserveOwner, Boolean skipSubscriptions, String appOwner, Boolean skipApplicationKeys, Boolean update, MessageContext messageContext) throws APIManagementException; + public Response applicationsImportPost(InputStream fileInputStream, Attachment fileDetail, Boolean preserveOwner, Boolean skipSubscriptions, String appOwner, Boolean skipApplicationKeys, Boolean update, Boolean ignoreTier, MessageContext messageContext) throws APIManagementException; public Response applicationsPost(ApplicationDTO applicationDTO, MessageContext messageContext) throws APIManagementException; } diff --git a/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/java/org/wso2/carbon/apimgt/rest/api/store/v1/impl/ApplicationsApiServiceImpl.java b/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/java/org/wso2/carbon/apimgt/rest/api/store/v1/impl/ApplicationsApiServiceImpl.java index a8c6373f9682..7db29bcbc4c9 100644 --- a/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/java/org/wso2/carbon/apimgt/rest/api/store/v1/impl/ApplicationsApiServiceImpl.java +++ b/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/java/org/wso2/carbon/apimgt/rest/api/store/v1/impl/ApplicationsApiServiceImpl.java @@ -131,7 +131,7 @@ public Response applicationsGet(String groupId, String query, String sortBy, Str // todo: Do a second level filtering for the incoming group ID. // todo: eg: use case is when there are lots of applications which is accessible to his group "g1", he wants to see - // todo: what are the applications shared to group "g2" among them. + // todo: what are the applications shared to group "g2" among them. groupId = RestApiUtil.getLoggedInUserGroupId(); try { String organization = RestApiUtil.getValidatedOrganization(messageContext); @@ -199,12 +199,13 @@ public Response applicationsGet(String groupId, String query, String sortBy, Str * @param appOwner Target owner of the application * @param skipApplicationKeys Skip application keys while importing * @param update Update if existing application found or import + * @param ignoreTier Ignore tier and proceed with subscribed APIs * @param messageContext Message Context * @return imported Application */ @Override public Response applicationsImportPost(InputStream fileInputStream, Attachment fileDetail, Boolean preserveOwner, Boolean skipSubscriptions, String appOwner, Boolean skipApplicationKeys, - Boolean update, MessageContext messageContext) throws APIManagementException { + Boolean update, Boolean ignoreTier, MessageContext messageContext) throws APIManagementException { String ownerId; Application application; @@ -257,7 +258,7 @@ public Response applicationsGet(String groupId, String query, String sortBy, Str if (skipSubscriptions == null || !skipSubscriptions) { skippedAPIs = ImportUtils .importSubscriptions(exportedApplication.getSubscribedAPIs(), ownerId, application, - update, apiConsumer, organization); + update, ignoreTier, apiConsumer, organization); } Application importedApplication = apiConsumer.getApplicationById(application.getId()); importedApplication.setOwner(ownerId); diff --git a/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/java/org/wso2/carbon/apimgt/rest/api/store/v1/models/ExportedSubscribedAPI.java b/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/java/org/wso2/carbon/apimgt/rest/api/store/v1/models/ExportedSubscribedAPI.java index 01653ad6912c..f9fc894667f6 100644 --- a/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/java/org/wso2/carbon/apimgt/rest/api/store/v1/models/ExportedSubscribedAPI.java +++ b/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/java/org/wso2/carbon/apimgt/rest/api/store/v1/models/ExportedSubscribedAPI.java @@ -25,11 +25,13 @@ public class ExportedSubscribedAPI { private APIIdentifier apiId; private Subscriber subscriber; private String throttlingPolicy; + private String subscriptionStatus; - public ExportedSubscribedAPI(APIIdentifier apiId, Subscriber subscriber, String throttlingPolicy) { + public ExportedSubscribedAPI(APIIdentifier apiId, Subscriber subscriber, String throttlingPolicy, String subscriptionStatus) { this.apiId = apiId; this.subscriber = subscriber; this.throttlingPolicy = throttlingPolicy; + this.subscriptionStatus = subscriptionStatus; } public APIIdentifier getApiId() { @@ -55,4 +57,12 @@ public String getThrottlingPolicy() { public void setThrottlingPolicy(String throttlingPolicy) { this.throttlingPolicy = throttlingPolicy; } + + public String getSubscriptionStatus() { + return subscriptionStatus; + } + + public void setSubscriptionStatus(String subscriptionStatus) { + this.subscriptionStatus = subscriptionStatus; + } } diff --git a/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/java/org/wso2/carbon/apimgt/rest/api/store/v1/utils/ExportUtils.java b/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/java/org/wso2/carbon/apimgt/rest/api/store/v1/utils/ExportUtils.java index 109d4ab7a553..24ea7c6104d7 100644 --- a/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/java/org/wso2/carbon/apimgt/rest/api/store/v1/utils/ExportUtils.java +++ b/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/java/org/wso2/carbon/apimgt/rest/api/store/v1/utils/ExportUtils.java @@ -150,7 +150,7 @@ private static ExportedApplication createApplicationDTOToExport(Application appl Set exportedSubscribedAPIs = new HashSet<>(); for (SubscribedAPI subscribedAPI : subscribedAPIs) { ExportedSubscribedAPI exportedSubscribedAPI = new ExportedSubscribedAPI(subscribedAPI.getAPIIdentifier(), - subscribedAPI.getSubscriber(), subscribedAPI.getTier().getName()); + subscribedAPI.getSubscriber(), subscribedAPI.getTier().getName(), subscribedAPI.getSubStatus()); exportedSubscribedAPIs.add(exportedSubscribedAPI); } diff --git a/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/java/org/wso2/carbon/apimgt/rest/api/store/v1/utils/ImportUtils.java b/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/java/org/wso2/carbon/apimgt/rest/api/store/v1/utils/ImportUtils.java index 03f88b15c196..64ec2f55842e 100644 --- a/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/java/org/wso2/carbon/apimgt/rest/api/store/v1/utils/ImportUtils.java +++ b/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/java/org/wso2/carbon/apimgt/rest/api/store/v1/utils/ImportUtils.java @@ -152,7 +152,8 @@ public static APIKey getAPIKeyFromApplicationKeyDTO(ApplicationKeyDTO applicatio * @throws UserStoreException if an error occurs while checking whether the tenant domain exists */ public static List importSubscriptions(Set subscribedAPIs, String userId, - Application application, Boolean update, APIConsumer apiConsumer, String organization) + Application application, Boolean update,Boolean ignoreTier, + APIConsumer apiConsumer, String organization) throws APIManagementException, UserStoreException { List skippedAPIList = new ArrayList<>(); @@ -189,6 +190,18 @@ public static List importSubscriptions(Set // on update skip subscriptions that already exists apiConsumer.addSubscription(apiTypeWrapper, userId, application); } + } else if (ignoreTier != null && ignoreTier && apiTypeWrapper.getStatus() != null + && APIConstants.PUBLISHED.equals(apiTypeWrapper.getStatus())) { + apiTypeWrapper.setTier(targetTier); + // Add subscription if update flag is not specified + // It will throw an error if subscriber already exists + if (update == null || !update) { + apiConsumer.addSubscription(apiTypeWrapper, userId, application); + } else if (!apiConsumer.isSubscribedToApp(subscribedAPI.getApiId(), userId + , application.getId())) { + // on update skip subscriptions that already exists + apiConsumer.addSubscription(apiTypeWrapper, userId, application); + } } else { log.error("Failed to import Subscription as API/API Product " + apiIdentifier.getName() + "-" + apiIdentifier.getVersion() + " as one or more tiers may " @@ -243,10 +256,10 @@ private static boolean isTierAvailable(String targetTierName, ApiTypeWrapper api } } if (!apiTypeWrapper.isAPIProduct()) { - log.error("Tier:" + targetTierName + " is not available for API " + api.getId().getApiName() + "-" + api + log.warn("Tier:" + targetTierName + " is not available for API " + api.getId().getApiName() + "-" + api .getId().getVersion()); } else { - log.error( + log.warn( "Tier:" + targetTierName + " is not available for API Product " + apiProduct.getId().getName() + "-" + apiProduct.getId().getVersion()); } diff --git a/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/resources/devportal-api.yaml b/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/resources/devportal-api.yaml index 3fef7471ffff..f892f555a3ef 100644 --- a/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/resources/devportal-api.yaml +++ b/components/apimgt/org.wso2.carbon.apimgt.rest.api.store.v1/src/main/resources/devportal-api.yaml @@ -2534,6 +2534,12 @@ paths: Update if application exists schema: type: boolean + - name: ignoreTier + in: query + description: | + Ignore tier and proceed with subscribed APIs + schema: + type: boolean requestBody: content: multipart/form-data: