Skip to content

Commit f527679

Browse files
committed
Validate tenant creation request data
Fix tenant already exists exception not being thrown
1 parent fe1770e commit f527679

File tree

7 files changed

+110
-10
lines changed

7 files changed

+110
-10
lines changed

gradlew

100644100755
File mode changed.

src/main/java/org/fineract/messagegateway/exception/SecurityException.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ private SecurityException(final String msg) {
2424
super(msg);
2525
}
2626

27-
public static SecurityException tenantAlreadyExisits(final String tenant) {
27+
public static SecurityException tenantAlreadyExists(final String tenant) {
2828
return new SecurityException("Tenant Already existing with "+tenant+" identifier");
2929
}
3030
}

src/main/java/org/fineract/messagegateway/service/SecurityService.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -73,7 +73,7 @@ public String generateApiKey(final SMSBridge smsBridge) {
7373
public String generateApiKey(final String tenantId) {
7474
Tenant tenant = this.tenantRepository.findByTenantId(tenantId) ;
7575
if(tenant != null) {
76-
org.fineract.messagegateway.exception.SecurityException.tenantAlreadyExisits(tenantId) ;
76+
throw org.fineract.messagegateway.exception.SecurityException.tenantAlreadyExists(tenantId) ;
7777
}
7878

7979
final String randomKey = UUID.randomUUID().toString();

src/main/java/org/fineract/messagegateway/tenants/api/TenantsApiResource.java

+19-4
Original file line numberDiff line numberDiff line change
@@ -18,11 +18,16 @@
1818
*/
1919
package org.fineract.messagegateway.tenants.api;
2020

21-
import org.fineract.messagegateway.tenants.domain.Tenant;
21+
import org.fineract.messagegateway.exception.PlatformApiDataValidationException;
22+
import org.fineract.messagegateway.exception.UnsupportedParameterException;
23+
import org.fineract.messagegateway.helpers.ApiGlobalErrorResponse;
24+
import org.fineract.messagegateway.helpers.PlatformApiDataValidationExceptionMapper;
25+
import org.fineract.messagegateway.helpers.UnsupportedParameterExceptionMapper;
2226
import org.fineract.messagegateway.tenants.service.TenantsService;
2327
import org.springframework.beans.factory.annotation.Autowired;
2428
import org.springframework.http.HttpStatus;
2529
import org.springframework.http.ResponseEntity;
30+
import org.springframework.web.bind.annotation.ExceptionHandler;
2631
import org.springframework.web.bind.annotation.RequestBody;
2732
import org.springframework.web.bind.annotation.RequestMapping;
2833
import org.springframework.web.bind.annotation.RequestMethod;
@@ -40,8 +45,18 @@ public TenantsApiResource(final TenantsService tenantService) {
4045
}
4146

4247
@RequestMapping(method = RequestMethod.POST, consumes = {"application/json"}, produces = {"application/json"})
43-
public ResponseEntity<String> createSMSBridgeConfig(@RequestBody final Tenant tenant) {
44-
String appKey = this.tenantService.createTenant(tenant) ;
45-
return new ResponseEntity<>(appKey, HttpStatus.CREATED);
48+
public ResponseEntity<String> createSMSBridgeConfig(@RequestBody final String requestJson) {
49+
String appKey = this.tenantService.createTenant(requestJson) ;
50+
return new ResponseEntity<>(appKey, HttpStatus.CREATED);
4651
}
52+
53+
@ExceptionHandler({PlatformApiDataValidationException.class})
54+
public ResponseEntity<ApiGlobalErrorResponse> handlePlatformApiDataValidationException(PlatformApiDataValidationException e) {
55+
return PlatformApiDataValidationExceptionMapper.toResponse(e) ;
56+
}
57+
58+
@ExceptionHandler({UnsupportedParameterException.class})
59+
public ResponseEntity<ApiGlobalErrorResponse> handleUnsupportedParameterException(UnsupportedParameterException e) {
60+
return UnsupportedParameterExceptionMapper.toResponse(e) ;
61+
}
4762
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
package org.fineract.messagegateway.tenants.constants;
2+
3+
import java.util.Arrays;
4+
import java.util.HashSet;
5+
import java.util.Set;
6+
7+
import org.fineract.messagegateway.constants.MessageGatewayConstants;
8+
9+
public interface TenantConstants extends MessageGatewayConstants {
10+
11+
String TENANTS_RESOURCE_NAME = "tenants";
12+
13+
String TENANT_ID = "tenantId";
14+
String TENANT_DESCRIPTION = "description";
15+
16+
17+
// list of allowed parameters for tenant creation request
18+
Set<String> CREATE_REQUEST_PARAMETERS = new HashSet<>(Arrays.asList(TENANT_ID, TENANT_DESCRIPTION));
19+
20+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,60 @@
1+
package org.fineract.messagegateway.tenants.serialization;
2+
3+
import java.lang.reflect.Type;
4+
import java.util.ArrayList;
5+
import java.util.List;
6+
import java.util.Map;
7+
8+
import org.fineract.messagegateway.exception.PlatformApiDataValidationException;
9+
import org.fineract.messagegateway.helpers.ApiParameterError;
10+
import org.fineract.messagegateway.helpers.DataValidatorBuilder;
11+
import org.fineract.messagegateway.helpers.FromJsonHelper;
12+
import org.fineract.messagegateway.tenants.constants.TenantConstants;
13+
import org.fineract.messagegateway.tenants.domain.Tenant;
14+
import org.springframework.beans.factory.annotation.Autowired;
15+
import org.springframework.stereotype.Component;
16+
17+
import com.google.gson.JsonElement;
18+
import com.google.gson.reflect.TypeToken;
19+
20+
@Component
21+
public class TenantSerializer {
22+
23+
private final FromJsonHelper fromJsonHelper;
24+
25+
@Autowired
26+
public TenantSerializer(FromJsonHelper fromJsonHelper) {
27+
this.fromJsonHelper = fromJsonHelper;
28+
}
29+
30+
public Tenant validateCreateRequest(final String json) {
31+
final Type typeOfMap = new TypeToken<Map<String, Object>>() {}.getType();
32+
this.fromJsonHelper.checkForUnsupportedParameters(typeOfMap, json,
33+
TenantConstants.CREATE_REQUEST_PARAMETERS);
34+
35+
final List<ApiParameterError> dataValidationErrors = new ArrayList<>();
36+
final DataValidatorBuilder baseDataValidator = new DataValidatorBuilder(dataValidationErrors)
37+
.resource(TenantConstants.TENANTS_RESOURCE_NAME);
38+
final JsonElement element = this.fromJsonHelper.parse(json);
39+
40+
41+
final String tenantId = this.fromJsonHelper.extractStringNamed(TenantConstants.TENANT_ID, element);
42+
baseDataValidator.reset().parameter(TenantConstants.TENANT_ID)
43+
.value(tenantId).notBlank().notExceedingLengthOf(32);
44+
45+
String tenantDescription = null;
46+
if(this.fromJsonHelper.parameterExists(TenantConstants.TENANT_DESCRIPTION, element)) {
47+
tenantDescription = this.fromJsonHelper.extractStringNamed(TenantConstants.TENANT_DESCRIPTION,
48+
element);
49+
baseDataValidator.reset().parameter(TenantConstants.TENANT_DESCRIPTION).value(tenantDescription)
50+
.notBlank().notExceedingLengthOf(500);
51+
}
52+
53+
if (!dataValidationErrors.isEmpty()) {
54+
throw new PlatformApiDataValidationException("validation.msg.validation.errors.exist",
55+
"Validation errors exist.", dataValidationErrors);
56+
}
57+
58+
return new Tenant(tenantId, null, tenantDescription);
59+
}
60+
}

src/main/java/org/fineract/messagegateway/tenants/service/TenantsService.java

+9-4
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@
2222
import org.fineract.messagegateway.tenants.domain.Tenant;
2323
import org.fineract.messagegateway.tenants.exception.TenantNotFoundException;
2424
import org.fineract.messagegateway.tenants.repository.TenantRepository;
25+
import org.fineract.messagegateway.tenants.serialization.TenantSerializer;
2526
import org.springframework.beans.factory.annotation.Autowired;
2627
import org.springframework.stereotype.Service;
2728

@@ -31,15 +32,19 @@ public class TenantsService {
3132
private final TenantRepository tenantRepository ;
3233

3334
private final SecurityService securityService ;
34-
35+
36+
private final TenantSerializer tenantSerializer;
37+
3538
@Autowired
3639
public TenantsService(final TenantRepository tenantRepository,
37-
final SecurityService securityService) {
40+
final SecurityService securityService, TenantSerializer tenantSerializer) {
3841
this.tenantRepository = tenantRepository ;
3942
this.securityService = securityService ;
43+
this.tenantSerializer = tenantSerializer;
4044
}
41-
42-
public String createTenant(final Tenant tenant) {
45+
46+
public String createTenant(final String requestJson) {
47+
Tenant tenant = tenantSerializer.validateCreateRequest(requestJson);
4348
tenant.setTenantAppKey(this.securityService.generateApiKey(tenant.getTenantId()));
4449
this.tenantRepository.save(tenant) ;
4550
return tenant.getTenantAppKey() ;

0 commit comments

Comments
 (0)