From f440bda0fce975ea7c19c98ee57952055d838a7b Mon Sep 17 00:00:00 2001 From: Ashirwada Date: Thu, 22 Aug 2024 15:37:33 +0530 Subject: [PATCH] Removing existing implementation --- .../findbugs-exclude.xml | 28 - .../pom.xml | 212 - .../service/dao/AccountMetadataDAO.java | 128 - .../service/dao/AccountMetadataDAOImpl.java | 493 - .../dao/queries/AccountMetadataDBQueries.java | 83 - .../AccountMetadataDBQueriesMySQLImpl.java | 99 - .../internal/AccountMetadataDataHolder.java | 52 - .../AccountMetadataServiceComponent.java | 88 - .../service/AccountMetadataService.java | 190 - .../service/AccountMetadataServiceImpl.java | 240 - .../service/dao/AccountMetadataDAOTests.java | 315 - .../service/AccountMetadataServiceTests.java | 610 - .../util/AccountMetadataDAOTestData.java | 86 - .../metadata/service/util/DAOUtils.java | 64 - .../src/test/resources/dbScripts/h2.sql | 8 - .../src/test/resources/testng.xml | 28 - .../pom.xml | 219 - .../CIBAConstants.java | 37 - ...BCIBARequestObjectValidationExtension.java | 77 - ...ignatureAlgorithmEnforcementValidator.java | 54 - .../OBCibaGrantHandler.java | 69 - .../src/main/resources/findbugs-exclude.xml | 21 - .../src/main/resources/findbugs-include.xml | 23 - ...ARequestObjectValidationExtensionTest.java | 93 - ...tureAlgorithmEnforcementValidatorTest.java | 124 - .../accelerator/ciba/TestConstants.java | 66 - .../src/test/resources/testng.xml | 29 - .../pom.xml | 306 - .../common/caching/OpenBankingBaseCache.java | 212 - .../caching/OpenBankingBaseCacheKey.java | 38 - .../config/OpenBankingConfigParser.java | 1483 --- .../OpenBankingConfigurationService.java | 47 - .../OpenBankingConfigurationServiceImpl.java | 87 - .../common/config/TextFileReader.java | 118 - .../common/constant/OpenBankingConstants.java | 272 - .../common/error/OpenBankingErrorCodes.java | 60 - .../executor/DefaultOBEventExecutor.java | 33 - .../event/executor/OBEventExecutor.java | 34 - .../common/event/executor/OBEventQueue.java | 66 - .../common/event/executor/OBQueueWorker.java | 69 - .../common/event/executor/model/OBEvent.java | 46 - .../CertificateValidationException.java | 83 - .../exception/ConsentManagementException.java | 33 - .../ConsentManagementRuntimeException.java | 35 - .../exception/OBThrottlerException.java | 34 - .../exception/OpenBankingException.java | 34 - .../OpenBankingRuntimeException.java | 50 - .../exception/TPPValidationException.java | 33 - .../identity/ApplicationIdentityService.java | 66 - .../common/identity/IdentityConstants.java | 90 - .../common/identity/cache/JWKSetCache.java | 42 - .../common/identity/cache/JWKSetCacheKey.java | 63 - .../base/OpenBankingIdentityBaseCache.java | 77 - .../identity/retriever/JWKRetriever.java | 127 - .../retriever/ServerIdentityRetriever.java | 153 - .../sp/CommonServiceProviderRetriever.java | 82 - .../internal/OpenBankingCommonDataHolder.java | 126 - .../OpenBankingCommonServiceComponent.java | 84 - .../common/model/PSD2RoleEnum.java | 49 - .../persistence/JDBCPersistenceManager.java | 160 - .../JDBCRetentionDataPersistenceManager.java | 161 - .../common/util/AnalyticsLogsUtils.java | 59 - .../accelerator/common/util/CarbonUtils.java | 97 - .../common/util/CertificateUtils.java | 105 - .../accelerator/common/util/DatabaseUtil.java | 80 - .../common/util/ErrorConstants.java | 344 - .../accelerator/common/util/Generated.java | 28 - .../common/util/HTTPClientUtils.java | 205 - .../accelerator/common/util/JWTUtils.java | 254 - .../common/util/OpenBankingUtils.java | 117 - .../common/util/SPQueryExecutorUtil.java | 94 - .../common/util/SecurityUtils.java | 80 - .../common/util/ServiceProviderUtils.java | 47 - .../extractor/CertificateContent.java | 106 - .../CertificateContentExtractor.java | 107 - .../certificate/extractor/common/NcaId.java | 40 - .../certificate/extractor/common/NcaName.java | 41 - .../extractor/common/PSD2Constants.java | 43 - .../extractor/common/PSD2QCStatement.java | 119 - .../extractor/common/PSD2QCType.java | 64 - .../certificate/extractor/common/PSPRole.java | 88 - .../extractor/common/PSPRoles.java | 62 - .../extractor/error/CertValidationErrors.java | 42 - .../validator/OpenBankingValidator.java | 75 - .../validator/OsgiServiceDiscoverer.java | 38 - .../annotation/RequiredParameter.java | 52 - .../annotation/RequiredParameters.java | 36 - .../validator/annotation/ValidAudience.java | 52 - .../annotation/ValidScopeFormat.java | 50 - .../validator/impl/AudienceValidator.java | 102 - .../impl/MandatoryParameterValidator.java | 68 - .../common/validator/impl/ScopeValidator.java | 72 - .../src/main/resources/findbugs-exclude.xml | 41 - .../src/main/resources/findbugs-include.xml | 35 - .../common/test/OBConfigParserTests.java | 470 - .../test/config/TextFileReaderTest.java | 73 - .../event/executor/OBEventExecutorTests.java | 92 - .../test/util/CertificateUtilsTest.java | 86 - .../common/test/util/CommonTestUtil.java | 219 - .../common/test/util/HTTPClientUtilsTest.java | 71 - .../common/test/util/JWTUtilsTest.java | 72 - .../test/util/OpenBankingUtilsTest.java | 165 - .../test/util/SPQueryExecutorUtilTest.java | 100 - .../common/test/util/SecurityUtilsTest.java | 65 - .../CertificateContentExtractorTest.java | 65 - .../testutils/JWTUtilsTestDataProvider.java | 132 - .../test/validator/LogicValidatorsTest.java | 88 - .../test/validator/ModelValidatorsTest.java | 131 - .../resources/SampleChildRequestObject.java | 38 - .../resources/SampleDifferentClass.java | 58 - .../resources/SampleRequestObject.java | 45 - .../resources/ValidatorTestDataProvider.java | 155 - .../src/test/resources/open-banking-empty.xml | 17 - .../src/test/resources/open-banking.xml | 284 - .../src/test/resources/test-data.json | 11 - .../src/test/resources/testFile.js | 78 - .../src/test/resources/testng.xml | 68 - .../src/test/resources/wso2carbon.jks | Bin 98874 -> 0 bytes .../pom.xml | 208 - .../constant/AuthPublisherConstants.java | 40 - .../extension/AbstractAuthDataPublisher.java | 31 - .../extension/DefaultAuthDataPublisher.java | 36 - ...AuthenticationDataPublisherDataHolder.java | 64 - ...ticationDataPublisherServiceComponent.java | 106 - .../AuthenticationDataPublisherService.java | 35 - ...uthenticationDataPublisherServiceImpl.java | 111 - .../src/main/resources/findbugs-include.xml | 23 - .../service/OBAuthDataPublisherTest.java | 206 - .../src/test/resources/testng.xml | 27 - .../pom.xml | 201 - .../common/DataPublisherFactory.java | 43 - .../publisher/common/DataPublisherPool.java | 36 - .../data/publisher/common/EventQueue.java | 65 - .../common/OBThriftDataPublisher.java | 288 - .../common/OpenBankingDataPublisher.java | 29 - .../data/publisher/common/QueueWorker.java | 61 - .../constants/DataPublishingConstants.java | 41 - .../internal/OBAnalyticsDataHolder.java | 117 - .../internal/OBAnalyticsServiceComponent.java | 73 - .../common/model/OBAnalyticsEvent.java | 53 - .../common/util/OBDataPublisherUtil.java | 92 - .../src/main/resources/findbugs-exclude.xml | 28 - .../src/main/resources/findbugs-include.xml | 22 - .../common/DataPublisherPoolTest.java | 87 - .../common/OBAnalyticsEventQueueTest.java | 127 - .../common/OBThriftDataPublisherTest.java | 387 - .../src/test/resources/testng.xml | 29 - .../pom.xml | 36 - .../pom.xml | 301 - .../cache/CertificateRevocationCache.java | 78 - .../gateway/cache/GatewayCache.java | 65 - .../gateway/cache/GatewayCacheKey.java | 62 - .../gateway/cache/TppValidationCache.java | 78 - .../executor/core/AbstractRequestRouter.java | 84 - .../executor/core/DefaultRequestRouter.java | 72 - .../core/OBExtensionListenerImpl.java | 280 - .../core/OpenBankingGatewayExecutor.java | 56 - .../gateway/executor/dcr/DCRExecutor.java | 1032 -- .../OpenBankingExecutorException.java | 78 - .../APIResourceAccessValidationExecutor.java | 104 - .../executor/CommonReportingDataExecutor.java | 150 - .../consent/ConsentEnforcementExecutor.java | 312 - .../error/handler/OBDefaultErrorHandler.java | 193 - .../CertRevocationValidationExecutor.java | 260 - .../executor/MTLSEnforcementExecutor.java | 116 - .../UserPermissionValidationExecutor.java | 173 - .../executor/APITPPValidationExecutor.java | 196 - .../executor/DCRTPPValidationExecutor.java | 191 - .../JwsRequestSignatureHandlingExecutor.java | 627 - .../JwsResponseSignatureHandlingExecutor.java | 256 - .../executor/model/OBAPIRequestContext.java | 263 - .../executor/model/OBAPIResponseContext.java | 199 - .../model/OpenBankingExecutorError.java | 106 - .../executor/model/RevocationStatus.java | 43 - .../executor/revocation/CRLValidator.java | 383 - .../executor/revocation/OCSPValidator.java | 416 - .../revocation/RevocationValidator.java | 48 - .../service/CertValidationService.java | 245 - .../service/RevocationValidatorFactory.java | 73 - .../service/TPPValidationService.java | 60 - .../util/CertificateValidationUtils.java | 300 - .../GatewayClientAuthenticationHandler.java | 79 - .../handler/JwsResponseSignatureHandler.java | 293 - .../gateway/internal/GatewayDataHolder.java | 372 - .../internal/GatewayServiceComponent.java | 87 - .../internal/TPPCertValidatorComponent.java | 129 - .../internal/TPPCertValidatorDataHolder.java | 476 - .../gateway/mediator/BasicAuthMediator.java | 61 - .../reporter/OBAnalyticsMetricReporter.java | 67 - .../gateway/reporter/OBCounterMetric.java | 111 - .../reporter/OBTimestampPublisher.java | 82 - .../DisputeResolutionSynapseHandler.java | 187 - .../throttling/OBThrottlingExtensionImpl.java | 70 - .../throttling/ThrottleDataPublisher.java | 32 - .../gateway/util/GatewayConstants.java | 145 - .../util/GatewaySignatureHandlingUtils.java | 161 - .../gateway/util/GatewayUtils.java | 829 -- .../gateway/util/IdempotencyConstants.java | 60 - .../src/main/resources/findbugs-exclude.xml | 40 - .../src/main/resources/findbugs-include.xml | 23 - .../core/DefaultRequestRouterTest.java | 119 - .../executor/core/TestOBExtensionImpl.java | 219 - .../gateway/executor/core/UtilityTest.java | 422 - .../gateway/executor/dcr/DCRExecutorTest.java | 838 -- .../CommonReportingDataExecutorTest.java | 125 - .../impl/consent/TestEnforcementExecutor.java | 110 - .../handler/OBDefaultErrorHandlerTest.java | 106 - .../CertRevocationValidationExecutorTest.java | 229 - .../executor/MTLSEnforcementExecutorTest.java | 45 - .../UserPermissionValidationExecutorTest.java | 94 - .../APITPPValidationExecutorTest.java | 159 - .../DCRTPPValidationExecutorTest.java | 90 - ...RequestSignatureHandlingExecutorTests.java | 299 - ...esponseSignatureHandlingExecutorTests.java | 292 - .../executor/revocation/CRLValidatorTest.java | 314 - .../revocation/OCSPValidatorTest.java | 224 - .../service/CertValidationServiceTest.java | 351 - .../RevocationValidatorFactoryTest.java | 67 - .../gateway/executor/test/TestConstants.java | 61 - .../test/executor/MockOBExecutor.java | 60 - .../executor/test/executor/TestRouter.java | 44 - .../gateway/executor/test/util/TestUtil.java | 52 - .../util/CertificateValidationUtilsTest.java | 184 - .../executor/util/TestValidationUtil.java | 384 - .../JwsResponseSignatureHandlerTest.java | 163 - .../TPPCertValidatorDataHolderTest.java | 71 - .../mediator/BasicAuthMediatorTests.java | 387 - .../OBAnalyticsMetricReporterTest.java | 68 - .../reporter/TimestampPublishingTest.java | 96 - .../DisputeResolutionSynapseHandlerTest.java | 193 - .../OBThrottlingExtensionImplTest.java | 53 - .../ThrottleDataPublisherTestImpl.java | 39 - .../gateway/util/GatewayUtilsTest.java | 195 - .../src/test/resources/client-truststore.jks | Bin 104882 -> 0 bytes .../src/test/resources/open-banking.xml | 153 - .../src/test/resources/test_crl_entries.pem | 1225 -- .../src/test/resources/testng.xml | 76 - .../src/test/resources/wso2carbon.jks | Bin 98874 -> 0 bytes .../pom.xml | 377 - .../app2app/App2AppAuthenticator.java | 248 - .../App2AppAuthenticatorConstants.java | 51 - .../identity/app2app/cache/JTICache.java | 73 - .../exception/JWTValidationException.java | 38 - .../model/DeviceVerificationToken.java | 165 - .../DeviceVerificationTokenConstants.java | 34 - .../app2app/utils/App2AppAuthUtils.java | 149 - .../app2app/validations/DigestValidator.java | 105 - .../app2app/validations/ExpiryValidator.java | 52 - .../app2app/validations/JTIValidator.java | 69 - .../app2app/validations/NBFValidator.java | 55 - .../PublicKeySignatureValidator.java | 77 - .../annotations/ValidateDigest.java | 48 - .../annotations/ValidateExpiry.java | 48 - .../validations/annotations/ValidateJTI.java | 48 - .../validations/annotations/ValidateNBF.java | 48 - .../annotations/ValidateSignature.java | 48 - .../App2AppValidationOrder.java | 34 - .../OpenBankingAuthenticationWorker.java | 33 - ...enBankingAuthenticationWorkerFunction.java | 34 - ...nkingAuthenticationWorkerFunctionImpl.java | 50 - .../authz/request/OBOAuthAuthzRequest.java | 148 - .../DefaultOBRequestObjectValidator.java | 93 - .../OBRequestObjectValidationExtension.java | 151 - .../validator/OBRequestObjectValidator.java | 51 - .../SigningAlgorithmValidator.java | 88 - .../annotations/ValidSigningAlgorithm.java | 49 - .../validator/models/OBRequestObject.java | 142 - .../validator/models/ValidationResponse.java | 52 - .../OBCodeResponseTypeHandlerExtension.java | 87 - .../OBHybridResponseTypeHandlerExtension.java | 85 - .../handler/OBResponseTypeHandler.java | 45 - .../OBDefaultResponseTypeHandlerImpl.java | 169 - .../OBCodeResponseTypeValidator.java | 80 - .../OBHybridResponseTypeValidator.java | 95 - .../OBIdentifierAuthenticator.java | 608 - .../constants/IdentifierHandlerConstants.java | 40 - .../util/OBIdentifierAuthUtil.java | 70 - ...faultOBRequestUriRequestObjectBuilder.java | 197 - .../identity/cache/IdentityCache.java | 63 - .../identity/cache/IdentityCacheKey.java | 62 - .../identity/claims/OBClaimProvider.java | 63 - .../claims/OBDefaultClaimProvider.java | 151 - .../OBDefaultOIDCClaimsCallbackHandler.java | 170 - .../claims/RoleClaimProviderImpl.java | 101 - .../OBMutualTLSClientAuthenticator.java | 99 - .../common/IdentityServiceExporter.java | 37 - .../validationgroups/AttributeChecks.java | 28 - .../validationgroups/MandatoryChecks.java | 28 - .../validationgroups/SignatureCheck.java | 28 - .../validationgroups/ValidityChecks.java | 28 - .../dcr/exception/DCRValidationException.java | 63 - .../identity/dcr/model/RegistrationError.java | 47 - .../dcr/model/RegistrationRequest.java | 366 - .../dcr/model/RegistrationResponse.java | 220 - .../dcr/model/SoftwareStatementBody.java | 145 - .../identity/dcr/utils/ValidatorUtils.java | 102 - .../dcr/validation/AlgorithmValidator.java | 89 - .../dcr/validation/DCRCommonConstants.java | 46 - .../DefaultRegistrationValidatorImpl.java | 105 - .../dcr/validation/IssuerValidator.java | 73 - .../dcr/validation/RegistrationValidator.java | 91 - .../validation/RequiredParamsValidator.java | 157 - .../dcr/validation/SignatureValidator.java | 110 - .../annotation/ValidateAlgorithm.java | 53 - .../validation/annotation/ValidateIssuer.java | 52 - .../annotation/ValidateRequiredParams.java | 46 - .../annotation/ValidateSignature.java | 50 - .../validationgroups/AttributeChecks.java | 29 - .../validationgroups/MandatoryChecks.java | 29 - .../validationgroups/SignatureCheck.java | 29 - .../validationgroups/ValidationOrder.java | 29 - .../validationgroups/ValidityChecks.java | 29 - .../validationorder/ValidationOrder.java | 32 - .../resolution/DisputeResolutionFilter.java | 148 - .../OBAuthorizationCodeGrantHandler.java | 83 - .../OBClientCredentialsGrantHandler.java | 85 - .../type/handlers/OBPasswordGrantHandler.java | 72 - .../type/handlers/OBRefreshGrantHandler.java | 120 - .../identity/idtoken/OBIDTokenBuilder.java | 293 - .../OBDefaultIntrospectionDataProvider.java | 45 - .../OBIntrospectionDataProvider.java | 65 - .../IdentityExtensionsDataHolder.java | 459 - .../IdentityExtensionsServiceComponent.java | 279 - .../keyidprovider/OBKeyIDProvider.java | 70 - .../listener/TokenRevocationListener.java | 108 - .../AbstractApplicationUpdater.java | 72 - .../application/ApplicationUpdaterImpl.java | 278 - .../OBApplicationManagementListener.java | 179 - .../validator/PushAuthRequestValidator.java | 304 - .../constants/PushAuthRequestConstants.java | 58 - .../PushAuthRequestValidatorException.java | 79 - .../model/PushAuthErrorResponse.java | 48 - .../util/PushAuthRequestValidatorUtils.java | 648 - .../metadata/extension/SPMetadataFilter.java | 35 - .../impl/DefaultSPMetadataFilter.java | 51 - .../identity/token/DefaultTokenFilter.java | 88 - .../identity/token/TokenFilter.java | 298 - .../token/util/TokenFilterException.java | 69 - .../ClientAuthenticatorValidator.java | 126 - .../validators/MTLSCertificateValidator.java | 63 - .../validators/MTLSEnforcementValidator.java | 66 - .../validators/OBIdentityFilterValidator.java | 34 - ...ignatureAlgorithmEnforcementValidator.java | 108 - .../token/wrapper/RequestWrapper.java | 130 - .../token/wrapper/ResponseWrapper.java | 84 - .../util/ClientAuthenticatorEnum.java | 50 - .../identity/util/HTTPClientUtils.java | 198 - .../util/IdentityCommonConstants.java | 112 - .../identity/util/IdentityCommonHelper.java | 271 - .../identity/util/IdentityCommonUtil.java | 436 - .../src/main/resources/findbugs-exclude.xml | 21 - .../src/main/resources/findbugs-include.xml | 23 - .../identity/HTTPClientUtilsTest.java | 70 - .../app2app/App2AppAuthUtilsTest.java | 194 - .../app2app/App2AppAuthValidationTest.java | 179 - .../app2app/App2AppAuthenticatorTest.java | 281 - .../App2AppAuthenticatorTestDataProvider.java | 62 - .../App2AppUtilsTestJWTDataProvider.java | 83 - .../ApplicationManagementListenerTest.java | 302 - ...gAuthenticationWorkerFunctionImplTest.java | 69 - .../request/OBOAuthAuthzRequestTest.java | 328 - .../DefaultOBRequestObjectValidatorTest.java | 161 - .../validator/ReqObjectTestDataProvider.java | 133 - .../validator/RequestObjectValidatorTest.java | 141 - .../handler/ResponseTypeHandlerTest.java | 132 - .../OBCodeResponseTypeValidatorTest.java | 60 - .../OBHybridResponseTypeValidatorTest.java | 103 - .../OBIdentifierAuthenticatorTest.java | 1226 -- .../OBIdentifierAuthenticatorTestData.java | 48 - ...tOBRequestUriRequestObjectBuilderTest.java | 149 - ...BDefaultOIDCClaimsCallbackHandlerTest.java | 146 - .../claims/RoleClaimProviderImplTest.java | 152 - .../OBMutualTLSClientAuthenticatorTest.java | 190 - .../dcr/DCRExtendedValidatorTest.java | 103 - .../identity/dcr/DCRValidationTest.java | 301 - .../identity/dcr/DCRValidationUtilTest.java | 69 - .../dcr/util/ExtendedRegistrationRequest.java | 85 - .../util/ExtendedRegistrationResponse.java | 49 - .../util/ExtendedSoftwareStatementBody.java | 42 - .../dcr/util/ExtendedValidatorImpl.java | 79 - .../dcr/util/RegistrationTestConstants.java | 99 - .../validation/AlgorithmValidatorTest.java | 108 - .../dcr/validation/IssuerValidatorTest.java | 81 - .../RequiredParamsValidatorTest.java | 160 - .../DisputeResolutionFilterTest.java | 116 - .../idtoken/OBIDTokenBuilderTests.java | 259 - .../listener/TokenRevocationListenerTest.java | 64 - .../PushAuthRequestValidatorTest.java | 421 - .../util/test/jwt/builder/TestJwtBuilder.java | 401 - .../constants/TestJwtBuilderConstants.java | 40 - .../identity/token/TokenFilterTest.java | 348 - .../identity/token/util/TestConstants.java | 104 - .../identity/token/util/TestUtil.java | 62 - .../ClientAuthenticatorValidatorTest.java | 217 - .../MTLSCertificateValidatorTest.java | 103 - .../MTLSEnforcementValidatorTest.java | 251 - ...tureAlgorithmEnforcementValidatorTest.java | 215 - .../src/test/resources/common.auth.script.js | 61 - .../src/test/resources/testng.xml | 151 - .../src/test/resources/wso2carbon.jks | Bin 98874 -> 0 bytes .../pom.xml | 216 - .../keymanager/KeyManagerUtil.java | 225 - .../keymanager/OBKeyManagerConfiguration.java | 136 - .../keymanager/OBKeyManagerConstants.java | 28 - .../OBKeyManagerExtensionInterface.java | 78 - .../keymanager/OBKeyManagerImpl.java | 470 - .../internal/KeyManagerDataHolder.java | 158 - .../internal/KeyManagerServiceComponent.java | 127 - .../src/main/resources/findbugs-include.xml | 34 - .../keymanager/KeyManagerTest.java | 521 - .../keymanager/KeyManagerUtilTest.java | 202 - .../src/test/resources/testng.xml | 28 - .../pom.xml | 105 - .../OBOAuthClientAuthenticatorProxy.java | 106 - .../src/main/resources/findbugs-include.xml | 23 - .../pom.xml | 36 - .../pom.xml | 191 - .../service/activator/OBServiceObserver.java | 30 - .../activator/internal/ServiceObservable.java | 64 - .../internal/ServiceRegisterComponent.java | 71 - .../src/main/resources/findbugs-include.xml | 22 - .../internal/ServiceObservableTest.java | 54 - .../src/test/resources/testng.xml | 27 - .../pom.xml | 251 - .../admin/builder/ConsentAdminBuilder.java | 48 - .../impl/DefaultConsentAdminHandler.java | 417 - .../admin/model/ConsentAdminData.java | 117 - .../admin/model/ConsentAdminHandler.java | 63 - .../builder/ConsentStepsBuilder.java | 77 - .../impl/CIBAConsentPersistStep.java | 69 - .../impl/CIBAConsentRetrievalStep.java | 74 - .../impl/DefaultConsentPersistStep.java | 126 - .../impl/DefaultConsentRetrievalStep.java | 122 - .../impl/NonRegulatoryConsentStep.java | 91 - .../impl/RequestObjectCheckStep.java | 57 - .../authorize/model/ConsentData.java | 208 - .../authorize/model/ConsentPersistData.java | 109 - .../authorize/model/ConsentPersistStep.java | 35 - .../authorize/model/ConsentRetrievalStep.java | 38 - .../authorize/utils/ConsentRetrievalUtil.java | 700 - .../impl/ConsentMgrAuthServletImpl.java | 104 - .../impl/ISDefaultAuthServletImpl.java | 125 - .../impl/OBDefaultAuthServletImpl.java | 96 - .../authservlet/impl/util/Constants.java | 35 - .../authservlet/impl/util/Utils.java | 296 - .../model/OBAuthServletInterface.java | 44 - .../authenticator/CIBAPushAuthenticator.java | 365 - .../CIBAPushAuthenticatorConstants.java | 49 - ...CIBAAuthenticationEndpointDefaultImpl.java | 34 - ...BAAuthenticationEndpointErrorResponse.java | 50 - .../CIBAAuthenticationEndpointInterface.java | 34 - .../extensions/common/AuthErrorCode.java | 112 - .../extensions/common/ConsentCache.java | 189 - .../extensions/common/ConsentException.java | 130 - .../common/ConsentExtensionConstants.java | 227 - .../common/ConsentExtensionExporter.java | 81 - .../common/ConsentExtensionUtils.java | 417 - .../extensions/common/ConsentServiceUtil.java | 34 - .../extensions/common/ResponseStatus.java | 258 - .../AcceleratorConsentExtensionFactory.java | 59 - .../idempotency/IdempotencyConstants.java | 42 - .../IdempotencyValidationException.java | 35 - .../IdempotencyValidationResult.java | 82 - .../IdempotencyValidationUtils.java | 131 - .../idempotency/IdempotencyValidator.java | 301 - .../ConsentAmendmentHistoryEventExecutor.java | 92 - .../event/executors/VRPEventExecutor.java | 119 - .../internal/ConsentExtensionsComponent.java | 108 - .../internal/ConsentExtensionsDataHolder.java | 135 - .../manage/builder/ConsentManageBuilder.java | 48 - .../AccountConsentManageRequestHandler.java | 343 - .../manage/impl/CofConsentRequestHandler.java | 127 - .../impl/ConsentManageRequestHandler.java | 47 - .../impl/DefaultConsentManageHandler.java | 162 - .../impl/PaymentConsentRequestHandler.java | 167 - .../manage/impl/VRPConsentRequestHandler.java | 342 - .../manage/model/ConsentManageData.java | 125 - .../manage/model/ConsentManageHandler.java | 93 - .../manage/model/PeriodicLimit.java | 202 - .../validator/CofConsentRequestValidator.java | 152 - .../PaymentsConsentRequestValidator.java | 61 - .../validator/VRPConsentRequestValidator.java | 815 -- .../extensions/util/ConsentManageUtil.java | 664 - .../util/DebtorAccountSchemeNameEnum.java | 61 - .../util/PaymentPayloadValidator.java | 97 - .../extensions/util/PeriodicTypesEnum.java | 76 - .../util/PeriodicalConsentJobActivator.java | 106 - .../jobs/ExpiredConsentStatusUpdateJob.java | 158 - .../util/jobs/RetentionDatabaseSyncJob.java | 68 - .../PeriodicalConsentJobScheduler.java | 102 - .../builder/ConsentValidateBuilder.java | 57 - .../impl/DefaultConsentValidator.java | 508 - ...mentFundsConfirmationPayloadValidator.java | 84 - .../PaymentSubmissionPayloadValidator.java | 122 - .../impl/VRPSubmissionPayloadValidator.java | 516 - .../validate/model/ConsentValidateData.java | 125 - .../model/ConsentValidationResult.java | 99 - .../validate/model/ConsentValidator.java | 38 - .../validate/util/ConsentValidatorUtil.java | 302 - .../util/PaymentSubmissionValidationUtil.java | 103 - .../src/main/resources/findbugs-exclude.xml | 20 - .../src/main/resources/findbugs-include.xml | 22 - .../flow/ConsentPersistStepTests.java | 258 - .../flow/ConsentExtensionDataProvider.java | 36 - .../flow/VRPConsentRetrievalStepTest.java | 282 - .../flow/VRPConsentRetrievalUtilTest.java | 333 - .../authservlet/impl/AuthServletTest.java | 181 - .../impl/ConsentMgrAuthServletImplTest.java | 88 - .../CIBAPushAuthenticatorTests.java | 258 - .../IdempotencyValidatorTests.java | 303 - ...entAmendmentHistoryEventExecutorTests.java | 149 - .../manage/vrp/VRPConsentHandlerTest.java | 149 - .../vrp/VRPConsentRequestValidatorTest.java | 2009 --- .../manage/vrp/VRPTestConstants.java | 1122 -- .../utils/AuthServletTestConstants.java | 256 - .../utils/ConsentAuthorizeTestConstants.java | 405 - .../utils/ConsentExtensionDataProvider.java | 49 - .../utils/ConsentExtensionTestConstants.java | 65 - .../utils/ConsentExtensionTestUtils.java | 98 - .../utils/ConsentValidateTestConstants.java | 1091 -- .../validate/VRPSubmissionTest.java | 887 -- .../src/test/resources/testng.xml | 37 - .../pom.xml | 207 - .../consent/mgt/dao/ConsentCoreDAO.java | 506 - .../dao/constants/ConsentMgtDAOConstants.java | 139 - .../OBConsentDataDeletionException.java | 34 - .../OBConsentDataInsertionException.java | 34 - .../OBConsentDataRetrievalException.java | 34 - .../OBConsentDataUpdationException.java | 34 - .../mgt/dao/impl/ConsentCoreDAOImpl.java | 2151 --- .../mgt/dao/impl/MssqlConsentCoreDAOImpl.java | 360 - .../dao/impl/OracleConsentCoreDAOImpl.java | 359 - .../mgt/dao/models/AuthorizationResource.java | 107 - .../mgt/dao/models/ConsentAttributes.java | 61 - .../consent/mgt/dao/models/ConsentFile.java | 59 - .../dao/models/ConsentHistoryResource.java | 97 - .../dao/models/ConsentMappingResource.java | 95 - .../mgt/dao/models/ConsentResource.java | 177 - .../dao/models/ConsentStatusAuditRecord.java | 119 - .../dao/models/DetailedConsentResource.java | 201 - .../persistence/ConsentStoreInitializer.java | 118 - .../queries/ConsentMgtCommonDBQueries.java | 474 - .../dao/queries/ConsentMgtMssqlDBQueries.java | 199 - .../queries/ConsentMgtOracleDBQueries.java | 202 - .../queries/ConsentMgtPostgresDBQueries.java | 229 - .../mgt/dao/utils/ConsentDAOUtils.java | 276 - .../src/main/resources/findbugs-exclude.xml | 43 - .../src/main/resources/findbugs-include.xml | 22 - .../mgt/dao/impl/OBConsentMgtDAOTests.java | 2546 ---- .../mgt/dao/util/ConsentMgtDAOTestData.java | 558 - .../consent/mgt/dao/util/DAOUtils.java | 64 - .../src/test/resources/dbScripts/h2.sql | 138 - .../src/test/resources/testng.xml | 27 - .../pom.xml | 238 - .../mgt/service/ConsentCoreService.java | 732 -- .../ConsentCoreServiceConstants.java | 71 - .../service/impl/ConsentCoreServiceImpl.java | 2848 ---- .../impl/ConsentStateChangeListenerImpl.java | 71 - .../internal/ConsentManagementDataHolder.java | 74 - .../ConsentManagementServiceComponent.java | 130 - .../listener/ConsentStateChangeListener.java | 45 - .../src/main/resources/findbugs-include.xml | 17 - .../impl/OBConsentMgtCoreServiceTests.java | 3401 ----- .../util/ConsentMgtServiceTestData.java | 823 -- .../src/test/resources/testng.xml | 27 - .../pom.xml | 237 - .../constants/EventNotificationConstants.java | 118 - .../service/dao/AggregatedPollingDAO.java | 105 - .../service/dao/AggregatedPollingDAOImpl.java | 448 - .../service/dao/EventPublisherDAO.java | 44 - .../service/dao/EventPublisherDAOImpl.java | 89 - .../service/dao/EventSubscriptionDAO.java | 120 - .../service/dao/EventSubscriptionDAOImpl.java | 309 - .../dao/EventSubscriptionSqlStatements.java | 71 - ...MSSQLNotificationPollingSqlStatements.java | 35 - .../dao/NotificationPollingSqlStatements.java | 60 - .../NotificationPublisherSqlStatements.java | 40 - .../PostgreSqlEventSubscriptionDAOImpl.java | 197 - .../service/dao/PostgreSqlPollingDAOImpl.java | 250 - .../dto/EventNotificationErrorDTO.java | 53 - .../service/dto/EventPollingDTO.java | 76 - .../service/dto/EventSubscriptionDTO.java | 54 - .../service/dto/NotificationCreationDTO.java | 59 - .../service/dto/NotificationDTO.java | 70 - .../OBEventNotificationException.java | 35 - .../DefaultEventCreationServiceHandler.java | 105 - .../DefaultEventPollingServiceHandler.java | 160 - ...efaultEventSubscriptionServiceHandler.java | 345 - .../handler/EventCreationServiceHandler.java | 38 - ...NotificationPersistenceServiceHandler.java | 61 - .../handler/EventPollingServiceHandler.java | 48 - .../EventSubscriptionServiceHandler.java | 94 - .../internal/EventNotificationComponent.java | 102 - .../internal/EventNotificationDataHolder.java | 75 - .../model/AggregatedPollingResponse.java | 64 - .../service/model/EventSubscription.java | 99 - .../service/model/Notification.java | 126 - .../service/model/NotificationError.java | 52 - .../service/model/NotificationEvent.java | 64 - .../EventPollingStoreInitializer.java | 76 - .../EventPublisherStoreInitializer.java | 74 - .../EventSubscriptionStoreInitializer.java | 75 - .../model/RealtimeEventNotification.java | 59 - ...timeEventNotificationRequestGenerator.java | 40 - .../EventNotificationProducerService.java | 81 - ...ealtimeEventNotificationLoaderService.java | 86 - ...timeEventNotificationRequestGenerator.java | 48 - ...ealtimeEventNotificationSenderService.java | 195 - ...EventNotificationConsumerJobActivator.java | 92 - .../job/EventNotificationConsumerJob.java | 91 - ...EventNotificationConsumerJobScheduler.java | 77 - .../response/EventCreationResponse.java | 55 - .../response/EventPollingResponse.java | 55 - .../response/EventSubscriptionResponse.java | 54 - .../DefaultEventNotificationGenerator.java | 87 - .../service/service/EventCreationService.java | 121 - .../service/EventNotificationGenerator.java | 48 - .../service/service/EventPollingService.java | 153 - .../service/EventSubscriptionService.java | 241 - .../ExtendedEventNotificationGenerator.java | 46 - .../util/EventNotificationServiceUtil.java | 171 - .../src/main/resources/findbugs-exclude.xml | 46 - .../src/main/resources/findbugs-include.xml | 22 - .../EventNotificationTestConstants.java | 72 - .../dao/AggregatedPollingDAOImplTests.java | 263 - .../dao/EventPublisherDAOImplTests.java | 96 - .../dao/EventSubscriptionDAOImplTests.java | 390 - ...stgreSqlEventSubscriptionDAOImplTests.java | 382 - ...efaultEventCreationServiceHandlerTest.java | 162 - ...efaultEventPollingServiceHandlerTests.java | 173 - ...tEventSubscriptionServiceHandlerTests.java | 298 - ...icationPersistenceServiceHandlerTests.java | 68 - ...ventNotificationPayloadGeneratorTests.java | 61 - ...EventNotificationProducerServiceTests.java | 99 - ...imeEventNotificationLoaderServiceTest.java | 117 - ...meEventNotificationSenderServiceTests.java | 138 - ...efaultEventNotificationGeneratorTests.java | 71 - .../service/EventCreationServiceTests.java | 102 - .../service/EventPollingServiceTests.java | 113 - .../EventSubscriptionServiceTests.java | 171 - .../utils/EventNotificationTestUtils.java | 275 - .../src/test/resources/testng.xml | 42 - .../pom.xml | 189 - .../throttler/dao/OBThrottlerDAO.java | 99 - .../constants/OBThrottlerDAOConstants.java | 41 - .../OBThrottlerDataDeletionException.java | 36 - .../OBThrottlerDataInsertionException.java | 36 - .../OBThrottlerDataRetrievalException.java | 36 - .../OBThrottlerDataUpdationException.java | 36 - .../dao/impl/OBThrottlerDAOImpl.java | 232 - .../dao/model/ThrottleDataModel.java | 100 - .../dao/persistence/DataStoreInitializer.java | 44 - .../DataStoreInitializerFactory.java | 54 - .../dao/queries/OBThrottlerSQLStatements.java | 51 - .../src/main/resources/findbugs-exclude.xml | 21 - .../src/main/resources/findbugs-include.xml | 22 - .../dao/impl/OBThrottlerDAOTests.java | 275 - .../dao/util/OBThrottlerDAOTestData.java | 61 - .../dao/util/OBThrottlerDAOUtils.java | 63 - .../src/test/resources/dbScripts/h2.sql | 8 - .../src/test/resources/testng.xml | 27 - .../pom.xml | 213 - .../throttler/service/OBThrottleService.java | 245 - .../OBThrottlerServiceConstants.java | 35 - .../internal/OBThrottlerDataHolder.java | 52 - .../internal/OBThrottlerServiceComponent.java | 87 - .../src/main/resources/findbugs-include.xml | 22 - .../service/OBThrottleServiceTests.java | 303 - .../util/OBThrottleServiceTestData.java | 63 - .../src/test/resources/testng.xml | 27 - .../.openapi-generator-ignore | 25 - .../findbugs-exclude.xml | 24 - .../pom.xml | 166 - .../api/ApplicationInformationApi.java | 93 - .../ApplicationBulkMetadataSuccessDTO.java | 88 - .../model/ApplicationInfoErrorDTO.java | 126 - .../model/ApplicationMetadataResourceDTO.java | 133 - .../ApplicationSingleMetadataSuccessDTO.java | 78 - .../api/constants/MetaDataSQLStatements.java | 38 - .../endpoint/api/data/MetaDataDAOImpl.java | 84 - .../ApplicationInformationApiServiceImpl.java | 254 - .../info/endpoint/api/utils/MappingUtil.java | 150 - .../main/resources/application-info-300.yaml | 147 - .../src/main/resources/findbugs-exclude.xml | 30 - .../src/main/resources/findbugs-include.xml | 22 - .../webapp/META-INF/webapp-classloading.xml | 34 - .../src/main/webapp/WEB-INF/beans.xml | 34 - .../src/main/webapp/WEB-INF/web.xml | 70 - .../pom.xml | 124 - .../impl/api/CIBAAuthenticationEndpoint.java | 685 - .../CIBAAuthenticationEndpointConstants.java | 118 - .../CIBAAuthenticationEndpointException.java | 79 - .../src/main/resources/findbugs-include.xml | 22 - .../src/main/webapp/META-INF/MANIFEST.mf | 1 - .../webapp/META-INF/webapp-classloading.xml | 35 - .../src/main/webapp/WEB-INF/beans.xml | 34 - .../src/main/webapp/WEB-INF/web.xml | 69 - .../pom.xml | 181 - .../endpoint/api/ConsentAdminEndpoint.java | 204 - .../api/ConsentAuthorizeEndpoint.java | 361 - .../endpoint/api/ConsentManageEndpoint.java | 228 - .../api/ConsentValidationEndpoint.java | 200 - .../error/ConsentThrowableMapper.java | 62 - .../endpoint/util/ConsentConstants.java | 52 - .../consent/endpoint/util/ConsentUtils.java | 358 - .../src/main/resources/findbugs-exclude.xml | 21 - .../src/main/resources/findbugs-include.xml | 22 - .../webapp/META-INF/webapp-classloading.xml | 35 - .../src/main/webapp/WEB-INF/beans.xml | 39 - .../src/main/webapp/WEB-INF/web.xml | 71 - .../pom.xml | 203 - .../dto/RegistrationErrorDTO.java | 74 - .../endpoint/impl/RegistrationConstants.java | 30 - .../impl/api/ClientRegistrationApiImpl.java | 321 - .../endpoint/impl/model/DCRRequestData.java | 144 - .../service/RegistrationServiceHandler.java | 267 - .../endpoint/impl/util/RegistrationUtils.java | 359 - .../endpoint/impl/util/ResponseStatus.java | 231 - .../src/main/resources/findbugs-include.xml | 22 - .../src/main/webapp/META-INF/MANIFEST.mf | 1 - .../webapp/META-INF/webapp-classloading.xml | 35 - .../src/main/webapp/WEB-INF/beans.xml | 32 - .../src/main/webapp/WEB-INF/web.xml | 69 - .../pom.xml | 93 - .../demo/backend/BankException.java | 37 - .../demo/backend/BankExceptionHandler.java | 35 - .../demo/backend/services/AccountService.java | 203 - .../services/FundsConfirmationService.java | 82 - .../demo/backend/services/PaymentService.java | 277 - .../demo/backend/services/VrpService.java | 290 - .../src/main/resources/findbugs-include.xml | 22 - .../webapp/META-INF/webapp-classloading.xml | 34 - .../src/main/webapp/WEB-INF/cxf-servlet.xml | 59 - .../src/main/webapp/WEB-INF/web.xml | 55 - .../pom.xml | 123 - .../endpoint/api/JWTGeneratorEndpoint.java | 98 - .../JWTGeneratorEndpointErrorResponse.java | 50 - .../demosite/endpoint/model/PayloadData.java | 108 - .../demosite/endpoint/util/GeneratorUtil.java | 434 - .../main/resources/configurations.properties | 21 - .../src/main/resources/findbugs-include.xml | 22 - .../webapp/META-INF/webapp-classloading.xml | 35 - .../src/main/webapp/WEB-INF/beans.xml | 32 - .../src/main/webapp/WEB-INF/web.xml | 69 - .../findbugs-exclude.xml | 20 - .../pom.xml | 172 - .../endpoint/api/EventCreationEndpoint.java | 154 - .../endpoint/api/EventPollingEndpoint.java | 130 - .../api/EventSubscriptionEndpoint.java | 258 - .../EventNotificationEndPointConstants.java | 41 - .../endpoint/util/EventNotificationUtils.java | 142 - .../endpoint/util/EventSubscriptionUtils.java | 99 - .../src/main/resources/findbugs-include.xml | 22 - .../webapp/META-INF/webapp-classloading.xml | 35 - .../src/main/webapp/WEB-INF/beans.xml | 34 - .../src/main/webapp/WEB-INF/web.xml | 69 - .../pom.xml | 122 - .../api/PushAuthorisationEndpoint.java | 185 - .../model/PushAuthorisationResponse.java | 86 - .../src/main/resources/findbugs-include.xml | 22 - .../src/main/webapp/META-INF/MANIFEST.mf | 1 - .../webapp/META-INF/webapp-classloading.xml | 35 - .../src/main/webapp/WEB-INF/beans.xml | 34 - .../src/main/webapp/WEB-INF/web.xml | 69 - .../pom.xml | 190 - .../webapp/OBConsentConfirmServlet.java | 155 - .../webapp/OBConsentServlet.java | 290 - .../authentication/webapp/util/Constants.java | 49 - .../authentication/webapp/i18n.properties | 125 - .../main/resources/configurations.properties | 19 - .../src/main/resources/findbugs-exclude.xml | 45 - .../src/main/resources/findbugs-include.xml | 22 - .../src/main/webapp/WEB-INF/web.xml | 104 - .../assets/img/glyphicons-halflings-white.png | Bin 8777 -> 0 bytes .../assets/img/glyphicons-halflings.png | Bin 13826 -> 0 bytes .../src/main/webapp/assets/js/html5.js | 3 - .../src/main/webapp/cookie_policy.jsp | 240 - .../src/main/webapp/css/Roboto.css | 26 - .../src/main/webapp/css/custom-common.css | 597 - .../src/main/webapp/css/localstyles-ie7.css | 21 - .../src/main/webapp/css/localstyles.css | 282 - .../src/main/webapp/css/openid-provider.css | 195 - .../src/main/webapp/default_consent.jsp | 181 - .../main/webapp/default_displayconsent.jsp | 143 - .../fonts/Roboto/Roboto-Bold-webfont.svg | 593 - .../fonts/Roboto/Roboto-Bold-webfont.ttf | Bin 45008 -> 0 bytes .../fonts/Roboto/Roboto-Bold-webfont.woff | Bin 24808 -> 0 bytes .../webapp/generic-exception-response.jsp | 81 - .../src/main/webapp/images/U2F.png | Bin 33071 -> 0 bytes .../src/main/webapp/images/body-back.png | Bin 39121 -> 0 bytes .../src/main/webapp/images/container-back.png | Bin 199 -> 0 bytes .../src/main/webapp/images/favicon.png | Bin 1242 -> 0 bytes .../src/main/webapp/images/icon-default.png | Bin 5377 -> 0 bytes .../src/main/webapp/images/login-back.svg | 147 - .../src/main/webapp/images/login-icon.png | Bin 9344 -> 0 bytes .../src/main/webapp/images/logo-dark.svg | 153 - .../src/main/webapp/images/logo-inverse.svg | 60 - .../src/main/webapp/images/logo.png | Bin 27990 -> 0 bytes .../src/main/webapp/images/openid-input.gif | Bin 237 -> 0 bytes .../src/main/webapp/images/repeat.jpg | Bin 4368 -> 0 bytes .../webapp/images/wso2-open-banking-logo.png | Bin 4316 -> 0 bytes .../webapp/images/wso2-open-banking-new.png | Bin 314626 -> 0 bytes .../main/webapp/includes/consent_bottom.jsp | 30 - .../src/main/webapp/includes/consent_top.jsp | 52 - .../src/main/webapp/includes/footer.jsp | 28 - .../src/main/webapp/includes/head.jsp | 35 - .../src/main/webapp/includes/localize.jsp | 23 - .../src/main/webapp/js/auth-functions.js | 152 - .../src/main/webapp/js/html5shiv.min.js | 4 - .../src/main/webapp/js/respond.min.js | 5 - .../src/main/webapp/js/scripts.js | 42 - .../src/main/webapp/js/u2f-api.js | 373 - .../bootstrap_3.4.1/css/bootstrap-theme.css | 587 - .../css/bootstrap-theme.css.map | 1 - .../css/bootstrap-theme.min.css | 6 - .../css/bootstrap-theme.min.css.map | 1 - .../libs/bootstrap_3.4.1/css/bootstrap.css | 6834 ---------- .../bootstrap_3.4.1/css/bootstrap.css.map | 1 - .../bootstrap_3.4.1/css/bootstrap.min.css | 6 - .../bootstrap_3.4.1/css/bootstrap.min.css.map | 1 - .../fonts/glyphicons-halflings-regular.eot | Bin 20127 -> 0 bytes .../fonts/glyphicons-halflings-regular.svg | 288 - .../fonts/glyphicons-halflings-regular.ttf | Bin 45404 -> 0 bytes .../fonts/glyphicons-halflings-regular.woff | Bin 23424 -> 0 bytes .../fonts/glyphicons-halflings-regular.woff2 | Bin 18028 -> 0 bytes .../libs/bootstrap_3.4.1/js/bootstrap.js | 2580 ---- .../libs/bootstrap_3.4.1/js/bootstrap.min.js | 6 - .../webapp/libs/bootstrap_3.4.1/js/npm.js | 13 - .../webapp/libs/clipboard-1.7.1/clipboard.js | 790 -- .../libs/clipboard-1.7.1/clipboard.min.js | 7 - .../libs/font-wso2_1.3.0/css/font-wso2.css | 1507 --- .../font-wso2_1.3.0/css/font-wso2.min.css | 15 - .../libs/font-wso2_1.3.0/fonts/font-wso2.eot | Bin 72004 -> 0 bytes .../libs/font-wso2_1.3.0/fonts/font-wso2.svg | 2326 ---- .../libs/font-wso2_1.3.0/fonts/font-wso2.ttf | Bin 71828 -> 0 bytes .../libs/font-wso2_1.3.0/fonts/font-wso2.woff | Bin 142064 -> 0 bytes .../font-wso2_1.3.0/fonts/font-wso2.woff2 | Bin 38660 -> 0 bytes .../webapp/libs/jquery_3.5.0/jquery-3.5.0.js | 10872 ---------------- .../src/main/webapp/oauth2_authz_confirm.jsp | 46 - .../webapp/oauth2_authz_displayconsent.jsp | 125 - .../src/main/webapp/ob_default.jsp | 89 - .../src/main/webapp/ob_default_consent.jsp | 89 - .../src/main/webapp/privacy_policy.jsp | 238 - open-banking-accelerator/pom.xml | 44 +- 844 files changed, 22 insertions(+), 149195 deletions(-) delete mode 100644 open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/findbugs-exclude.xml delete mode 100644 open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/pom.xml delete mode 100644 open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/AccountMetadataDAO.java delete mode 100644 open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/AccountMetadataDAOImpl.java delete mode 100644 open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/queries/AccountMetadataDBQueries.java delete mode 100644 open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/queries/AccountMetadataDBQueriesMySQLImpl.java delete mode 100644 open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/internal/AccountMetadataDataHolder.java delete mode 100644 open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/internal/AccountMetadataServiceComponent.java delete mode 100644 open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/service/AccountMetadataService.java delete mode 100644 open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/service/AccountMetadataServiceImpl.java delete mode 100644 open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/AccountMetadataDAOTests.java delete mode 100644 open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/java/com/wso2/openbanking/accelerator/account/metadata/service/service/AccountMetadataServiceTests.java delete mode 100644 open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/java/com/wso2/openbanking/accelerator/account/metadata/service/util/AccountMetadataDAOTestData.java delete mode 100644 open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/java/com/wso2/openbanking/accelerator/account/metadata/service/util/DAOUtils.java delete mode 100644 open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/resources/dbScripts/h2.sql delete mode 100644 open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/resources/testng.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/pom.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/java/com.wso2.openbanking.accelerator.ciba/CIBAConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/java/com.wso2.openbanking.accelerator.ciba/OBCIBARequestObjectValidationExtension.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/java/com.wso2.openbanking.accelerator.ciba/OBCIBASignatureAlgorithmEnforcementValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/java/com.wso2.openbanking.accelerator.ciba/OBCibaGrantHandler.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/resources/findbugs-exclude.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/test/java/com/wso2/openbanking/accelerator/ciba/OBCIBARequestObjectValidationExtensionTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/test/java/com/wso2/openbanking/accelerator/ciba/OBCIBASignatureAlgorithmEnforcementValidatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/test/java/com/wso2/openbanking/accelerator/ciba/TestConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/test/resources/testng.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/pom.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/caching/OpenBankingBaseCache.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/caching/OpenBankingBaseCacheKey.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/config/OpenBankingConfigParser.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/config/OpenBankingConfigurationService.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/config/OpenBankingConfigurationServiceImpl.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/config/TextFileReader.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/constant/OpenBankingConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/error/OpenBankingErrorCodes.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/DefaultOBEventExecutor.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/OBEventExecutor.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/OBEventQueue.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/OBQueueWorker.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/model/OBEvent.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/CertificateValidationException.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/ConsentManagementException.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/ConsentManagementRuntimeException.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/OBThrottlerException.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/OpenBankingException.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/OpenBankingRuntimeException.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/TPPValidationException.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/ApplicationIdentityService.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/IdentityConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/cache/JWKSetCache.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/cache/JWKSetCacheKey.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/cache/base/OpenBankingIdentityBaseCache.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/retriever/JWKRetriever.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/retriever/ServerIdentityRetriever.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/retriever/sp/CommonServiceProviderRetriever.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/internal/OpenBankingCommonDataHolder.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/internal/OpenBankingCommonServiceComponent.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/model/PSD2RoleEnum.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/persistence/JDBCPersistenceManager.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/persistence/JDBCRetentionDataPersistenceManager.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/AnalyticsLogsUtils.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/CarbonUtils.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/CertificateUtils.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/DatabaseUtil.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/ErrorConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/Generated.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/HTTPClientUtils.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/JWTUtils.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/OpenBankingUtils.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/SPQueryExecutorUtil.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/SecurityUtils.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/ServiceProviderUtils.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/CertificateContent.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/CertificateContentExtractor.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/NcaId.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/NcaName.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSD2Constants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSD2QCStatement.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSD2QCType.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSPRole.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSPRoles.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/error/CertValidationErrors.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/OpenBankingValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/OsgiServiceDiscoverer.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/annotation/RequiredParameter.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/annotation/RequiredParameters.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/annotation/ValidAudience.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/annotation/ValidScopeFormat.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/impl/AudienceValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/impl/MandatoryParameterValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/impl/ScopeValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/resources/findbugs-exclude.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/OBConfigParserTests.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/config/TextFileReaderTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/event/executor/OBEventExecutorTests.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/CertificateUtilsTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/CommonTestUtil.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/HTTPClientUtilsTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/JWTUtilsTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/OpenBankingUtilsTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/SPQueryExecutorUtilTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/SecurityUtilsTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/eidas/certificate/extractor/CertificateContentExtractorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/testutils/JWTUtilsTestDataProvider.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/LogicValidatorsTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/ModelValidatorsTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/resources/SampleChildRequestObject.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/resources/SampleDifferentClass.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/resources/SampleRequestObject.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/resources/ValidatorTestDataProvider.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/open-banking-empty.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/open-banking.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/test-data.json delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/testFile.js delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/testng.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/wso2carbon.jks delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/pom.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/constant/AuthPublisherConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/extension/AbstractAuthDataPublisher.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/extension/DefaultAuthDataPublisher.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/internal/AuthenticationDataPublisherDataHolder.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/internal/AuthenticationDataPublisherServiceComponent.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/service/AuthenticationDataPublisherService.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/service/AuthenticationDataPublisherServiceImpl.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/test/java/com/wso2/openbanking/accelerator/authentication/data/publisher/service/OBAuthDataPublisherTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/test/resources/testng.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/pom.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/DataPublisherFactory.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/DataPublisherPool.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/EventQueue.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/OBThriftDataPublisher.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/OpenBankingDataPublisher.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/QueueWorker.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/constants/DataPublishingConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/internal/OBAnalyticsDataHolder.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/internal/OBAnalyticsServiceComponent.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/model/OBAnalyticsEvent.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/util/OBDataPublisherUtil.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/resources/findbugs-exclude.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/test/java/com/wso2/openbanking/accelerator/data/publisher/common/DataPublisherPoolTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/test/java/com/wso2/openbanking/accelerator/data/publisher/common/OBAnalyticsEventQueueTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/test/java/com/wso2/openbanking/accelerator/data/publisher/common/OBThriftDataPublisherTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/test/resources/testng.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/pom.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/pom.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/cache/CertificateRevocationCache.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/cache/GatewayCache.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/cache/GatewayCacheKey.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/cache/TppValidationCache.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/core/AbstractRequestRouter.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/core/DefaultRequestRouter.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/core/OBExtensionListenerImpl.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/core/OpenBankingGatewayExecutor.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/dcr/DCRExecutor.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/exception/OpenBankingExecutorException.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/api/resource/access/validation/APIResourceAccessValidationExecutor.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/common/reporting/data/executor/CommonReportingDataExecutor.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/consent/ConsentEnforcementExecutor.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/error/handler/OBDefaultErrorHandler.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/mtls/cert/validation/executor/CertRevocationValidationExecutor.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/mtls/cert/validation/executor/MTLSEnforcementExecutor.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/selfcare/portal/UserPermissionValidationExecutor.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/tpp/validation/executor/APITPPValidationExecutor.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/tpp/validation/executor/DCRTPPValidationExecutor.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/jws/JwsRequestSignatureHandlingExecutor.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/jws/JwsResponseSignatureHandlingExecutor.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/model/OBAPIRequestContext.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/model/OBAPIResponseContext.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/model/OpenBankingExecutorError.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/model/RevocationStatus.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/CRLValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/OCSPValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/RevocationValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/service/CertValidationService.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/service/RevocationValidatorFactory.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/service/TPPValidationService.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/util/CertificateValidationUtils.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/handler/GatewayClientAuthenticationHandler.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/handler/JwsResponseSignatureHandler.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/internal/GatewayDataHolder.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/internal/GatewayServiceComponent.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/internal/TPPCertValidatorComponent.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/internal/TPPCertValidatorDataHolder.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/mediator/BasicAuthMediator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/reporter/OBAnalyticsMetricReporter.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/reporter/OBCounterMetric.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/reporter/OBTimestampPublisher.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/synapse/handler/DisputeResolutionSynapseHandler.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/throttling/OBThrottlingExtensionImpl.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/throttling/ThrottleDataPublisher.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/util/GatewayConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/util/GatewaySignatureHandlingUtils.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/util/GatewayUtils.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/util/IdempotencyConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/resources/findbugs-exclude.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/core/DefaultRequestRouterTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/core/TestOBExtensionImpl.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/core/UtilityTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/dcr/DCRExecutorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/common/reporting/data/executor/CommonReportingDataExecutorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/consent/TestEnforcementExecutor.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/error/handler/OBDefaultErrorHandlerTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/mtls/cert/validation/executor/CertRevocationValidationExecutorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/mtls/cert/validation/executor/MTLSEnforcementExecutorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/selfcare/portal/UserPermissionValidationExecutorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/tpp/validation/executor/APITPPValidationExecutorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/tpp/validation/executor/DCRTPPValidationExecutorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/jws/JwsRequestSignatureHandlingExecutorTests.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/jws/JwsResponseSignatureHandlingExecutorTests.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/CRLValidatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/OCSPValidatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/service/CertValidationServiceTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/service/RevocationValidatorFactoryTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/test/TestConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/test/executor/MockOBExecutor.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/test/executor/TestRouter.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/test/util/TestUtil.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/util/CertificateValidationUtilsTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/util/TestValidationUtil.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/handler/JwsResponseSignatureHandlerTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/internal/TPPCertValidatorDataHolderTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/mediator/BasicAuthMediatorTests.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/reporter/OBAnalyticsMetricReporterTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/reporter/TimestampPublishingTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/synapse/handler/DisputeResolutionSynapseHandlerTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/throttling/OBThrottlingExtensionImplTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/throttling/ThrottleDataPublisherTestImpl.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/util/GatewayUtilsTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/resources/client-truststore.jks delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/resources/open-banking.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/resources/test_crl_entries.pem delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/resources/testng.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/resources/wso2carbon.jks delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/pom.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthenticator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthenticatorConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/cache/JTICache.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/exception/JWTValidationException.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/model/DeviceVerificationToken.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/model/DeviceVerificationTokenConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/utils/App2AppAuthUtils.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/DigestValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/ExpiryValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/JTIValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/NBFValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/PublicKeySignatureValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateDigest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateExpiry.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateJTI.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateNBF.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateSignature.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/validationorder/App2AppValidationOrder.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/adaptive/function/OpenBankingAuthenticationWorker.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/adaptive/function/OpenBankingAuthenticationWorkerFunction.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/adaptive/function/OpenBankingAuthenticationWorkerFunctionImpl.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/authz/request/OBOAuthAuthzRequest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/DefaultOBRequestObjectValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/OBRequestObjectValidationExtension.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/OBRequestObjectValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/annotations/SigningAlgorithmValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/annotations/ValidSigningAlgorithm.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/models/OBRequestObject.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/models/ValidationResponse.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/OBCodeResponseTypeHandlerExtension.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/OBHybridResponseTypeHandlerExtension.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/OBResponseTypeHandler.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/impl/OBDefaultResponseTypeHandlerImpl.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/validator/OBCodeResponseTypeValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/validator/OBHybridResponseTypeValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/authenticator/OBIdentifierAuthenticator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/authenticator/constants/IdentifierHandlerConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/authenticator/util/OBIdentifierAuthUtil.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/builders/DefaultOBRequestUriRequestObjectBuilder.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/cache/IdentityCache.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/cache/IdentityCacheKey.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/claims/OBClaimProvider.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/claims/OBDefaultClaimProvider.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/claims/OBDefaultOIDCClaimsCallbackHandler.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/claims/RoleClaimProviderImpl.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/clientauth/OBMutualTLSClientAuthenticator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/IdentityServiceExporter.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/annotations/validationgroups/AttributeChecks.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/annotations/validationgroups/MandatoryChecks.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/annotations/validationgroups/SignatureCheck.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/annotations/validationgroups/ValidityChecks.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/exception/DCRValidationException.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/model/RegistrationError.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/model/RegistrationRequest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/model/RegistrationResponse.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/model/SoftwareStatementBody.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/utils/ValidatorUtils.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/AlgorithmValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/DCRCommonConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/DefaultRegistrationValidatorImpl.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/IssuerValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/RegistrationValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/RequiredParamsValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/SignatureValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/annotation/ValidateAlgorithm.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/annotation/ValidateIssuer.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/annotation/ValidateRequiredParams.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/annotation/ValidateSignature.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/AttributeChecks.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/MandatoryChecks.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/SignatureCheck.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/ValidationOrder.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/ValidityChecks.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationorder/ValidationOrder.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dispute/resolution/DisputeResolutionFilter.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/grant/type/handlers/OBAuthorizationCodeGrantHandler.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/grant/type/handlers/OBClientCredentialsGrantHandler.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/grant/type/handlers/OBPasswordGrantHandler.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/grant/type/handlers/OBRefreshGrantHandler.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/idtoken/OBIDTokenBuilder.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/interceptor/OBDefaultIntrospectionDataProvider.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/interceptor/OBIntrospectionDataProvider.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/internal/IdentityExtensionsDataHolder.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/internal/IdentityExtensionsServiceComponent.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/keyidprovider/OBKeyIDProvider.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/listener/TokenRevocationListener.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/listener/application/AbstractApplicationUpdater.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/listener/application/ApplicationUpdaterImpl.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/listener/application/OBApplicationManagementListener.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/PushAuthRequestValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/constants/PushAuthRequestConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/exception/PushAuthRequestValidatorException.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/model/PushAuthErrorResponse.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/util/PushAuthRequestValidatorUtils.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/sp/metadata/extension/SPMetadataFilter.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/sp/metadata/extension/impl/DefaultSPMetadataFilter.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/DefaultTokenFilter.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/TokenFilter.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/util/TokenFilterException.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/ClientAuthenticatorValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/MTLSCertificateValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/MTLSEnforcementValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/OBIdentityFilterValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/SignatureAlgorithmEnforcementValidator.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/wrapper/RequestWrapper.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/wrapper/ResponseWrapper.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/ClientAuthenticatorEnum.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/HTTPClientUtils.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/IdentityCommonConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/IdentityCommonHelper.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/IdentityCommonUtil.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/resources/findbugs-exclude.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/HTTPClientUtilsTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthUtilsTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthValidationTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthenticatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/testutils/App2AppAuthenticatorTestDataProvider.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/testutils/App2AppUtilsTestJWTDataProvider.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/application/listener/ApplicationManagementListenerTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/adaptive/function/OpenBankingAuthenticationWorkerFunctionImplTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/authz/request/OBOAuthAuthzRequestTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/DefaultOBRequestObjectValidatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/ReqObjectTestDataProvider.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/RequestObjectValidatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/ResponseTypeHandlerTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/validator/OBCodeResponseTypeValidatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/validator/OBHybridResponseTypeValidatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/authenticator/OBIdentifierAuthenticatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/authenticator/util/OBIdentifierAuthenticatorTestData.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/builders/DefaultOBRequestUriRequestObjectBuilderTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/claims/OBDefaultOIDCClaimsCallbackHandlerTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/claims/RoleClaimProviderImplTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/clientauth/OBMutualTLSClientAuthenticatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/DCRExtendedValidatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/DCRValidationTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/DCRValidationUtilTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/ExtendedRegistrationRequest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/ExtendedRegistrationResponse.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/ExtendedSoftwareStatementBody.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/ExtendedValidatorImpl.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/RegistrationTestConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/validation/AlgorithmValidatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/validation/IssuerValidatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/validation/RequiredParamsValidatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dispute/resolution/DisputeResolutionFilterTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/idtoken/OBIDTokenBuilderTests.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/listener/TokenRevocationListenerTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/PushAuthRequestValidatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/util/test/jwt/builder/TestJwtBuilder.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/util/test/jwt/builder/constants/TestJwtBuilderConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/TokenFilterTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/util/TestConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/util/TestUtil.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/validators/ClientAuthenticatorValidatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/validators/MTLSCertificateValidatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/validators/MTLSEnforcementValidatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/validators/SignatureAlgorithmEnforcementValidatorTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/resources/common.auth.script.js delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/resources/testng.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/resources/wso2carbon.jks delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/pom.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/KeyManagerUtil.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/OBKeyManagerConfiguration.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/OBKeyManagerConstants.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/OBKeyManagerExtensionInterface.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/OBKeyManagerImpl.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/internal/KeyManagerDataHolder.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/internal/KeyManagerServiceComponent.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/test/java/com/wso2/openbanking/accelerator/keymanager/KeyManagerTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/test/java/com/wso2/openbanking/accelerator/keymanager/KeyManagerUtilTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/test/resources/testng.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.runtime/com.wso2.openbanking.accelerator.runtime.identity.authn.filter/pom.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.runtime/com.wso2.openbanking.accelerator.runtime.identity.authn.filter/src/main/java/com/wso2/openbanking/accelerator/runtime/identity/authn/filter/OBOAuthClientAuthenticatorProxy.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.runtime/com.wso2.openbanking.accelerator.runtime.identity.authn.filter/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.runtime/pom.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/pom.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/main/java/com/wso2/openbanking/accelerator/service/activator/OBServiceObserver.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/main/java/com/wso2/openbanking/accelerator/service/activator/internal/ServiceObservable.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/main/java/com/wso2/openbanking/accelerator/service/activator/internal/ServiceRegisterComponent.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/test/java/com/wso2/openbanking/accelerator/service/activator/internal/ServiceObservableTest.java delete mode 100644 open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/test/resources/testng.xml delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/pom.xml delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/admin/builder/ConsentAdminBuilder.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/admin/impl/DefaultConsentAdminHandler.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/admin/model/ConsentAdminData.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/admin/model/ConsentAdminHandler.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/builder/ConsentStepsBuilder.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/CIBAConsentPersistStep.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/CIBAConsentRetrievalStep.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/DefaultConsentPersistStep.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/DefaultConsentRetrievalStep.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/NonRegulatoryConsentStep.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/RequestObjectCheckStep.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/model/ConsentData.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/model/ConsentPersistData.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/model/ConsentPersistStep.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/model/ConsentRetrievalStep.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/utils/ConsentRetrievalUtil.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/ConsentMgrAuthServletImpl.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/ISDefaultAuthServletImpl.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/OBDefaultAuthServletImpl.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/util/Constants.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/util/Utils.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/model/OBAuthServletInterface.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/authenticator/CIBAPushAuthenticator.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/authenticator/CIBAPushAuthenticatorConstants.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/impl/CIBAAuthenticationEndpointDefaultImpl.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/model/CIBAAuthenticationEndpointErrorResponse.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/model/CIBAAuthenticationEndpointInterface.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/AuthErrorCode.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentCache.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentException.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentExtensionConstants.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentExtensionExporter.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentExtensionUtils.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentServiceUtil.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ResponseStatus.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/factory/AcceleratorConsentExtensionFactory.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyConstants.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidationException.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidationResult.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidationUtils.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidator.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/event/executors/ConsentAmendmentHistoryEventExecutor.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/event/executors/VRPEventExecutor.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/internal/ConsentExtensionsComponent.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/internal/ConsentExtensionsDataHolder.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/builder/ConsentManageBuilder.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/AccountConsentManageRequestHandler.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/CofConsentRequestHandler.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/ConsentManageRequestHandler.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/DefaultConsentManageHandler.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/PaymentConsentRequestHandler.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/VRPConsentRequestHandler.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/model/ConsentManageData.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/model/ConsentManageHandler.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/model/PeriodicLimit.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/validator/CofConsentRequestValidator.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/validator/PaymentsConsentRequestValidator.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/validator/VRPConsentRequestValidator.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/ConsentManageUtil.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/DebtorAccountSchemeNameEnum.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/PaymentPayloadValidator.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/PeriodicTypesEnum.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/PeriodicalConsentJobActivator.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/jobs/ExpiredConsentStatusUpdateJob.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/jobs/RetentionDatabaseSyncJob.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/scheduler/PeriodicalConsentJobScheduler.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/builder/ConsentValidateBuilder.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/impl/DefaultConsentValidator.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/impl/PaymentFundsConfirmationPayloadValidator.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/impl/PaymentSubmissionPayloadValidator.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/impl/VRPSubmissionPayloadValidator.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/model/ConsentValidateData.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/model/ConsentValidationResult.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/model/ConsentValidator.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/util/ConsentValidatorUtil.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/util/PaymentSubmissionValidationUtil.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/resources/findbugs-exclude.xml delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/vrp/persistence/flow/ConsentPersistStepTests.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/vrp/retrieval/flow/ConsentExtensionDataProvider.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/vrp/retrieval/flow/VRPConsentRetrievalStepTest.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/vrp/retrieval/flow/VRPConsentRetrievalUtilTest.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/AuthServletTest.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/ConsentMgrAuthServletImplTest.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/authenticator/CIBAPushAuthenticatorTests.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidatorTests.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/event/executors/ConsentAmendmentHistoryEventExecutorTests.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/manage/vrp/VRPConsentHandlerTest.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/manage/vrp/VRPConsentRequestValidatorTest.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/manage/vrp/VRPTestConstants.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/AuthServletTestConstants.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentAuthorizeTestConstants.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentExtensionDataProvider.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentExtensionTestConstants.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentExtensionTestUtils.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentValidateTestConstants.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/validate/VRPSubmissionTest.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/resources/testng.xml delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/pom.xml delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/ConsentCoreDAO.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/constants/ConsentMgtDAOConstants.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/exceptions/OBConsentDataDeletionException.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/exceptions/OBConsentDataInsertionException.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/exceptions/OBConsentDataRetrievalException.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/exceptions/OBConsentDataUpdationException.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/impl/ConsentCoreDAOImpl.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/impl/MssqlConsentCoreDAOImpl.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/impl/OracleConsentCoreDAOImpl.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/AuthorizationResource.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentAttributes.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentFile.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentHistoryResource.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentMappingResource.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentResource.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentStatusAuditRecord.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/DetailedConsentResource.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/persistence/ConsentStoreInitializer.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/queries/ConsentMgtCommonDBQueries.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/queries/ConsentMgtMssqlDBQueries.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/queries/ConsentMgtOracleDBQueries.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/queries/ConsentMgtPostgresDBQueries.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/utils/ConsentDAOUtils.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/resources/findbugs-exclude.xml delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/dao/impl/OBConsentMgtDAOTests.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/dao/util/ConsentMgtDAOTestData.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/dao/util/DAOUtils.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/resources/dbScripts/h2.sql delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/resources/testng.xml delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/pom.xml delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/ConsentCoreService.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/constants/ConsentCoreServiceConstants.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/impl/ConsentCoreServiceImpl.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/impl/ConsentStateChangeListenerImpl.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/internal/ConsentManagementDataHolder.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/internal/ConsentManagementServiceComponent.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/listener/ConsentStateChangeListener.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/service/impl/OBConsentMgtCoreServiceTests.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/service/util/ConsentMgtServiceTestData.java delete mode 100644 open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/test/resources/testng.xml delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/pom.xml delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/constants/EventNotificationConstants.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/AggregatedPollingDAO.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/AggregatedPollingDAOImpl.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventPublisherDAO.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventPublisherDAOImpl.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventSubscriptionDAO.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventSubscriptionDAOImpl.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventSubscriptionSqlStatements.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/MSSQLNotificationPollingSqlStatements.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/NotificationPollingSqlStatements.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/NotificationPublisherSqlStatements.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/PostgreSqlEventSubscriptionDAOImpl.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/PostgreSqlPollingDAOImpl.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/EventNotificationErrorDTO.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/EventPollingDTO.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/EventSubscriptionDTO.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/NotificationCreationDTO.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/NotificationDTO.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/exceptions/OBEventNotificationException.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventCreationServiceHandler.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventPollingServiceHandler.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventSubscriptionServiceHandler.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventCreationServiceHandler.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventNotificationPersistenceServiceHandler.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventPollingServiceHandler.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventSubscriptionServiceHandler.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/internal/EventNotificationComponent.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/internal/EventNotificationDataHolder.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/AggregatedPollingResponse.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/EventSubscription.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/Notification.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/NotificationError.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/NotificationEvent.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/persistence/EventPollingStoreInitializer.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/persistence/EventPublisherStoreInitializer.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/persistence/EventSubscriptionStoreInitializer.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/model/RealtimeEventNotification.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/DefaultRealtimeEventNotificationRequestGenerator.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/EventNotificationProducerService.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationLoaderService.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationRequestGenerator.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationSenderService.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/util/activator/PeriodicalEventNotificationConsumerJobActivator.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/util/job/EventNotificationConsumerJob.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/util/scheduler/PeriodicalEventNotificationConsumerJobScheduler.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/response/EventCreationResponse.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/response/EventPollingResponse.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/response/EventSubscriptionResponse.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/DefaultEventNotificationGenerator.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventCreationService.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventNotificationGenerator.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventPollingService.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventSubscriptionService.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/ExtendedEventNotificationGenerator.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/util/EventNotificationServiceUtil.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/resources/findbugs-exclude.xml delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/constants/EventNotificationTestConstants.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/AggregatedPollingDAOImplTests.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventPublisherDAOImplTests.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventSubscriptionDAOImplTests.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/PostgreSqlEventSubscriptionDAOImplTests.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventCreationServiceHandlerTest.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventPollingServiceHandlerTests.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventSubscriptionServiceHandlerTests.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventNotificationPersistenceServiceHandlerTests.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/DefaultRealtimeEventNotificationPayloadGeneratorTests.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/EventNotificationProducerServiceTests.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationLoaderServiceTest.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationSenderServiceTests.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/service/DefaultEventNotificationGeneratorTests.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventCreationServiceTests.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventPollingServiceTests.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventSubscriptionServiceTests.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/utils/EventNotificationTestUtils.java delete mode 100644 open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/resources/testng.xml delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/pom.xml delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/OBThrottlerDAO.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/constants/OBThrottlerDAOConstants.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/exception/OBThrottlerDataDeletionException.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/exception/OBThrottlerDataInsertionException.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/exception/OBThrottlerDataRetrievalException.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/exception/OBThrottlerDataUpdationException.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/impl/OBThrottlerDAOImpl.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/model/ThrottleDataModel.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/persistence/DataStoreInitializer.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/persistence/DataStoreInitializerFactory.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/queries/OBThrottlerSQLStatements.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/resources/findbugs-exclude.xml delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/java/com/wso2/openbanking/accelerator/throttler/dao/impl/OBThrottlerDAOTests.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/java/com/wso2/openbanking/accelerator/throttler/dao/util/OBThrottlerDAOTestData.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/java/com/wso2/openbanking/accelerator/throttler/dao/util/OBThrottlerDAOUtils.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/resources/dbScripts/h2.sql delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/resources/testng.xml delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/pom.xml delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/java/com/wso2/openbanking/accelerator/throttler/service/OBThrottleService.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/java/com/wso2/openbanking/accelerator/throttler/service/constants/OBThrottlerServiceConstants.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/java/com/wso2/openbanking/accelerator/throttler/service/internal/OBThrottlerDataHolder.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/java/com/wso2/openbanking/accelerator/throttler/service/internal/OBThrottlerServiceComponent.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/test/java/com/wso2/openbanking/accelerator/throttler/service/OBThrottleServiceTests.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/test/java/com/wso2/openbanking/accelerator/throttler/service/util/OBThrottleServiceTestData.java delete mode 100644 open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/test/resources/testng.xml delete mode 100755 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/.openapi-generator-ignore delete mode 100755 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/findbugs-exclude.xml delete mode 100755 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/pom.xml delete mode 100755 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/api/ApplicationInformationApi.java delete mode 100755 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/model/ApplicationBulkMetadataSuccessDTO.java delete mode 100755 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/model/ApplicationInfoErrorDTO.java delete mode 100755 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/model/ApplicationMetadataResourceDTO.java delete mode 100755 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/model/ApplicationSingleMetadataSuccessDTO.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/java/com/wso2/open/banking/application/info/endpoint/api/constants/MetaDataSQLStatements.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/java/com/wso2/open/banking/application/info/endpoint/api/data/MetaDataDAOImpl.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/java/com/wso2/open/banking/application/info/endpoint/api/impl/ApplicationInformationApiServiceImpl.java delete mode 100755 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/java/com/wso2/open/banking/application/info/endpoint/api/utils/MappingUtil.java delete mode 100755 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/resources/application-info-300.yaml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/resources/findbugs-exclude.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/resources/findbugs-include.xml delete mode 100755 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/webapp/META-INF/webapp-classloading.xml delete mode 100755 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/webapp/WEB-INF/beans.xml delete mode 100755 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/webapp/WEB-INF/web.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/pom.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/java/com/wso2/openbanking/accelerator/ciba/authentication/endpoint/impl/api/CIBAAuthenticationEndpoint.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/java/com/wso2/openbanking/accelerator/ciba/authentication/endpoint/impl/api/CIBAAuthenticationEndpointConstants.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/java/com/wso2/openbanking/accelerator/ciba/authentication/endpoint/impl/exception/CIBAAuthenticationEndpointException.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/webapp/META-INF/MANIFEST.mf delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/webapp/META-INF/webapp-classloading.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/webapp/WEB-INF/beans.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/webapp/WEB-INF/web.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/pom.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/api/ConsentAdminEndpoint.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/api/ConsentAuthorizeEndpoint.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/api/ConsentManageEndpoint.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/api/ConsentValidationEndpoint.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/error/ConsentThrowableMapper.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/util/ConsentConstants.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/util/ConsentUtils.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/resources/findbugs-exclude.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/webapp/META-INF/webapp-classloading.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/webapp/WEB-INF/beans.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/webapp/WEB-INF/web.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/pom.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/gen/java/com/wso2/openbanking/accelerator/dynamic/client/registration/dto/RegistrationErrorDTO.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/RegistrationConstants.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/api/ClientRegistrationApiImpl.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/model/DCRRequestData.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/service/RegistrationServiceHandler.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/util/RegistrationUtils.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/util/ResponseStatus.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/webapp/META-INF/MANIFEST.mf delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/webapp/META-INF/webapp-classloading.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/webapp/WEB-INF/beans.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/webapp/WEB-INF/web.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/pom.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/BankException.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/BankExceptionHandler.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/services/AccountService.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/services/FundsConfirmationService.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/services/PaymentService.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/services/VrpService.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/webapp/META-INF/webapp-classloading.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/webapp/WEB-INF/cxf-servlet.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/webapp/WEB-INF/web.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/pom.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/java/com/wso2/openbanking/accelerator/demosite/endpoint/api/JWTGeneratorEndpoint.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/java/com/wso2/openbanking/accelerator/demosite/endpoint/model/JWTGeneratorEndpointErrorResponse.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/java/com/wso2/openbanking/accelerator/demosite/endpoint/model/PayloadData.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/java/com/wso2/openbanking/accelerator/demosite/endpoint/util/GeneratorUtil.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/resources/configurations.properties delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/webapp/META-INF/webapp-classloading.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/webapp/WEB-INF/beans.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/webapp/WEB-INF/web.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/findbugs-exclude.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/pom.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/api/EventCreationEndpoint.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/api/EventPollingEndpoint.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/api/EventSubscriptionEndpoint.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/constants/EventNotificationEndPointConstants.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/util/EventNotificationUtils.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/util/EventSubscriptionUtils.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/webapp/META-INF/webapp-classloading.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/webapp/WEB-INF/beans.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/webapp/WEB-INF/web.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/pom.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/java/com/wso2/openbanking/accelerator/push/authorization/endpoint/api/PushAuthorisationEndpoint.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/java/com/wso2/openbanking/accelerator/push/authorization/endpoint/model/PushAuthorisationResponse.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/webapp/META-INF/MANIFEST.mf delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/webapp/META-INF/webapp-classloading.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/webapp/WEB-INF/beans.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/webapp/WEB-INF/web.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/pom.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/java/com/wso2/openbanking/accelerator/authentication/webapp/OBConsentConfirmServlet.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/java/com/wso2/openbanking/accelerator/authentication/webapp/OBConsentServlet.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/java/com/wso2/openbanking/accelerator/authentication/webapp/util/Constants.java delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/resources/com/wso2/openbanking/authentication/webapp/i18n.properties delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/resources/configurations.properties delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/resources/findbugs-exclude.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/resources/findbugs-include.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/WEB-INF/web.xml delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/assets/img/glyphicons-halflings-white.png delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/assets/img/glyphicons-halflings.png delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/assets/js/html5.js delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/cookie_policy.jsp delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/Roboto.css delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/custom-common.css delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/localstyles-ie7.css delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/localstyles.css delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/openid-provider.css delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/default_consent.jsp delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/default_displayconsent.jsp delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/fonts/Roboto/Roboto-Bold-webfont.svg delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/fonts/Roboto/Roboto-Bold-webfont.ttf delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/fonts/Roboto/Roboto-Bold-webfont.woff delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/generic-exception-response.jsp delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/U2F.png delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/body-back.png delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/container-back.png delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/favicon.png delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/icon-default.png delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/login-back.svg delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/login-icon.png delete mode 100755 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/logo-dark.svg delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/logo-inverse.svg delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/logo.png delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/openid-input.gif delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/repeat.jpg delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/wso2-open-banking-logo.png delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/wso2-open-banking-new.png delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/includes/consent_bottom.jsp delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/includes/consent_top.jsp delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/includes/footer.jsp delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/includes/head.jsp delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/includes/localize.jsp delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/js/auth-functions.js delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/js/html5shiv.min.js delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/js/respond.min.js delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/js/scripts.js delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/js/u2f-api.js delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/bootstrap_3.4.1/css/bootstrap-theme.css delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/bootstrap_3.4.1/css/bootstrap-theme.css.map delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/bootstrap_3.4.1/css/bootstrap-theme.min.css delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/bootstrap_3.4.1/css/bootstrap-theme.min.css.map delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/bootstrap_3.4.1/css/bootstrap.css delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/bootstrap_3.4.1/css/bootstrap.css.map delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/bootstrap_3.4.1/css/bootstrap.min.css delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/bootstrap_3.4.1/css/bootstrap.min.css.map delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/bootstrap_3.4.1/fonts/glyphicons-halflings-regular.eot delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/bootstrap_3.4.1/fonts/glyphicons-halflings-regular.svg delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/bootstrap_3.4.1/fonts/glyphicons-halflings-regular.ttf delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/bootstrap_3.4.1/fonts/glyphicons-halflings-regular.woff delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/bootstrap_3.4.1/fonts/glyphicons-halflings-regular.woff2 delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/bootstrap_3.4.1/js/bootstrap.js delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/bootstrap_3.4.1/js/bootstrap.min.js delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/bootstrap_3.4.1/js/npm.js delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/clipboard-1.7.1/clipboard.js delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/clipboard-1.7.1/clipboard.min.js delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/font-wso2_1.3.0/css/font-wso2.css delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/font-wso2_1.3.0/css/font-wso2.min.css delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/font-wso2_1.3.0/fonts/font-wso2.eot delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/font-wso2_1.3.0/fonts/font-wso2.svg delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/font-wso2_1.3.0/fonts/font-wso2.ttf delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/font-wso2_1.3.0/fonts/font-wso2.woff delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/font-wso2_1.3.0/fonts/font-wso2.woff2 delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/libs/jquery_3.5.0/jquery-3.5.0.js delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/oauth2_authz_confirm.jsp delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/oauth2_authz_displayconsent.jsp delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/ob_default.jsp delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/ob_default_consent.jsp delete mode 100644 open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/privacy_policy.jsp diff --git a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/findbugs-exclude.xml b/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/findbugs-exclude.xml deleted file mode 100644 index cd226386..00000000 --- a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/findbugs-exclude.xml +++ /dev/null @@ -1,28 +0,0 @@ - - - - - - - - - - - - diff --git a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/pom.xml b/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/pom.xml deleted file mode 100644 index 8003b149..00000000 --- a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/pom.xml +++ /dev/null @@ -1,212 +0,0 @@ - - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../../pom.xml - - 4.0.0 - - com.wso2.openbanking.accelerator.account.metadata.service - WSO2 Open Banking - Account Metadata Service Module - bundle - - - - org.springframework - spring-web - provided - - - org.apache.cxf - cxf-bundle-jaxrs - provided - - - org.apache.commons - commons-lang3 - provided - - - commons-logging - commons-logging - provided - - - org.testng - testng - test - - - com.h2database - h2 - test - - - org.mockito - mockito-all - - - org.powermock - powermock-module-testng - - - org.powermock - powermock-api-mockito - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - - - - - - - org.apache.maven.plugins - maven-surefire-plugin - - - src/test/resources/testng.xml - - - - - org.jacoco - jacoco-maven-plugin - ${jacoco.version} - - - - **/*Constants.class - **/*Component.class - **/*DataHolder.class - - - - - default-prepare-agent - - prepare-agent - - - - default-prepare-agent-integration - - prepare-agent-integration - - - - default-report - - report - - - - default-report-integration - - report-integration - - - - default-check - - check - - - - - BUNDLE - - - INSTRUCTION - COVEREDRATIO - 0.73 - - - - - - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - false - ${project.build.directory}/spotbugs - ${project.basedir}/findbugs-exclude.xml - - - - analyze-compile - compile - - check - - - - - - org.apache.maven.plugins - maven-compiler-plugin - - 1.8 - 1.8 - UTF-8 - - - - org.apache.felix - maven-bundle-plugin - true - - - - ${project.artifactId} - - - com.wso2.openbanking.accelerator.event.notifications.service.internal - - - org.osgi.framework;version="${osgi.framework.imp.pkg.version.range}", - org.osgi.service.component;version="${osgi.service.component.imp.pkg.version.range}", - org.apache.commons.lang3;version="${commons-lang.version}" - - - !com.wso2.openbanking.accelerator.account.metadata.service.internal, - com.wso2.openbanking.accelerator.account.metadata.service.service.*;version="${project.version}", - com.wso2.openbanking.accelerator.account.metadata.service.dao.*;version="${project.version}", -\ - - javax.ws.rs-api;scope=compile;inline=false, - - * - <_dsannotations>* - - - - - - diff --git a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/AccountMetadataDAO.java b/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/AccountMetadataDAO.java deleted file mode 100644 index 3bf5ea3b..00000000 --- a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/AccountMetadataDAO.java +++ /dev/null @@ -1,128 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.account.metadata.service.dao; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; - -import java.sql.Connection; -import java.util.Map; - -/** - * AccountMetadataDAO - *

- * Contains the methods to store, retrieve and delete account - * metadata in the database. - */ -public interface AccountMetadataDAO { - - /** - * Store account metadata. - * - * @param accountId - Account ID - * @param userId - User ID - * @param metadataKey - Metadata key - * @param metadataValue - Metadata value - * @return number of rows affected - * @throws OpenBankingException - OpenBankingException - */ - int storeAccountMetadata(Connection dbConnection, String accountId, String userId, String metadataKey, - String metadataValue) throws OpenBankingException; - - /** - * Store or update account metadata. - * If the key already exists for the account-id and user-id combination, the value be updated. - * - * @param accountId - Account ID - * @param userId - User ID - * @param metadataKey - Metadata key - * @param metadataValue - Metadata value - * @return number of rows affected - * @throws OpenBankingException - OpenBankingException - */ - int updateAccountMetadata(Connection dbConnection, String accountId, String userId, String metadataKey, - String metadataValue) throws OpenBankingException; - - /** - * Retrieve account metadata for a given user-id and account-id combination. - * - * @param accountId - Account ID - * @param userId - User ID - * @return Map of account metadata - * @throws OpenBankingException - OpenBankingException - */ - Map getAccountMetadataMap(Connection dbConnection, String accountId, String userId) - throws OpenBankingException; - - /** - * Retrieve account metadata for a given account-id and key combination. - * - * @param accountId - Account ID - * @param key - Attribute key - * @return Map of user-id and attribute value - * @throws OpenBankingException - OpenBankingException - */ - Map getMetadataForAccountIdAndKey(Connection dbConnection, String accountId, String key) - throws OpenBankingException; - - /** - * Retrieve the value for the given account-id, user-id and key combination. - * - * @param accountId - Account ID - * @param userId - User ID - * @param key - Key - * @return Attribute value - * @throws OpenBankingException - OpenBankingException - */ - String getAccountMetadataByKey(Connection dbConnection, String accountId, String userId, String key) - throws OpenBankingException; - - /** - * Delete all account metadata for a given user-id and account-id combination. - * - * @param accountId - Account ID - * @param userId - User ID - * @return number of rows affected - * @throws OpenBankingException - OpenBankingException - */ - int deleteAccountMetadata(Connection dbConnection, String accountId, String userId) throws OpenBankingException; - - /** - * Delete account metadata for a given user-id, account-id and key combination. - * - * @param accountId - Account ID - * @param userId - User ID - * @param key - Key - * @return number of rows affected - * @throws OpenBankingException - OpenBankingException - */ - int deleteAccountMetadataByKey(Connection dbConnection, String accountId, String userId, String key) throws - OpenBankingException; - - /** - * Delete all account metadata for a given account-id and key combination. - * - * @param accountId - Account ID - * @param key - Key - * @return number of rows affected - * @throws OpenBankingException - OpenBankingException - */ - int deleteAccountMetadataByKeyForAllUsers(Connection dbConnection, String accountId, String key) throws - OpenBankingException; - -} diff --git a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/AccountMetadataDAOImpl.java b/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/AccountMetadataDAOImpl.java deleted file mode 100644 index 1fe8c4e8..00000000 --- a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/AccountMetadataDAOImpl.java +++ /dev/null @@ -1,493 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.account.metadata.service.dao; - -import com.wso2.openbanking.accelerator.account.metadata.service.dao.queries.AccountMetadataDBQueries; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.DatabaseUtil; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.sql.Savepoint; -import java.sql.Timestamp; -import java.util.Date; -import java.util.HashMap; -import java.util.Map; - -/** - * Implementation of AccountMetadataDAO. - */ -public class AccountMetadataDAOImpl implements AccountMetadataDAO { - - private static final Log log = LogFactory.getLog(AccountMetadataDAOImpl.class); - private static final String KEY = "METADATA_KEY"; - private static final String VALUE = "METADATA_VALUE"; - private static final String USER_ID = "USER_ID"; - - //Error messages - private static final String DB_CONNECTION_NULL_ERROR = "Database connection is null."; - private static final String ACCOUNT_ID_USER_ID_MISSING_ERROR = "Account Id or User Id is not provided"; - private static final String ACCOUNT_METADATA_MISSING_ERROR = "Metadata key or Metadata value is not provided"; - private static final String ERROR_WHILE_DELETING_METADATA = "Error occurred while deleting account metadata."; - - - AccountMetadataDBQueries sqlStatements; - - public AccountMetadataDAOImpl(AccountMetadataDBQueries sqlStatements) { - this.sqlStatements = sqlStatements; - } - - /** - * {@inheritDoc} - */ - @Override - public int storeAccountMetadata(Connection dbConnection, String accountId, String userId, - String metadataKey, String metadataValue) throws OpenBankingException { - - int noOfRows; - - if (StringUtils.isBlank(accountId) || StringUtils.isBlank(userId)) { - log.error(ACCOUNT_ID_USER_ID_MISSING_ERROR); - throw new OpenBankingException(ACCOUNT_ID_USER_ID_MISSING_ERROR); - } - if (StringUtils.isBlank(metadataKey) || StringUtils.isBlank(metadataValue)) { - log.error(ACCOUNT_METADATA_MISSING_ERROR); - throw new OpenBankingException(ACCOUNT_METADATA_MISSING_ERROR); - } - if (dbConnection == null) { - log.error(DB_CONNECTION_NULL_ERROR); - throw new OpenBankingException(DB_CONNECTION_NULL_ERROR); - } - - try { - String storeAttributeSqlStatement = sqlStatements.getStoreAccountMetadataPreparedStatement(); - Savepoint savepoint = dbConnection.setSavepoint(); - log.debug("Storing account metadata data in the database for account-id " + accountId + " and " + - "user-id " + userId); - try (PreparedStatement prepStmt = dbConnection.prepareStatement(storeAttributeSqlStatement)) { - prepStmt.setString(1, accountId); - prepStmt.setString(2, userId); - prepStmt.setString(3, metadataKey); - prepStmt.setString(4, metadataValue); - prepStmt.setTimestamp(5, new Timestamp(new Date().getTime())); - - if (log.isDebugEnabled()) { - log.debug("Added data for the key " + metadataKey + " and value " + metadataValue + - " to be inserted to the database."); - } - noOfRows = prepStmt.executeUpdate(); - if (noOfRows > 0) { - if (log.isDebugEnabled()) { - log.debug("The query affected " + noOfRows + " rows."); - log.debug("Stored attributes for account-id" + accountId + " and user-id " + userId + - " in the database."); - } - dbConnection.commit(); - } else { - dbConnection.rollback(savepoint); - log.error("Error occurred while inserting account metadata data. Any changes occurred " + - "during the failed transaction are rolled back."); - throw new OpenBankingException("Error occurred while inserting account metadata " + - "data."); - } - } catch (SQLException e) { - dbConnection.rollback(savepoint); - log.error("Error occurred while inserting account metadata data.", e); - throw new OpenBankingException("Error occurred while inserting account metadata " + - "data.", e); - } - } catch (SQLException e) { - log.error("Error occurred while interacting with the database connection.", e); - throw new OpenBankingException("Error occurred while interacting with the database connection", e); - } finally { - DatabaseUtil.closeConnection(dbConnection); - } - return noOfRows; - } - - /** - * {@inheritDoc} - */ - @Override - public int updateAccountMetadata(Connection dbConnection, String accountId, String userId, - String metadataKey, String metadataValue) throws OpenBankingException { - - int noOfRows; - - if (StringUtils.isBlank(accountId) || StringUtils.isBlank(userId)) { - log.error(ACCOUNT_ID_USER_ID_MISSING_ERROR); - throw new OpenBankingException(ACCOUNT_ID_USER_ID_MISSING_ERROR); - } - if (StringUtils.isBlank(metadataKey) || StringUtils.isBlank(metadataValue)) { - log.error(ACCOUNT_METADATA_MISSING_ERROR); - throw new OpenBankingException(ACCOUNT_METADATA_MISSING_ERROR); - } - if (dbConnection == null) { - log.error(DB_CONNECTION_NULL_ERROR); - throw new OpenBankingException(DB_CONNECTION_NULL_ERROR); - } - - try { - String storeAttributeSqlStatement = sqlStatements.getUpdateAccountMetadataPreparedStatement(); - Savepoint savepoint = dbConnection.setSavepoint(); - log.debug("Storing account metadata data in the database for account-id " + accountId + " and " + - "user-id " + userId); - try (PreparedStatement prepStmt = dbConnection.prepareStatement(storeAttributeSqlStatement)) { - prepStmt.setString(1, metadataValue); - prepStmt.setTimestamp(2, new Timestamp(new Date().getTime())); - prepStmt.setString(3, accountId); - prepStmt.setString(4, userId); - prepStmt.setString(5, metadataKey); - prepStmt.executeUpdate(); - if (log.isDebugEnabled()) { - log.debug("Added data for the key " + metadataKey + " and value" + metadataValue + - " to be updated in the database."); - } - noOfRows = prepStmt.executeUpdate(); - if (noOfRows > 0) { - if (log.isDebugEnabled()) { - log.debug("The query affected " + noOfRows + " rows."); - log.debug("Updated attributes for account-id" + accountId + " and user-id " + userId + - " in the database."); - } - } else { - log.info("No rows were affected in the transaction. No change was made to existing account " + - "metadata."); - } - dbConnection.commit(); - } catch (SQLException e) { - dbConnection.rollback(savepoint); - log.error("Error occurred while updating account metadata.", e); - throw new OpenBankingException("Error occurred while inserting account metadata.", e); - } - } catch (SQLException e) { - log.error("Error occurred while interacting with the database connection.", e); - throw new OpenBankingException("Error occurred while interacting with the database connection", e); - } finally { - DatabaseUtil.closeConnection(dbConnection); - } - return noOfRows; - } - - /** - * {@inheritDoc} - */ - @Override - public Map getAccountMetadataMap(Connection dbConnection, String accountId, String userId) throws - OpenBankingException { - - if (StringUtils.isBlank(accountId) || StringUtils.isBlank(userId)) { - log.error(ACCOUNT_ID_USER_ID_MISSING_ERROR); - throw new OpenBankingException(ACCOUNT_ID_USER_ID_MISSING_ERROR); - } - if (dbConnection == null) { - log.error(DB_CONNECTION_NULL_ERROR); - throw new OpenBankingException(DB_CONNECTION_NULL_ERROR); - } - HashMap attributesMap = new HashMap<>(); - final String retrieveAttributeSqlStatement = sqlStatements.getRetrieveAccountMetadataPreparedStatement(); - - if (log.isDebugEnabled()) { - log.debug("Retrieving account metadata for account-id " + accountId + " and user-id " + - userId); - } - try (PreparedStatement prepStmt = dbConnection.prepareStatement(retrieveAttributeSqlStatement)) { - prepStmt.setString(1, accountId); - prepStmt.setString(2, userId); - try (ResultSet rs = prepStmt.executeQuery()) { - while (rs.next()) { - attributesMap.put(rs.getString(KEY), rs.getString(VALUE)); - if (log.isDebugEnabled()) { - log.debug("Added attribute with key " + rs.getString(KEY) + " and value " + - rs.getString(VALUE) + "to the map."); - } - } - } catch (SQLException e) { - log.error("Error occurred while reading retrieved result set for account-id " + accountId + - "and user-id " + userId, e); - throw new OpenBankingException("Error occurred while reading retrieved result set for " + - "account-id " + accountId + " and user-id " + userId, e); - } - } catch (SQLException e) { - log.error("Error occurred while retrieving account metadata from database for account-id " + - accountId + " and user-id " + userId, e); - throw new OpenBankingException("Error occurred while retrieving account metadata " + - "from database for account-id " + accountId + " and user-id " + userId, e); - } finally { - DatabaseUtil.closeConnection(dbConnection); - } - return attributesMap; - } - - @Override - public Map getMetadataForAccountIdAndKey(Connection dbConnection, String accountId, String key) - throws OpenBankingException { - - if (StringUtils.isBlank(accountId) || StringUtils.isBlank(key)) { - log.error("AccountId or Key not found in the request"); - throw new OpenBankingException("AccountId and Key should be submitted in order to proceed."); - } - if (dbConnection == null) { - log.error(DB_CONNECTION_NULL_ERROR); - throw new OpenBankingException(DB_CONNECTION_NULL_ERROR); - } - HashMap attributesMap = new HashMap<>(); - final String retrieveAttributeSqlStatement = sqlStatements. - getRetrieveMetadataByAccountIdAndKeyPreparedStatement(); - - if (log.isDebugEnabled()) { - log.debug("Retrieving account metadata for account-id " + accountId + " and metadata-key " + - key); - } - try (PreparedStatement prepStmt = dbConnection.prepareStatement(retrieveAttributeSqlStatement)) { - prepStmt.setString(1, accountId); - prepStmt.setString(2, key); - try (ResultSet rs = prepStmt.executeQuery()) { - while (rs.next()) { - attributesMap.put(rs.getString(USER_ID), rs.getString(VALUE)); - if (log.isDebugEnabled()) { - log.debug("Added attribute with user-id " + rs.getString(USER_ID) + " and value " + - rs.getString(VALUE) + "to the map."); - } - } - } catch (SQLException e) { - log.error("Error occurred while reading retrieved result set for account-id " + accountId + - "and key " + key, e); - throw new OpenBankingException("Error occurred while reading retrieved result set for " + - "account-id " + accountId + " and key " + key, e); - } - } catch (SQLException e) { - log.error("Error occurred while retrieving account metadata from database for account-id " + - accountId + " and key " + key, e); - throw new OpenBankingException("Error occurred while retrieving account metadata " + - "from database for account-id " + accountId + " and key " + key, e); - } finally { - DatabaseUtil.closeConnection(dbConnection); - } - return attributesMap; - } - - /** - * {@inheritDoc} - */ - @Override - public String getAccountMetadataByKey(Connection dbConnection, String accountId, String userId, String key) throws - OpenBankingException { - - if (StringUtils.isBlank(accountId) || StringUtils.isBlank(userId) || StringUtils.isBlank(key)) { - log.error("AccountId, UserId or Key not found in the request"); - throw new OpenBankingException("AccountId, UserId and Key should be submitted in order to " + - "proceed."); - } - if (dbConnection == null) { - log.error(DB_CONNECTION_NULL_ERROR); - throw new OpenBankingException(DB_CONNECTION_NULL_ERROR); - } - String attributeValue = null; - final String retrieveAttributeSqlStatement = sqlStatements. - getRetrieveAccountMetadataByKeyPreparedStatement(); - - if (log.isDebugEnabled()) { - log.debug("Retrieving account metadata for account-id " + accountId + " and user-id " + userId); - } - try (PreparedStatement prepStmt = dbConnection.prepareStatement(retrieveAttributeSqlStatement)) { - prepStmt.setString(1, accountId); - prepStmt.setString(2, userId); - prepStmt.setString(3, key); - try (ResultSet rs = prepStmt.executeQuery()) { - if (rs.next()) { - attributeValue = rs.getString(VALUE); - if (log.isDebugEnabled()) { - log.debug("Retrieved attribute with key " + key + " and value " + attributeValue); - } - } - dbConnection.commit(); - } catch (SQLException e) { - log.error("Error occurred while reading retrieved result set for account-id " + accountId + - " and user-id " + userId, e); - throw new OpenBankingException("Error occurred while reading retrieved result set for " + - "account-id " + accountId + " and user-id " + userId, e); - } - } catch (SQLException e) { - log.error("Error occurred while retrieving account metadata from database for account-id " + - accountId + " and user-id " + userId, e); - throw new OpenBankingException("Error occurred while retrieving account metadata " + - "from database for account-id " + accountId + " and user-id " + userId, e); - } finally { - DatabaseUtil.closeConnection(dbConnection); - } - return attributeValue; - } - - /** - * {@inheritDoc} - */ - @Override - public int deleteAccountMetadata(Connection dbConnection, String accountId, String userId) - throws OpenBankingException { - - int noOfRows; - if (StringUtils.isBlank(accountId) || StringUtils.isBlank(userId)) { - log.error(ACCOUNT_ID_USER_ID_MISSING_ERROR); - throw new OpenBankingException(ACCOUNT_ID_USER_ID_MISSING_ERROR); - } - if (dbConnection == null) { - log.error(DB_CONNECTION_NULL_ERROR); - throw new OpenBankingException(DB_CONNECTION_NULL_ERROR); - } - try { - String deleteAttributeSqlStatement = sqlStatements.getDeleteAccountMetadataPreparedStatement(); - dbConnection.setAutoCommit(false); - Savepoint savepoint = dbConnection.setSavepoint(); - - try (PreparedStatement prepStmt = dbConnection.prepareStatement(deleteAttributeSqlStatement)) { - prepStmt.setString(1, accountId); - prepStmt.setString(2, userId); - noOfRows = prepStmt.executeUpdate(); - if (noOfRows >= 0) { - dbConnection.commit(); - if (log.isDebugEnabled()) { - log.debug("Deleted " + noOfRows + " account metadata for account-id " + accountId + - "and user-id " + userId); - } - } else { - dbConnection.rollback(savepoint); - log.error(ERROR_WHILE_DELETING_METADATA + "Any changes occurred " + - "during the failed transaction are rolled back."); - throw new OpenBankingException(ERROR_WHILE_DELETING_METADATA); - } - } catch (SQLException e) { - dbConnection.rollback(savepoint); - log.error(ERROR_WHILE_DELETING_METADATA, e); - throw new OpenBankingException(ERROR_WHILE_DELETING_METADATA, e); - } - } catch (SQLException e) { - log.error("Error occurred while interacting with the database connection.", e); - throw new OpenBankingException("Error occurred while interacting with the database connection", e); - } finally { - DatabaseUtil.closeConnection(dbConnection); - } - return noOfRows; - } - - /** - * {@inheritDoc} - */ - @Override - public int deleteAccountMetadataByKey(Connection dbConnection, String accountId, String userId, String key) throws - OpenBankingException { - - if (StringUtils.isBlank(accountId) || StringUtils.isBlank(userId) || StringUtils.isBlank(key)) { - log.error("AccountId, UserId or Key not found in the request"); - throw new OpenBankingException("AccountId, UserId and Key should be submitted in order to " + - "proceed."); - } - if (dbConnection == null) { - log.error(DB_CONNECTION_NULL_ERROR); - throw new OpenBankingException(DB_CONNECTION_NULL_ERROR); - } - int noOfRows; - try { - String deleteAttributeSqlStatement = sqlStatements.getDeleteAccountMetadataByKeyPreparedStatement(); - Savepoint savepoint = dbConnection.setSavepoint(); - - try (PreparedStatement prepStmt = dbConnection.prepareStatement(deleteAttributeSqlStatement)) { - prepStmt.setString(1, accountId); - prepStmt.setString(2, userId); - prepStmt.setString(3, key); - noOfRows = prepStmt.executeUpdate(); - if (noOfRows >= 0) { - dbConnection.commit(); - if (log.isDebugEnabled()) { - log.debug("Deleted account metadata for account-id " + accountId + "and user-id " + - userId + "for the key " + key); - } - } else { - dbConnection.rollback(savepoint); - log.error(ERROR_WHILE_DELETING_METADATA + "Any changes occurred " + - "during the failed transaction are rolled back."); - throw new OpenBankingException(ERROR_WHILE_DELETING_METADATA); - } - } catch (SQLException e) { - dbConnection.rollback(savepoint); - log.error("Error occurred while deleting account metadata data.", e); - throw new OpenBankingException("Error occurred while deleting account metadata data.", e); - } - } catch (SQLException e) { - log.error("Error occurred while interacting with the database connection.", e); - throw new OpenBankingException("Error occurred while interacting with the database connection", e); - } finally { - DatabaseUtil.closeConnection(dbConnection); - } - return noOfRows; - } - - @Override - public int deleteAccountMetadataByKeyForAllUsers(Connection dbConnection, String accountId, String key) throws - OpenBankingException { - - int noOfRows; - if (StringUtils.isBlank(accountId) || StringUtils.isBlank(key)) { - log.error("AccountId or Key not found in the request"); - throw new OpenBankingException("AccountId and Key should be submitted in order to " + - "proceed."); - } - if (dbConnection == null) { - log.error(DB_CONNECTION_NULL_ERROR); - throw new OpenBankingException(DB_CONNECTION_NULL_ERROR); - } - try { - String deleteAttributeSqlStatement = sqlStatements. - getDeleteAccountMetadataByKeyForAllUsersPreparedStatement(); - Savepoint savepoint = dbConnection.setSavepoint(); - - try (PreparedStatement prepStmt = dbConnection.prepareStatement(deleteAttributeSqlStatement)) { - prepStmt.setString(1, accountId); - prepStmt.setString(2, key); - noOfRows = prepStmt.executeUpdate(); - if (noOfRows >= 0) { - dbConnection.commit(); - if (log.isDebugEnabled()) { - log.debug("Deleted account metadata for account-id " + accountId + "for the key " + key); - } - } else { - dbConnection.rollback(savepoint); - log.error(ERROR_WHILE_DELETING_METADATA + "Any changes occurred " + - "during the failed transaction are rolled back."); - throw new OpenBankingException(ERROR_WHILE_DELETING_METADATA); - } - } catch (SQLException e) { - dbConnection.rollback(savepoint); - log.error("Error occurred while deleting account metadata data.", e); - throw new OpenBankingException("Error occurred while deleting account metadata data.", e); - } - } catch (SQLException e) { - log.error("Error occurred while interacting with the database connection.", e); - throw new OpenBankingException("Error occurred while interacting with the database connection", e); - } finally { - DatabaseUtil.closeConnection(dbConnection); - } - return noOfRows; - } -} diff --git a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/queries/AccountMetadataDBQueries.java b/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/queries/AccountMetadataDBQueries.java deleted file mode 100644 index 56cde3ac..00000000 --- a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/queries/AccountMetadataDBQueries.java +++ /dev/null @@ -1,83 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.account.metadata.service.dao.queries; - -/** - * AccountMetadataDBQueries - * Contains the queries used by the AccountMetadataDAOImpl. - */ -public interface AccountMetadataDBQueries { - - /** - * Returns the query to store account metadata. - * - * @return String - */ - String getStoreAccountMetadataPreparedStatement(); - - /** - * Returns the query to update account metadata. - * - * @return String - */ - String getUpdateAccountMetadataPreparedStatement(); - - /** - * Returns the query to retrieve account metadata. - * - * @return String - */ - String getRetrieveAccountMetadataPreparedStatement(); - - /** - * Returns the query to retrieve user-ids and metadata values when - * account-id and metadata-key is given. - * - * @return String - */ - String getRetrieveMetadataByAccountIdAndKeyPreparedStatement(); - - /** - * Returns the query to retrieve the account metadata by key. - * - * @return String - */ - String getRetrieveAccountMetadataByKeyPreparedStatement(); - - /** - * Returns the query to delete the account metadata. - * - * @return String - */ - String getDeleteAccountMetadataPreparedStatement(); - - /** - * Returns the query to delete the account metadata by key. - * - * @return String - */ - String getDeleteAccountMetadataByKeyPreparedStatement(); - - /** - * Returns the query to delete the account metadata by key for al users. - * - * @return String - */ - String getDeleteAccountMetadataByKeyForAllUsersPreparedStatement(); -} diff --git a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/queries/AccountMetadataDBQueriesMySQLImpl.java b/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/queries/AccountMetadataDBQueriesMySQLImpl.java deleted file mode 100644 index e82bef48..00000000 --- a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/queries/AccountMetadataDBQueriesMySQLImpl.java +++ /dev/null @@ -1,99 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.account.metadata.service.dao.queries; - -/** - * AccountMetadataDBQueriesMySQLImpl - * Contains the MySQL queries used by the AccountMetadataDAOImpl. - */ -public class AccountMetadataDBQueriesMySQLImpl implements AccountMetadataDBQueries { - - /** - * {@inheritDoc} - */ - public String getStoreAccountMetadataPreparedStatement() { - - return "INSERT INTO OB_ACCOUNT_METADATA (ACCOUNT_ID, USER_ID, METADATA_KEY, METADATA_VALUE, " + - "LAST_UPDATED_TIMESTAMP) VALUES (?, ?, ?, ?, ?)"; - } - - /** - * {@inheritDoc} - */ - public String getUpdateAccountMetadataPreparedStatement() { - - return "UPDATE OB_ACCOUNT_METADATA SET METADATA_VALUE = ?, LAST_UPDATED_TIMESTAMP = ? WHERE ACCOUNT_ID = ? " + - "AND USER_ID = ? AND METADATA_KEY = ?"; - } - - /** - * {@inheritDoc} - */ - public String getRetrieveAccountMetadataPreparedStatement() { - - return "SELECT METADATA_KEY, METADATA_VALUE FROM OB_ACCOUNT_METADATA WHERE ACCOUNT_ID = ? AND USER_ID = ?"; - - } - - /** - * {@inheritDoc} - */ - public String getRetrieveMetadataByAccountIdAndKeyPreparedStatement() { - - return "SELECT USER_ID, METADATA_VALUE FROM OB_ACCOUNT_METADATA WHERE ACCOUNT_ID = ? AND METADATA_KEY = ?"; - - } - - /** - * {@inheritDoc} - */ - public String getRetrieveAccountMetadataByKeyPreparedStatement() { - - return "SELECT METADATA_VALUE FROM OB_ACCOUNT_METADATA WHERE ACCOUNT_ID = ? AND USER_ID = ? AND " + - "METADATA_KEY = ?"; - - } - - /** - * {@inheritDoc} - */ - public String getDeleteAccountMetadataPreparedStatement() { - - return "DELETE FROM OB_ACCOUNT_METADATA WHERE ACCOUNT_ID = ? AND USER_ID = ?"; - - } - - /** - * {@inheritDoc} - */ - public String getDeleteAccountMetadataByKeyPreparedStatement() { - - return "DELETE FROM OB_ACCOUNT_METADATA WHERE ACCOUNT_ID = ? AND USER_ID = ? AND METADATA_KEY = ?"; - - } - - /** - * {@inheritDoc} - */ - public String getDeleteAccountMetadataByKeyForAllUsersPreparedStatement() { - - return "DELETE FROM OB_ACCOUNT_METADATA WHERE ACCOUNT_ID = ? AND METADATA_KEY = ?"; - - } -} diff --git a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/internal/AccountMetadataDataHolder.java b/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/internal/AccountMetadataDataHolder.java deleted file mode 100644 index 5d25bd7a..00000000 --- a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/internal/AccountMetadataDataHolder.java +++ /dev/null @@ -1,52 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.account.metadata.service.internal; - -import org.wso2.carbon.user.core.service.RealmService; - -/** - * AccountMetadata Data Holder. - */ -public class AccountMetadataDataHolder { - - private static AccountMetadataDataHolder instance = new AccountMetadataDataHolder(); - - private RealmService realmService; - - private AccountMetadataDataHolder() { - - } - - public static AccountMetadataDataHolder getInstance() { - - return instance; - } - - public RealmService getRealmService() { - - if (realmService == null) { - throw new RuntimeException("Realm Service is not available. Component did not start correctly."); - } - return realmService; - } - - void setRealmService(RealmService realmService) { - - this.realmService = realmService; - } -} diff --git a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/internal/AccountMetadataServiceComponent.java b/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/internal/AccountMetadataServiceComponent.java deleted file mode 100644 index 9fe71917..00000000 --- a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/internal/AccountMetadataServiceComponent.java +++ /dev/null @@ -1,88 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.account.metadata.service.internal; - -import com.wso2.openbanking.accelerator.account.metadata.service.service.AccountMetadataService; -import com.wso2.openbanking.accelerator.account.metadata.service.service.AccountMetadataServiceImpl; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.osgi.service.component.ComponentContext; -import org.osgi.service.component.annotations.Activate; -import org.osgi.service.component.annotations.Component; -import org.osgi.service.component.annotations.Deactivate; -import org.osgi.service.component.annotations.Reference; -import org.osgi.service.component.annotations.ReferenceCardinality; -import org.osgi.service.component.annotations.ReferencePolicy; -import org.wso2.carbon.context.PrivilegedCarbonContext; -import org.wso2.carbon.user.core.service.RealmService; - -/** - * AccountMetadataService component. - */ -@Component( - name = "open.banking.account.metadata.service.component", - immediate = true -) -public class AccountMetadataServiceComponent { - - private static final Log log = LogFactory.getLog(AccountMetadataServiceComponent.class); - - public static RealmService getRealmService() { - return (RealmService) PrivilegedCarbonContext.getThreadLocalCarbonContext() - .getOSGiService(RealmService.class); - } - - @Reference( - name = "realm.service", - service = RealmService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetRealmService" - ) - protected void setRealmService(RealmService realmService) { - - log.debug("Setting the Realm Service"); - AccountMetadataDataHolder.getInstance().setRealmService(realmService); - } - - @Activate - protected void activate(ComponentContext ctxt) { - - try { - AccountMetadataService accountMetadataService = AccountMetadataServiceImpl.getInstance(); - ctxt.getBundleContext().registerService(AccountMetadataServiceImpl.class.getName(), - accountMetadataService, null); - log.debug("AccountMetadataService bundle is activated"); - - } catch (Throwable e) { - log.error("AccountMetadataService bundle activation Failed", e); - } - } - - @Deactivate - protected void deactivate(ComponentContext ctxt) { - - log.debug("AccountMetadataService bundle is deactivated"); - } - - protected void unsetRealmService(RealmService realmService) { - - log.debug("UnSetting the Realm Service"); - AccountMetadataDataHolder.getInstance().setRealmService(null); - } -} diff --git a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/service/AccountMetadataService.java b/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/service/AccountMetadataService.java deleted file mode 100644 index 2350e26b..00000000 --- a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/service/AccountMetadataService.java +++ /dev/null @@ -1,190 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.account.metadata.service.service; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; - -import java.util.Map; - -/** - * Account Metadata Service. - *

- * Handles the calls for persisting and retrieving metadata related to accounts. - */ -public interface AccountMetadataService { - - - /** - * Add or update multiple account metadata. - * If the key already exists for the account-id and user-id combination, the value be updated. - * - * @param accountId - Account ID - * @param userId - User ID - * @param accountMetadataMap - Map containing metadata key and value pairs - * @return number of records inserted/updated - * @throws OpenBankingException - OpenBankingException - */ - int addOrUpdateAccountMetadata(String accountId, String userId, Map accountMetadataMap) throws - OpenBankingException; - - /** - * Add or update multiple account metadata for account-id where the user id is N/A. - * If the key already exists for the account-id, the value be updated. - * - * @param accountId - Account ID - * @param accountMetadataMap - Map containing metadata key and value pairs - * @return number of records inserted/updated - * @throws OpenBankingException - OpenBankingException - */ - int addOrUpdateAccountMetadata(String accountId, Map accountMetadataMap) throws - OpenBankingException; - - /** - * Add or update account metadata. - * If the key already exists for the account-id and user-id combination, the value be updated. - * - * @param accountId - Account ID - * @param userId - User ID - * @param metadataKey - Metadata Key - * @param metadataValue - Metadata Value - * @return number of records inserted/updated - * @throws OpenBankingException - OpenBankingException - */ - int addOrUpdateAccountMetadata(String accountId, String userId, String metadataKey, String metadataValue) throws - OpenBankingException; - - /** - * Add or update metadata for account-id where the user id is N/A. - * If the key already exists for the account-id, the value be updated. - * - * @param accountId - Account ID - * @param metadataKey - Metadata Key - * @param metadataValue - Metadata Value - * @return number of records updated - * @throws OpenBankingException - OpenBankingException - */ - int addOrUpdateAccountMetadata(String accountId, String metadataKey, String metadataValue) throws - OpenBankingException; - - /** - * Get all metadata for an account-id user-id combination. - * - * @param accountId - Account ID - * @param userId - User ID - * @return Map of account metadata - * @throws OpenBankingException - OpenBankingException - */ - Map getAccountMetadataMap(String accountId, String userId) throws - OpenBankingException; - - /** - * Get all metadata affecting the account-id regardless of the user-id. - * - * @param accountId - Account ID - * @return Map of account metadata - * @throws OpenBankingException - OpenBankingException - */ - Map getAccountMetadataMap(String accountId) throws OpenBankingException; - - /** - * Get users and metadata values for an account-id and key combination. - * - * @param accountId - Account ID - * @param key - Metadata key - * @return Map of users and metadata values - * @throws OpenBankingException - OpenBankingException - */ - Map getUserMetadataForAccountIdAndKey(String accountId, String key) throws - OpenBankingException; - - /** - * Get metadata value for an account-id user-id and key combination. - * - * @param accountId - Account ID - * @param userId - User ID - * @param key - Metadata key - * @return Metadata value - * @throws OpenBankingException - OpenBankingException - */ - String getAccountMetadataByKey(String accountId, String userId, String key) throws - OpenBankingException; - - /** - * Given the key, get metadata value of the account-id where the user-id is N/A. - * - * @param accountId - Account ID - * @param key - Metadata key - * @return Metadata value - * @throws OpenBankingException - OpenBankingException - */ - String getAccountMetadataByKey(String accountId, String key) throws OpenBankingException; - - /** - * Remove all metadata for an account-id user-id combination. - * - * @param accountId - Account ID - * @param userId - User ID - * @return number of affected rows - * @throws OpenBankingException - OpenBankingException - */ - int removeAccountMetadata(String accountId, String userId) throws OpenBankingException; - - /** - * Remove all metadata for an account-id where the user-id is N/A. - * - * @param accountId - Account ID - * @return number of affected rows - * @throws OpenBankingException - OpenBankingException - */ - int removeAccountMetadata(String accountId) throws OpenBankingException; - - /** - * Remove metadata for an account-id user-id and key combination. - * - * @param accountId - Account ID - * @param userId - User ID - * @param key - Metadata key - * @return number of affected rows - * @throws OpenBankingException - OpenBankingException - */ - int removeAccountMetadataByKey(String accountId, String userId, String key) throws - OpenBankingException; - - /** - * Remove metadata for an account-id and key combination for all user-ids. - * - * @param accountId - Account ID - * @param key - Metadata key - * @return number of affected rows - * @throws OpenBankingException - OpenBankingException - */ - int removeAccountMetadataByKeyForAllUsers(String accountId, String key) throws - OpenBankingException; - - /** - * Given the key, remove metadata affecting the account-id where the user-id is N/A. - * - * @param accountId - Account ID - * @param key - Metadata key - * @return number of affected rows - * @throws OpenBankingException - OpenBankingException - */ - int removeAccountMetadataByKey(String accountId, String key) throws OpenBankingException; - -} diff --git a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/service/AccountMetadataServiceImpl.java b/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/service/AccountMetadataServiceImpl.java deleted file mode 100644 index 72db3fe5..00000000 --- a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/main/java/com/wso2/openbanking/accelerator/account/metadata/service/service/AccountMetadataServiceImpl.java +++ /dev/null @@ -1,240 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.account.metadata.service.service; - -import com.wso2.openbanking.accelerator.account.metadata.service.dao.AccountMetadataDAO; -import com.wso2.openbanking.accelerator.account.metadata.service.dao.AccountMetadataDAOImpl; -import com.wso2.openbanking.accelerator.account.metadata.service.dao.queries.AccountMetadataDBQueriesMySQLImpl; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.DatabaseUtil; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.util.Map; - -/** - * Implementation of AccountMetadataService. - */ -public class AccountMetadataServiceImpl implements AccountMetadataService { - - private static final Log log = LogFactory.getLog(AccountMetadataServiceImpl.class); - private static final String NOT_APPLICABLE = "N/A"; - private static AccountMetadataServiceImpl instance = null; - AccountMetadataDAO accountMetadataDAO = new AccountMetadataDAOImpl( - new AccountMetadataDBQueriesMySQLImpl()); - - // private constructor - private AccountMetadataServiceImpl() { - } - - /** - * @return AccountMetadataServiceImpl instance - */ - public static synchronized AccountMetadataServiceImpl getInstance() { - - if (instance == null) { - instance = new AccountMetadataServiceImpl(); - } - return instance; - } - - /** - * {@inheritDoc} - */ - @Override - public int addOrUpdateAccountMetadata(String accountId, String userId, Map accountMetadataMap) - throws OpenBankingException { - - if (StringUtils.isBlank(accountId) || StringUtils.isBlank(userId)) { - log.error("Account Id or User Id is not provided."); - throw new OpenBankingException("Account Id or User Id is not provided."); - } - if (accountMetadataMap == null || accountMetadataMap.isEmpty()) { - log.error("Account metadata is not present."); - throw new OpenBankingException("Account metadata is not present"); - } - int noOfRecords = 0; - // Add all entries in the accountMetadataMap to the database - for (Map.Entry accountMetadata : accountMetadataMap.entrySet()) { - addOrUpdateAccountMetadata(accountId, userId, accountMetadata.getKey(), accountMetadata.getValue()); - noOfRecords++; - } - return noOfRecords; - } - - @Override - public int addOrUpdateAccountMetadata(String accountId, Map accountMetadataMap) - throws OpenBankingException { - return addOrUpdateAccountMetadata(accountId, NOT_APPLICABLE, accountMetadataMap); - } - - /** - * {@inheritDoc} - */ - @Override - public int addOrUpdateAccountMetadata(String accountId, String userId, String metadataKey, String metadataValue) - throws OpenBankingException { - - if (StringUtils.isBlank(accountId) || StringUtils.isBlank(userId)) { - log.error("Account Id or User Id is not provided."); - throw new OpenBankingException("Account Id or User Id is not provided."); - } - if (StringUtils.isBlank(metadataKey) || StringUtils.isBlank(metadataValue)) { - log.error("Account metadata is not present."); - throw new OpenBankingException("Account metadata is not present"); - } - Connection dbConnection = DatabaseUtil.getDBConnection(); - // Check if the record is already present in the database. - if (getAccountMetadataByKey(accountId, userId, metadataKey) == null) { - // Add the record - return accountMetadataDAO.storeAccountMetadata(dbConnection, accountId, userId, metadataKey, metadataValue); - } else { - // Update the record - return accountMetadataDAO.updateAccountMetadata(dbConnection, accountId, userId, metadataKey, - metadataValue); - } - } - - /** - * {@inheritDoc} - */ - @Override - public int addOrUpdateAccountMetadata(String accountId, String metadataKey, String metadataValue) - throws OpenBankingException { - return addOrUpdateAccountMetadata(accountId, NOT_APPLICABLE, metadataKey, metadataValue); - } - - /** - * {@inheritDoc} - */ - @Override - public Map getAccountMetadataMap(String accountId, String userId) - throws OpenBankingException { - if (StringUtils.isBlank(userId) || StringUtils.isBlank(accountId)) { - log.error("Account Id or User Id is not provided."); - throw new OpenBankingException("Account Id or User Id is not provided."); - } - Connection dbConnection = DatabaseUtil.getDBConnection(); - return accountMetadataDAO.getAccountMetadataMap(dbConnection, accountId, userId); - } - - /** - * {@inheritDoc} - */ - @Override - public Map getAccountMetadataMap(String accountId) throws OpenBankingException { - return getAccountMetadataMap(accountId, NOT_APPLICABLE); - } - - /** - * {@inheritDoc} - */ - @Override - public Map getUserMetadataForAccountIdAndKey(String accountId, String key) - throws OpenBankingException { - if (StringUtils.isBlank(accountId) || StringUtils.isBlank(key)) { - log.error("Account Id or Key is not provided."); - throw new OpenBankingException("Account Id or Key is not provided."); - } - Connection dbConnection = DatabaseUtil.getDBConnection(); - return accountMetadataDAO.getMetadataForAccountIdAndKey(dbConnection, accountId, key); - } - - /** - * {@inheritDoc} - */ - @Override - public String getAccountMetadataByKey(String accountId, String userId, String key) - throws OpenBankingException { - if (StringUtils.isBlank(accountId) || StringUtils.isBlank(userId) || StringUtils.isBlank(key)) { - log.error("Account Id, User Id or Key is not provided."); - throw new OpenBankingException("Account Id, User Id or Key is not provided."); - } - Connection dbConnection = DatabaseUtil.getDBConnection(); - return accountMetadataDAO.getAccountMetadataByKey(dbConnection, accountId, userId, key); - } - - /** - * {@inheritDoc} - */ - @Override - public String getAccountMetadataByKey(String accountId, String key) throws OpenBankingException { - return getAccountMetadataByKey(accountId, NOT_APPLICABLE, key); - } - - /** - * {@inheritDoc} - */ - @Override - public int removeAccountMetadata(String accountId, String userId) throws OpenBankingException { - if (StringUtils.isBlank(accountId) || StringUtils.isBlank(userId)) { - log.error("Account Id or User Id is not provided."); - throw new OpenBankingException("Account Id or User Id is not provided."); - } - Connection dbConnection = DatabaseUtil.getDBConnection(); - return accountMetadataDAO.deleteAccountMetadata(dbConnection, accountId, userId); - } - - /** - * {@inheritDoc} - */ - @Override - public int removeAccountMetadata(String accountId) throws OpenBankingException { - return removeAccountMetadata(accountId, NOT_APPLICABLE); - } - - /** - * {@inheritDoc} - */ - @Override - public int removeAccountMetadataByKey(String accountId, String userId, String key) throws - OpenBankingException { - if (StringUtils.isBlank(accountId) || StringUtils.isBlank(userId) || StringUtils.isBlank(key)) { - log.error("Account Id, User Id or Key is not provided."); - throw new OpenBankingException("Account Id, User Id or Key is not provided."); - } - Connection dbConnection = DatabaseUtil.getDBConnection(); - return accountMetadataDAO.deleteAccountMetadataByKey(dbConnection, accountId, userId, key); - } - - /** - * {@inheritDoc} - */ - @Override - public int removeAccountMetadataByKeyForAllUsers(String accountId, String key) throws - OpenBankingException { - if (StringUtils.isBlank(accountId) || StringUtils.isBlank(key)) { - log.error("Account Id or Key is not provided."); - throw new OpenBankingException("Account Id or Key is not provided."); - } - Connection dbConnection = DatabaseUtil.getDBConnection(); - return accountMetadataDAO.deleteAccountMetadataByKeyForAllUsers(dbConnection, accountId, key); - } - - /** - * {@inheritDoc} - */ - @Override - public int removeAccountMetadataByKey(String accountId, String key) throws OpenBankingException { - return removeAccountMetadataByKey(accountId, NOT_APPLICABLE, key); - } - -} diff --git a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/AccountMetadataDAOTests.java b/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/AccountMetadataDAOTests.java deleted file mode 100644 index a4a0d2e8..00000000 --- a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/java/com/wso2/openbanking/accelerator/account/metadata/service/dao/AccountMetadataDAOTests.java +++ /dev/null @@ -1,315 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.account.metadata.service.dao; - -import com.wso2.openbanking.accelerator.account.metadata.service.dao.queries.AccountMetadataDBQueriesMySQLImpl; -import com.wso2.openbanking.accelerator.account.metadata.service.util.AccountMetadataDAOTestData; -import com.wso2.openbanking.accelerator.account.metadata.service.util.DAOUtils; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.DataProvider; -import org.testng.annotations.Test; - -import java.sql.Connection; -import java.util.Map; - -/** - * Implementation of AccountMetadataDAOTests class. - */ -public class AccountMetadataDAOTests { - - private static final String DB_NAME = "OPENBANKING_DB"; - private AccountMetadataDAO accountMetadataDAO; - - @BeforeClass - public void initTest() throws Exception { - - DAOUtils.initializeDataSource(DB_NAME, DAOUtils.getFilePath("dbScripts/h2.sql")); - accountMetadataDAO = new AccountMetadataDAOImpl(new AccountMetadataDBQueriesMySQLImpl()); - } - - @DataProvider(name = "accountMetadataDataProvider") - public Object[][] accountMetadataData() { - return AccountMetadataDAOTestData.DataProviders.METADATA_DATA_HOLDER; - } - - @DataProvider(name = "getAccountMetadataDataProvider") - public Object[][] getAccountMetadataData() { - return AccountMetadataDAOTestData.DataProviders.GET_METADATA_DATA_HOLDER; - } - - @Test - public void testStoreAccountMetadata() throws Exception { - - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String userId = AccountMetadataDAOTestData.SAMPLE_USER_ID; - Map metadataMap = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ATTRIBUTES_MAP; - int affectedRows = 0; - - for (Map.Entry entry : metadataMap.entrySet()) { - String metadataKey = entry.getKey(); - String metadataValue = entry.getValue(); - Connection dbConnection = DAOUtils.getConnection(DB_NAME); - affectedRows += accountMetadataDAO.storeAccountMetadata(dbConnection, accountId, userId, metadataKey, - metadataValue); - - } - Assert.assertEquals(affectedRows, 4); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testStoreAccountMetadataNullAccountIdAndUserIdError() throws Exception { - - String key = AccountMetadataDAOTestData.SAMPLE_KEY; - String value = AccountMetadataDAOTestData.SAMPLE_VALUE; - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - accountMetadataDAO.storeAccountMetadata(dbConnection, null, null, key, value); - } - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testStoreAccountMetadataEmptyMetadataMapError() throws Exception { - - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String userId = AccountMetadataDAOTestData.SAMPLE_USER_ID; - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - accountMetadataDAO.storeAccountMetadata(dbConnection, accountId, userId, "", ""); - } - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testStoreAccountMetadataNullMetadataMapError() throws Exception { - - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String userId = AccountMetadataDAOTestData.SAMPLE_USER_ID; - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - accountMetadataDAO.storeAccountMetadata(dbConnection, accountId, userId, null, null); - } - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testStoreAccountMetadataNullDBConnectionError() throws Exception { - - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String userId = AccountMetadataDAOTestData.SAMPLE_USER_ID; - String key = AccountMetadataDAOTestData.SAMPLE_KEY; - String value = AccountMetadataDAOTestData.SAMPLE_VALUE; - accountMetadataDAO.storeAccountMetadata(null, accountId, userId, key, value); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testUpdateAccountMetadataNullAccountIdAndUserIdError() throws Exception { - - String key = AccountMetadataDAOTestData.SAMPLE_KEY; - String value = AccountMetadataDAOTestData.SAMPLE_VALUE; - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - accountMetadataDAO.updateAccountMetadata(dbConnection, null, null, key, value); - } - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testUpdateAccountMetadataEmptyMetadataMapError() throws Exception { - - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String userId = AccountMetadataDAOTestData.SAMPLE_USER_ID; - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - accountMetadataDAO.updateAccountMetadata(dbConnection, accountId, userId, "", ""); - } - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testUpdateAccountMetadataNullMetadataMapError() throws Exception { - - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String userId = AccountMetadataDAOTestData.SAMPLE_USER_ID; - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - accountMetadataDAO.updateAccountMetadata(dbConnection, accountId, userId, null, null); - } - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testUpdateAccountMetadataNullDBConnectionError() throws Exception { - - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String userId = AccountMetadataDAOTestData.SAMPLE_USER_ID; - String key = AccountMetadataDAOTestData.SAMPLE_KEY; - String value = AccountMetadataDAOTestData.SAMPLE_VALUE; - accountMetadataDAO.updateAccountMetadata(null, accountId, userId, key, value); - } - - @Test(dataProvider = "getAccountMetadataDataProvider", dependsOnMethods = {"testStoreAccountMetadata"}, - priority = 1) - public void testGetAccountMetadata(String accountId, String userId) throws Exception { - - Map metadataMap; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - metadataMap = accountMetadataDAO.getAccountMetadataMap(dbConnection, accountId, userId); - } - Assert.assertEquals(metadataMap.size(), 4); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testGetAccountMetadataNullAccountIdAndUserIdError() throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - accountMetadataDAO.getAccountMetadataMap(dbConnection, null, null); - } - } - - @Test(expectedExceptions = OpenBankingException.class, dataProvider = "getAccountMetadataDataProvider", - dependsOnMethods = {"testStoreAccountMetadata"}, - priority = 1) - public void testGetAccountMetadataNullDBConnectionError(String accountId, String userId) throws Exception { - accountMetadataDAO.getAccountMetadataMap(null, accountId, userId); - } - - @Test(dataProvider = "accountMetadataDataProvider", dependsOnMethods = {"testStoreAccountMetadata"}, - priority = 1) - public void testGetAccountMetadataByKey(String accountId, String userId, String key, String value) - throws Exception { - - String metadataValue; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - metadataValue = accountMetadataDAO.getAccountMetadataByKey(dbConnection, accountId, userId, key); - } - Assert.assertEquals(metadataValue, value); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testGetAccountMetadataByKeyNullAccountIdUserIdAndKeyError() throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - accountMetadataDAO.getAccountMetadataByKey(dbConnection, null, null, null); - } - } - - @Test(expectedExceptions = OpenBankingException.class, dataProvider = "accountMetadataDataProvider") - public void testGetAccountMetadataByKeyNullDBConnectionError(String accountId, String userId, String key, - String value) throws Exception { - accountMetadataDAO.getAccountMetadataByKey(null, accountId, userId, key); - } - - @Test(dataProvider = "accountMetadataDataProvider", dependsOnMethods = {"testStoreAccountMetadata"}, - priority = 2) - public void testDeleteAccountMetadataByKey(String accountId, String userId, String key, String value) - throws Exception { - - int affectedRows; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - affectedRows = accountMetadataDAO.deleteAccountMetadataByKey(dbConnection, accountId, userId, key); - } - Assert.assertEquals(affectedRows, 1); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testDeleteAccountMetadataByKeyNullAccountIdUserIdAndKeyError() throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - accountMetadataDAO.deleteAccountMetadataByKey(dbConnection, null, null, null); - } - } - - @Test(expectedExceptions = OpenBankingException.class, dataProvider = "accountMetadataDataProvider") - public void testDeleteAccountMetadataByKeyNullDBConnectionError(String accountId, String userId, String key, - String value) throws Exception { - - accountMetadataDAO.deleteAccountMetadataByKey(null, accountId, userId, key); - } - - @Test(dependsOnMethods = {"testStoreAccountMetadata", "testDeleteAccountMetadataByKey"}, priority = 2) - public void testDeleteAccountMetadata() throws Exception { - - int affectedRows; - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String userId = AccountMetadataDAOTestData.SAMPLE_USER_ID; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - affectedRows = accountMetadataDAO.deleteAccountMetadata(dbConnection, accountId, userId); - } - Assert.assertEquals(affectedRows, 3); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testDeleteAccountMetadataNullAccountIdAndUserIdError() throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - accountMetadataDAO.deleteAccountMetadata(dbConnection, null, null); - } - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testDeleteAccountMetadataNullDBConnection() throws Exception { - - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String userId = AccountMetadataDAOTestData.SAMPLE_USER_ID; - accountMetadataDAO.deleteAccountMetadata(null, accountId, userId); - } - - @Test - public void testStoreAccountMetadataForSameAccount() throws Exception { - int affectedRows = 0; - Map userAttributeAMp = AccountMetadataDAOTestData.SAMPLE_USER_ID_ATTRIBUTE_VALUE_MAP; - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String attributeKey = AccountMetadataDAOTestData.SAMPLE_KEY; - - for (Map.Entry entry : userAttributeAMp.entrySet()) { - Connection dbConnection = DAOUtils.getConnection(DB_NAME); - String userId = entry.getKey(); - String attributeValue = entry.getValue(); - affectedRows += accountMetadataDAO.storeAccountMetadata(dbConnection, accountId, userId, attributeKey, - attributeValue); - } - Assert.assertEquals(affectedRows, 4); - } - - @Test(dependsOnMethods = {"testStoreAccountMetadataForSameAccount"}, priority = 2) - public void testDeleteAccountMetadataByKeyForAllUsers() throws Exception { - - int affectedRows; - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String key = AccountMetadataDAOTestData.SAMPLE_KEY; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - affectedRows = accountMetadataDAO.deleteAccountMetadataByKeyForAllUsers(dbConnection, accountId, key); - } - Assert.assertEquals(affectedRows, 4); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testDeleteAccountMetadataByKeyForAllUsersNullAccountIdAndKeyError() throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - accountMetadataDAO.deleteAccountMetadataByKeyForAllUsers(dbConnection, null, null); - } - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testDeleteAccountMetadataByKeyForAllUsersNullDBConnection() throws Exception { - - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String userId = AccountMetadataDAOTestData.SAMPLE_USER_ID; - accountMetadataDAO.deleteAccountMetadataByKeyForAllUsers(null, accountId, userId); - } - -} diff --git a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/java/com/wso2/openbanking/accelerator/account/metadata/service/service/AccountMetadataServiceTests.java b/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/java/com/wso2/openbanking/accelerator/account/metadata/service/service/AccountMetadataServiceTests.java deleted file mode 100644 index 330664d7..00000000 --- a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/java/com/wso2/openbanking/accelerator/account/metadata/service/service/AccountMetadataServiceTests.java +++ /dev/null @@ -1,610 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.account.metadata.service.service; - -import com.wso2.openbanking.accelerator.account.metadata.service.util.AccountMetadataDAOTestData; -import com.wso2.openbanking.accelerator.account.metadata.service.util.DAOUtils; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.DatabaseUtil; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.DataProvider; -import org.testng.annotations.Test; - -import java.sql.Connection; -import java.util.Collections; -import java.util.HashMap; -import java.util.Map; - -/** - * Implementation of AccountMetadataServiceTests class. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({DatabaseUtil.class}) -public class AccountMetadataServiceTests extends PowerMockTestCase { - - private static final String DB_NAME = "OPENBANKING_DB"; - private AccountMetadataService accountMetadataService; - - @BeforeClass - public void initTest() throws Exception { - DAOUtils.initializeDataSource(DB_NAME, DAOUtils.getFilePath("dbScripts/h2.sql")); - accountMetadataService = AccountMetadataServiceImpl.getInstance(); - } - - @DataProvider(name = "accountMetadataDataProvider") - public Object[][] accountMetadataData() { - return AccountMetadataDAOTestData.DataProviders.METADATA_DATA_HOLDER; - } - - @DataProvider(name = "globalAccountMetadataDataProvider") - public Object[][] globalAccountMetadataData() { - return AccountMetadataDAOTestData.DataProviders.GLOBAL_METADATA_DATA_HOLDER; - } - - @DataProvider(name = "getAccountMetadataDataProvider") - public Object[][] getAccountMetadataData() { - return AccountMetadataDAOTestData.DataProviders.GET_METADATA_DATA_HOLDER; - } - - @Test - public void testAddOrUpdateAccountMetadata() throws Exception { - - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String userId = AccountMetadataDAOTestData.SAMPLE_USER_ID; - Map metadataMap = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ATTRIBUTES_MAP; - int noOfEntries; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - noOfEntries = accountMetadataService.addOrUpdateAccountMetadata(accountId, userId, metadataMap); - - } - Assert.assertEquals(noOfEntries, 4); - } - - @Test(dependsOnMethods = {"testAddOrUpdateAccountMetadata"}, priority = 1) - public void testUpdateExistingAccountMetadata() throws Exception { - - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String userId = AccountMetadataDAOTestData.SAMPLE_USER_ID; - String updateMetadataKey = "secondary-account-privilege"; - String updateMetadataValue = "active"; - int noOfEntries; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - noOfEntries = accountMetadataService.addOrUpdateAccountMetadata(accountId, userId, updateMetadataKey, - updateMetadataValue); - - } - Assert.assertEquals(noOfEntries, 1); - } - - @Test - public void testAddOrUpdateAccountMetadataForSameAccount() throws Exception { - int noOfEntries = 0; - Map userAttributeAMp = AccountMetadataDAOTestData.SAMPLE_USER_ID_ATTRIBUTE_VALUE_MAP; - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String attributeKey = AccountMetadataDAOTestData.SAMPLE_KEY; - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - for (Map.Entry entry : userAttributeAMp.entrySet()) { - String userId = entry.getKey(); - String attributeValue = entry.getValue(); - Map metadataMap = Collections.singletonMap(attributeKey, attributeValue); - noOfEntries += accountMetadataService.addOrUpdateAccountMetadata(accountId, userId, - metadataMap); - } - } - Assert.assertEquals(noOfEntries, 4); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testAddOrUpdateAccountMetadataNullAccountIdError() throws Exception { - - String userId = AccountMetadataDAOTestData.SAMPLE_USER_ID; - Map metadataMap = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ATTRIBUTES_MAP; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.addOrUpdateAccountMetadata(null, userId, metadataMap); - } - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testAddOrUpdateAccountMetadataNullMetadataError() throws Exception { - - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String userId = AccountMetadataDAOTestData.SAMPLE_USER_ID; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.addOrUpdateAccountMetadata(accountId, userId, ""); - - } - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testAddOrUpdateAccountMetadataEmptyMetadataError() throws Exception { - - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String userId = AccountMetadataDAOTestData.SAMPLE_USER_ID; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.addOrUpdateAccountMetadata(accountId, userId, new HashMap<>()); - - } - } - - @Test - public void testAddOrUpdateGlobalAccountMetadataMap() throws Exception { - - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - Map metadataMap = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ATTRIBUTES_MAP; - int noOfEntries; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - noOfEntries = accountMetadataService.addOrUpdateAccountMetadata(accountId, metadataMap); - - } - Assert.assertEquals(noOfEntries, 4); - } - - @Test(dependsOnMethods = {"testAddOrUpdateGlobalAccountMetadataMap"}) - public void testUpdateGlobalAccountMetadata() throws Exception { - - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String updateMetadataKey = "secondary-account-privilege"; - String updateMetadataValue = "active"; int noOfEntries; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - noOfEntries = accountMetadataService.addOrUpdateAccountMetadata(accountId, updateMetadataKey, - updateMetadataValue); - - } - Assert.assertEquals(noOfEntries, 1); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testAddOrUpdateGlobalAccountMetadataNullAccountIdError() throws Exception { - - Map metadataMap = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ATTRIBUTES_MAP; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.addOrUpdateAccountMetadata(null, metadataMap); - } - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testAddOrUpdateGlobalAccountMetadataNullMetadataError() throws Exception { - - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.addOrUpdateAccountMetadata(accountId, null); - - } - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testAddOrUpdateGlobalAccountMetadataEmptyMetadataError() throws Exception { - - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.addOrUpdateAccountMetadata(accountId, new HashMap<>()); - - } - } - - @Test(dataProvider = "getAccountMetadataDataProvider", dependsOnMethods = {"testAddOrUpdateAccountMetadata"}, - priority = 1) - public void testGetAccountMetadataMap(String accountId, String userId) throws Exception { - - Map metadataMap; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - metadataMap = accountMetadataService.getAccountMetadataMap(accountId, userId); - - } - Assert.assertEquals(metadataMap.size(), 4); - } - - @Test(expectedExceptions = OpenBankingException.class, dataProvider = "getAccountMetadataDataProvider") - public void testGetAccountMetadataMapNullAccountIdError(String accountId, String userId) throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.getAccountMetadataMap(null, userId); - - } - } - - @Test(expectedExceptions = OpenBankingException.class, dataProvider = "getAccountMetadataDataProvider") - public void testGetAccountMetadataMapNullUserIdError(String accountId, String userId) throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.getAccountMetadataMap(accountId, null); - - } - } - - @Test(dependsOnMethods = {"testAddOrUpdateGlobalAccountMetadataMap"}, priority = 1) - public void testGetGlobalAccountMetadataMap() throws Exception { - - Map metadataMap; - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - metadataMap = accountMetadataService.getAccountMetadataMap(accountId); - - } - Assert.assertEquals(metadataMap.size(), 4); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testGetGlobalAccountMetadataMapNullAccountIdError() throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.getAccountMetadataMap(null); - - } - } - - @Test(dependsOnMethods = {"testAddOrUpdateAccountMetadataForSameAccount"}, priority = 1) - public void testGetUserAttributesForAccountIdAndKey() throws Exception { - - Map metadataMap; - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String attributeKey = AccountMetadataDAOTestData.SAMPLE_KEY; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - metadataMap = accountMetadataService.getUserMetadataForAccountIdAndKey(accountId, attributeKey); - - } - Assert.assertEquals(metadataMap.size(), 4); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testGetUserAttributesForAccountIdAndKeyNullAccountIdError() throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.getUserMetadataForAccountIdAndKey(null, - AccountMetadataDAOTestData.SAMPLE_KEY); - - } - } - - @Test(dataProvider = "accountMetadataDataProvider", dependsOnMethods = {"testAddOrUpdateAccountMetadata"}, - priority = 1) - public void testGetAccountMetadataByKey(String accountId, String userId, String key, String value) - throws Exception { - - String metadataValue; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - metadataValue = accountMetadataService.getAccountMetadataByKey(accountId, userId, key); - } - Assert.assertEquals(metadataValue, value); - } - - @Test(expectedExceptions = OpenBankingException.class, dataProvider = "accountMetadataDataProvider") - public void testGetAccountMetadataByKeyNullAccountIdError(String accountId, String userId, String key, String value) - throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.getAccountMetadataByKey(null, userId, key); - } - } - - @Test(expectedExceptions = OpenBankingException.class, dataProvider = "accountMetadataDataProvider") - public void testGetAccountMetadataByKeyNullUserIdError(String accountId, String userId, String key, String value) - throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.getAccountMetadataByKey(accountId, null, key); - } - } - - @Test(expectedExceptions = OpenBankingException.class, dataProvider = "accountMetadataDataProvider") - public void testGetAccountMetadataByKeyNullKeyError(String accountId, String userId, String key, String value) - throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.getAccountMetadataByKey(accountId, userId, null); - } - } - - @Test(dataProvider = "globalAccountMetadataDataProvider", dependsOnMethods = - {"testAddOrUpdateGlobalAccountMetadataMap"}, - priority = 1) - public void testGetGlobalAccountMetadataByKey(String accountId, String userId, String key, String value) - throws Exception { - - String metadataValue; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - metadataValue = accountMetadataService.getAccountMetadataByKey(accountId, key); - } - Assert.assertEquals(metadataValue, value); - } - - @Test(expectedExceptions = OpenBankingException.class, dataProvider = "globalAccountMetadataDataProvider") - public void testGetGlobalAccountMetadataByKeyNullAccountIdError(String accountId, String userId, String key, - String value) throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.getAccountMetadataByKey(null, key); - } - } - - @Test(expectedExceptions = OpenBankingException.class, dataProvider = "globalAccountMetadataDataProvider") - public void testGetGlobalAccountMetadataByKeyNullKeyError(String accountId, String userId, String key, - String value) throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.getAccountMetadataByKey(accountId, null); - } - } - - @Test(dataProvider = "accountMetadataDataProvider", dependsOnMethods = {"testAddOrUpdateAccountMetadata"}, - priority = 2) - public void testDeleteAccountMetadataByKey(String accountId, String userId, String key, String value) - throws Exception { - - int affectedRows; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - affectedRows = accountMetadataService.removeAccountMetadataByKey(accountId, userId, key); - } - Assert.assertEquals(affectedRows, 1); - } - - @Test(expectedExceptions = OpenBankingException.class, dataProvider = "accountMetadataDataProvider") - public void testDeleteAccountMetadataByKeyNullAccountIdError(String accountId, String userId, String key, - String value) throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.removeAccountMetadataByKey(null, userId, key); - } - } - - @Test(expectedExceptions = OpenBankingException.class, dataProvider = "accountMetadataDataProvider") - public void testDeleteAccountMetadataByKeyNullUserIdError(String accountId, String userId, String key, - String value) throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.removeAccountMetadataByKey(accountId, null, key); - } - } - - @Test(expectedExceptions = OpenBankingException.class, dataProvider = "accountMetadataDataProvider") - public void testDeleteAccountMetadataByKeyNullKeyError(String accountId, String userId, String key, - String value) throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.removeAccountMetadataByKey(accountId, userId, null); - } - } - - @Test(dependsOnMethods = {"testAddOrUpdateAccountMetadata", "testDeleteAccountMetadataByKey"}, priority = 2) - public void testDeleteAccountMetadata() - throws Exception { - - int affectedRows; - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String userId = AccountMetadataDAOTestData.SAMPLE_USER_ID; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - affectedRows = accountMetadataService.removeAccountMetadata(accountId, userId); - } - Assert.assertEquals(affectedRows, 3); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testDeleteAccountMetadataNullAccountIdError() throws Exception { - - String userId = AccountMetadataDAOTestData.SAMPLE_USER_ID; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.removeAccountMetadata(null, userId); - } - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testDeleteAccountMetadataNullUserIdError() throws Exception { - - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.removeAccountMetadata(accountId, null); - } - } - - @Test(dataProvider = "globalAccountMetadataDataProvider", dependsOnMethods = - {"testAddOrUpdateGlobalAccountMetadataMap"}, - priority = 2) - public void testDeleteGlobalAccountMetadataByKey(String accountId, String userId, String key, String value) - throws Exception { - - int affectedRows; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - affectedRows = accountMetadataService.removeAccountMetadataByKey(accountId, key); - } - Assert.assertEquals(affectedRows, 1); - } - - @Test(expectedExceptions = OpenBankingException.class, dataProvider = "globalAccountMetadataDataProvider") - public void testDeleteGlobalAccountMetadataByKeyNullAccountIdError(String accountId, String userId, String key, - String value) throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.removeAccountMetadataByKey(null, key); - } - } - - @Test(expectedExceptions = OpenBankingException.class, dataProvider = "globalAccountMetadataDataProvider") - public void testDeleteGlobalAccountMetadataByKeyNullKeyError(String accountId, String userId, String key, - String value) throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.removeAccountMetadataByKey(accountId, null); - } - } - - @Test(dependsOnMethods = {"testAddOrUpdateGlobalAccountMetadataMap", "testDeleteGlobalAccountMetadataByKey"}, - priority = 2) - public void testDeleteGlobalAccountMetadata() - throws Exception { - - int affectedRows; - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - affectedRows = accountMetadataService.removeAccountMetadata(accountId); - } - Assert.assertEquals(affectedRows, 3); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testDeleteGlobalAccountMetadataNullAccountIdError() throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.removeAccountMetadata(null); - } - } - - @Test(dependsOnMethods = {"testAddOrUpdateAccountMetadataForSameAccount"}, priority = 2) - public void testDeleteAccountMetadataByKeyForAllUsers() throws Exception { - - int affectedRows; - String accountId = AccountMetadataDAOTestData.SAMPLE_ACCOUNT_ID; - String attributeKey = AccountMetadataDAOTestData.SAMPLE_KEY; - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - affectedRows = accountMetadataService.removeAccountMetadataByKeyForAllUsers(accountId, attributeKey); - } - Assert.assertEquals(affectedRows, 4); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testDeleteAccountMetadataByKeyForAllUsersNullUserIdError() throws Exception { - - try (Connection dbConnection = DAOUtils.getConnection(DB_NAME)) { - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()) - .thenReturn(dbConnection); - accountMetadataService.removeAccountMetadataByKeyForAllUsers(null, - AccountMetadataDAOTestData.SAMPLE_KEY); - } - } -} diff --git a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/java/com/wso2/openbanking/accelerator/account/metadata/service/util/AccountMetadataDAOTestData.java b/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/java/com/wso2/openbanking/accelerator/account/metadata/service/util/AccountMetadataDAOTestData.java deleted file mode 100644 index fa79385e..00000000 --- a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/java/com/wso2/openbanking/accelerator/account/metadata/service/util/AccountMetadataDAOTestData.java +++ /dev/null @@ -1,86 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.account.metadata.service.util; - - -import java.util.HashMap; -import java.util.Map; - -/** - * Implementation of AccountMetadataDAOTestData class. - */ -public class AccountMetadataDAOTestData { - - public static final String SAMPLE_ACCOUNT_ID = "account-1"; - public static final String SAMPLE_USER_ID = "ann@gold.com"; - public static final String SAMPLE_KEY = "bnr-permission"; - public static final String SAMPLE_VALUE = "active"; - public static final String GLOBAL = "GLOBAL"; - - public static final Map SAMPLE_ACCOUNT_ATTRIBUTES_MAP = new HashMap() { - { - put("disclosure-option", "pre-approved"); - put("other-accounts-availability", "true"); - put("secondary-account-instruction", "active"); - put("secondary-account-privilege", "inactive"); - } - }; - - public static final Map SAMPLE_USER_ID_ATTRIBUTE_VALUE_MAP = new HashMap() { - { - put("sample_user_id_1", "active"); - put("sample_user_id_2", "inactive"); - put("sample_user_id_3", "active"); - put("sample_user_id_4", "inactive"); - } - }; - - /** - * Implementation of AccountMetadataServiceTests class. - */ - public static final class DataProviders { - - public static final Object[][] METADATA_DATA_HOLDER = new Object[][]{ - - { - SAMPLE_ACCOUNT_ID, - SAMPLE_USER_ID, - "disclosure-option", - "pre-approved", - } - }; - - public static final Object[][] GLOBAL_METADATA_DATA_HOLDER = new Object[][]{ - - { - SAMPLE_ACCOUNT_ID, - GLOBAL, - "disclosure-option", - "pre-approved", - } - }; - - public static final Object[][] GET_METADATA_DATA_HOLDER = new Object[][]{ - - { - SAMPLE_ACCOUNT_ID, - SAMPLE_USER_ID, - } - }; - } -} diff --git a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/java/com/wso2/openbanking/accelerator/account/metadata/service/util/DAOUtils.java b/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/java/com/wso2/openbanking/accelerator/account/metadata/service/util/DAOUtils.java deleted file mode 100644 index 83ae7020..00000000 --- a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/java/com/wso2/openbanking/accelerator/account/metadata/service/util/DAOUtils.java +++ /dev/null @@ -1,64 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.account.metadata.service.util; - -import org.apache.commons.dbcp.BasicDataSource; -import org.apache.commons.lang3.StringUtils; - -import java.nio.file.Paths; -import java.sql.Connection; -import java.sql.SQLException; -import java.util.HashMap; -import java.util.Map; - -/** - * Implementation of DAOUtils class. - */ -public class DAOUtils { - - private static Map dataSourceMap = new HashMap<>(); - - public static void initializeDataSource(String databaseName, String scriptPath) throws Exception { - BasicDataSource dataSource = new BasicDataSource(); - dataSource.setDriverClassName("org.h2.Driver"); - dataSource.setUsername("username"); - dataSource.setPassword("password"); - dataSource.setUrl("jdbc:h2:mem:" + databaseName); - - try (Connection connection = dataSource.getConnection()) { - connection.createStatement().executeUpdate("RUNSCRIPT FROM '" + scriptPath + "'"); - } - dataSourceMap.put(databaseName, dataSource); - } - - public static Connection getConnection(String database) throws SQLException { - if (dataSourceMap.get(database) != null) { - return dataSourceMap.get(database).getConnection(); - } - throw new RuntimeException("Invalid datasource."); - } - - public static String getFilePath(String fileName) { - if (StringUtils.isNotBlank(fileName)) { - return Paths.get(System.getProperty("user.dir"), "src", "test", "resources", fileName) - .toString(); - } - return null; - } -} diff --git a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/resources/dbScripts/h2.sql b/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/resources/dbScripts/h2.sql deleted file mode 100644 index e79fb9a7..00000000 --- a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/resources/dbScripts/h2.sql +++ /dev/null @@ -1,8 +0,0 @@ -CREATE TABLE IF NOT EXISTS OB_ACCOUNT_METADATA ( - ACCOUNT_ID VARCHAR(100) NOT NULL, - USER_ID VARCHAR(100) NOT NULL, - METADATA_KEY VARCHAR(100) NOT NULL, - METADATA_VALUE VARCHAR(100) NOT NULL, - LAST_UPDATED_TIMESTAMP TIMESTAMP NOT NULL, - PRIMARY KEY (USER_ID,ACCOUNT_ID,METADATA_KEY) -); diff --git a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/resources/testng.xml b/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/resources/testng.xml deleted file mode 100644 index af4222bb..00000000 --- a/open-banking-accelerator/components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service/src/test/resources/testng.xml +++ /dev/null @@ -1,28 +0,0 @@ - - - - - - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/pom.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/pom.xml deleted file mode 100644 index 3e35d80f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/pom.xml +++ /dev/null @@ -1,219 +0,0 @@ - - - - - 4.0.0 - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.0-SNAPSHOT - ../../pom.xml - - com.wso2.openbanking.accelerator.ciba - bundle - WSO2 Open Banking - Common component - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.identity - provided - - - org.wso2.carbon.identity.inbound.auth.oauth2 - org.wso2.carbon.identity.oauth - ${identity.inbound.auth.oauth.ciba.updated.version} - - - org.wso2.carbon.identity.inbound.auth.oauth2 - org.wso2.carbon.identity.oauth.ciba - ${identity.inbound.auth.oauth.ciba.updated.version} - - - org.powermock - powermock-module-testng - - - org.powermock - powermock-api-mockito - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.consent.service - - - org.springframework - spring-test - - - org.springframework - spring-core - - - - - - - org.apache.maven.plugins - maven-compiler-plugin - - - org.jacoco - jacoco-maven-plugin - - - - **/*GrantHandler.class - - - - - default-prepare-agent - - prepare-agent - - - - default-prepare-agent-integration - - prepare-agent-integration - - - - default-report - - report - - - - default-report-integration - - report-integration - - - - default-check - - check - - - - - BUNDLE - - - INSTRUCTION - COVEREDRATIO - 0.86 - - - - - - - - - - org.apache.maven.plugins - maven-surefire-plugin - - - src/test/resources/testng.xml - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - false - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-exclude.xml - ${project.basedir}/src/main/resources/findbugs-include.xml - false - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - org.apache.felix - maven-bundle-plugin - true - - - - ${project.artifactId} - - - com.wso2.openbanking.accelerator.identity.internal - - - com.nimbusds.jwt; version="${nimbusds.osgi.version.range}", - com.wso2.openbanking.accelerator.common.exception; version="${project.version}", - com.wso2.openbanking.accelerator.common.util; version="${project.version}", - com.wso2.openbanking.accelerator.consent.mgt.service.impl; version="${project.version}", - net.minidev.json; version="${json-smart}", - org.apache.commons.logging; version="${commons.logging.version}", - org.wso2.carbon.identity.oauth.ciba.grant; version="${identity.inbound.auth.oauth.version.range}", - org.wso2.carbon.identity.oauth2; version="${identity.inbound.auth.oauth.version.range}", - org.wso2.carbon.identity.oauth2.dto; version="${identity.inbound.auth.oauth.version.range}", - org.wso2.carbon.identity.oauth2.model; version="${identity.inbound.auth.oauth.version.range}", - org.wso2.carbon.identity.oauth2.token; version="${identity.inbound.auth.oauth.version.range}", - org.wso2.carbon.identity.openidconnect; version="${identity.inbound.auth.oauth.version.range}", - org.wso2.carbon.identity.openidconnect.model; version="${identity.inbound.auth.oauth.version.range}", - - - !com.wso2.openbanking.accelerator.identity.internal, - com.wso2.openbanking.accelerator.ciba - - * - <_dsannotations>* - - - - - - - 6.4.111.52 - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/java/com.wso2.openbanking.accelerator.ciba/CIBAConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/java/com.wso2.openbanking.accelerator.ciba/CIBAConstants.java deleted file mode 100644 index 62c25ffc..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/java/com.wso2.openbanking.accelerator.ciba/CIBAConstants.java +++ /dev/null @@ -1,37 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.ciba; - -/** - * CIBA Test Constants class. - */ -public class CIBAConstants { - - public static final String INVALID_REQUEST = "invalid_request"; - public static final String INTENT_CLAIM = "openbanking_intent_id"; - public static final String VALUE_TAG = "value"; - public static final String CONSENT_ID_PREFIX = "OB_CONSENT_ID_"; - - //Error Messages - public static final String PARSE_ERROR_MESSAGE = - "Request object invalid: Unable to parse the request object as json"; - public static final String EMPTY_CONTENT_ERROR = "Request object invalid: Empty value for intent"; - public static final String MESSAGE_CONTEXT_EMPTY_ERROR = "OAuth Token Request Message Context is empty"; - public static final String SCOPE_ADDING_ERROR = "Error while adding consent ID to scopes"; -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/java/com.wso2.openbanking.accelerator.ciba/OBCIBARequestObjectValidationExtension.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/java/com.wso2.openbanking.accelerator.ciba/OBCIBARequestObjectValidationExtension.java deleted file mode 100644 index 7b0251dc..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/java/com.wso2.openbanking.accelerator.ciba/OBCIBARequestObjectValidationExtension.java +++ /dev/null @@ -1,77 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.ciba; - -import com.wso2.openbanking.accelerator.common.util.Generated; -import net.minidev.json.JSONObject; -import org.apache.commons.lang.StringUtils; -import org.wso2.carbon.identity.oauth2.RequestObjectException; -import org.wso2.carbon.identity.oauth2.model.OAuth2Parameters; -import org.wso2.carbon.identity.openidconnect.CIBARequestObjectValidatorImpl; -import org.wso2.carbon.identity.openidconnect.model.RequestObject; - -import java.text.ParseException; - -/** - * The extension of RequestObjectValidatorImpl to enforce Open Banking specific validations of the - * request object. - */ -public class OBCIBARequestObjectValidationExtension extends CIBARequestObjectValidatorImpl { - - /** - * Validations related to clientId, response type, exp, redirect URL, mandatory params, - * issuer, audience are done. Called after signature validation. - * - * @param initialRequestObject request object - * @param oAuth2Parameters oAuth2Parameters - * @throws RequestObjectException - RequestObjectException - */ - @Override - public boolean validateRequestObject(RequestObject initialRequestObject, OAuth2Parameters oAuth2Parameters) - throws RequestObjectException { - - JSONObject intent; - try { - intent = initialRequestObject.getClaimsSet().getJSONObjectClaim(CIBAConstants.INTENT_CLAIM); - } catch (ParseException e) { - throw new RequestObjectException(CIBAConstants.INVALID_REQUEST, - CIBAConstants.PARSE_ERROR_MESSAGE, e); - } - if (StringUtils.isEmpty(intent.getAsString(CIBAConstants.VALUE_TAG))) { - throw new RequestObjectException(CIBAConstants.INVALID_REQUEST, CIBAConstants.EMPTY_CONTENT_ERROR); - } - - return validateIAMConstraints(initialRequestObject, oAuth2Parameters); - } - - /** - * Validate IAM related logic. - * @param requestObject - * @param oAuth2Parameters - * @return is IAM related constraints are validate - * @throws RequestObjectException - */ - @Generated(message = "super methods cannot be mocked") - boolean validateIAMConstraints(RequestObject requestObject, - OAuth2Parameters oAuth2Parameters) throws RequestObjectException { - - return super.validateRequestObject(requestObject, oAuth2Parameters); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/java/com.wso2.openbanking.accelerator.ciba/OBCIBASignatureAlgorithmEnforcementValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/java/com.wso2.openbanking.accelerator.ciba/OBCIBASignatureAlgorithmEnforcementValidator.java deleted file mode 100644 index b01a769e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/java/com.wso2.openbanking.accelerator.ciba/OBCIBASignatureAlgorithmEnforcementValidator.java +++ /dev/null @@ -1,54 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.ciba; - -import com.wso2.openbanking.accelerator.identity.token.util.TokenFilterException; -import com.wso2.openbanking.accelerator.identity.token.validators.SignatureAlgorithmEnforcementValidator; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import org.wso2.carbon.identity.oauth.config.OAuthServerConfiguration; - -import javax.servlet.http.HttpServletResponse; - -/** - * CIBA Signature Algorithm Enforcer class - */ -public class OBCIBASignatureAlgorithmEnforcementValidator extends SignatureAlgorithmEnforcementValidator { - - /** - * CIBA and FAPI related validations for Signature Algorithm. - * @param requestSigningAlgorithm the algorithm of signed message - * @param registeredSigningAlgorithm the algorithm registered during client authentication - * @throws TokenFilterException - */ - @Override - public void validateInboundSignatureAlgorithm(String requestSigningAlgorithm, String registeredSigningAlgorithm) - throws TokenFilterException { - - super.validateInboundSignatureAlgorithm(requestSigningAlgorithm, registeredSigningAlgorithm); - if (OAuthServerConfiguration.getInstance().isFapiCiba()) { - if (!(IdentityCommonConstants.ALG_ES256.equals(requestSigningAlgorithm) || - IdentityCommonConstants.ALG_PS256.equals(requestSigningAlgorithm))) { - String message = "FAPI unsupported signing algorithm " + requestSigningAlgorithm - + " used to sign the JWT"; - throw new TokenFilterException(HttpServletResponse.SC_UNAUTHORIZED, IdentityCommonConstants - .OAUTH2_INVALID_CLIENT_MESSAGE, message); - } - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/java/com.wso2.openbanking.accelerator.ciba/OBCibaGrantHandler.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/java/com.wso2.openbanking.accelerator.ciba/OBCibaGrantHandler.java deleted file mode 100644 index f656bc7c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/java/com.wso2.openbanking.accelerator.ciba/OBCibaGrantHandler.java +++ /dev/null @@ -1,69 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.ciba; - -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.oauth.ciba.grant.CibaGrantHandler; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.token.OAuthTokenReqMessageContext; - -import java.util.Arrays; - -/** - * OB specific CIBA grant handler. - */ -public class OBCibaGrantHandler extends CibaGrantHandler { - - private static final ConsentCoreServiceImpl consentCoreService = new ConsentCoreServiceImpl(); - private static Log log = LogFactory.getLog(CibaGrantHandler.class); - - public void setConsentIdScope(OAuthTokenReqMessageContext tokReqMsgCtx, String authReqId) - throws IdentityOAuth2Exception { - String[] scopesArray; - String[] tokenRequestMessageContextArray = tokReqMsgCtx.getScope(); - if (tokenRequestMessageContextArray != null) { - scopesArray = Arrays.copyOf(tokenRequestMessageContextArray, tokenRequestMessageContextArray.length + 1); - } else { - throw new IdentityOAuth2Exception(CIBAConstants.MESSAGE_CONTEXT_EMPTY_ERROR); - } - try { - scopesArray[scopesArray.length - 1] = CIBAConstants.CONSENT_ID_PREFIX + consentCoreService. - getConsentIdByConsentAttributeNameAndValue("auth_req_id", authReqId).get(0); - } catch (ConsentManagementException e) { - throw new IdentityOAuth2Exception(CIBAConstants.SCOPE_ADDING_ERROR, e); - } - tokReqMsgCtx.setScope(scopesArray); - - } - - @Override - public boolean validateGrant(OAuthTokenReqMessageContext tokReqMsgCtx) throws IdentityOAuth2Exception { - if (!super.validateGrant(tokReqMsgCtx)) { - log.error("Successful in validating grant.Validation failed for the token request made by client: " - + tokReqMsgCtx.getOauth2AccessTokenReqDTO().getClientId()); - return false; - } else { - setConsentIdScope(tokReqMsgCtx, super.getAuthReqId(tokReqMsgCtx)); - return true; - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/resources/findbugs-exclude.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/resources/findbugs-exclude.xml deleted file mode 100644 index c4f8e532..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/resources/findbugs-exclude.xml +++ /dev/null @@ -1,21 +0,0 @@ - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/resources/findbugs-include.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/resources/findbugs-include.xml deleted file mode 100644 index 649d044e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,23 +0,0 @@ - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/test/java/com/wso2/openbanking/accelerator/ciba/OBCIBARequestObjectValidationExtensionTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/test/java/com/wso2/openbanking/accelerator/ciba/OBCIBARequestObjectValidationExtensionTest.java deleted file mode 100644 index 74133c1a..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/test/java/com/wso2/openbanking/accelerator/ciba/OBCIBARequestObjectValidationExtensionTest.java +++ /dev/null @@ -1,93 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.ciba; - -import com.nimbusds.jwt.JWTClaimsSet; -import net.minidev.json.JSONObject; -import org.mockito.Mockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.oauth2.RequestObjectException; -import org.wso2.carbon.identity.oauth2.model.OAuth2Parameters; -import org.wso2.carbon.identity.openidconnect.model.RequestObject; - -import static org.mockito.Mockito.mock; -import static org.mockito.Mockito.when; -/** - * Test class for OBCIBARequestObjectValidationExtension. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({JWTClaimsSet.class, OAuth2Parameters.class, RequestObject.class, JSONObject.class}) -public class OBCIBARequestObjectValidationExtensionTest extends PowerMockTestCase { - - private final String dummyString = "dummyString"; - - @Test(expectedExceptions = RequestObjectException.class, description = "Empty intent key") - public void validateRequestObjectInvalidIntentKeyTest() throws Exception { - - OBCIBARequestObjectValidationExtensionMock obcibaRequestObjectValidationExtensionMock = - new OBCIBARequestObjectValidationExtensionMock(); - - JSONObject intent = mock(JSONObject.class); - - RequestObject requestObject = mock(RequestObject.class); - OAuth2Parameters oAuth2Parameters = mock(OAuth2Parameters.class); - JWTClaimsSet claimsSet = Mockito.mock(JWTClaimsSet.class); - - Mockito.when(requestObject.getClaimsSet()).thenReturn(claimsSet); - Mockito.when(claimsSet.getJSONObjectClaim(Mockito.anyString())).thenReturn(intent); - - when(intent.getAsString(dummyString)).thenReturn(dummyString); - - obcibaRequestObjectValidationExtensionMock.validateRequestObject(requestObject, oAuth2Parameters); - - } - - @Test(description = "success scenario") - public void validateRequestObjectValidObjectTest() throws Exception { - - OBCIBARequestObjectValidationExtensionMock obcibaRequestObjectValidationExtensionMock = - new OBCIBARequestObjectValidationExtensionMock(); - - JSONObject intent = mock(JSONObject.class); - - RequestObject requestObject = mock(RequestObject.class); - OAuth2Parameters oAuth2Parameters = mock(OAuth2Parameters.class); - JWTClaimsSet claimsSet = Mockito.mock(JWTClaimsSet.class); - - Mockito.when(requestObject.getClaimsSet()).thenReturn(claimsSet); - Mockito.when(claimsSet.getJSONObjectClaim(Mockito.anyString())).thenReturn(intent); - - when(intent.getAsString("value")).thenReturn(dummyString); - - obcibaRequestObjectValidationExtensionMock.validateRequestObject(requestObject, oAuth2Parameters); - - } -} - -class OBCIBARequestObjectValidationExtensionMock extends OBCIBARequestObjectValidationExtension { - - @Override - boolean validateIAMConstraints(RequestObject requestObject, - OAuth2Parameters oAuth2Parameters) throws RequestObjectException { - return Mockito.anyBoolean(); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/test/java/com/wso2/openbanking/accelerator/ciba/OBCIBASignatureAlgorithmEnforcementValidatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/test/java/com/wso2/openbanking/accelerator/ciba/OBCIBASignatureAlgorithmEnforcementValidatorTest.java deleted file mode 100644 index 8de0c7ce..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/test/java/com/wso2/openbanking/accelerator/ciba/OBCIBASignatureAlgorithmEnforcementValidatorTest.java +++ /dev/null @@ -1,124 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.ciba; - -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.token.DefaultTokenFilter; -import com.wso2.openbanking.accelerator.identity.token.TokenFilter; -import com.wso2.openbanking.accelerator.identity.token.validators.OBIdentityFilterValidator; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.http.HttpStatus; -import org.json.JSONObject; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.springframework.mock.web.MockHttpServletRequest; -import org.springframework.mock.web.MockHttpServletResponse; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.oauth.config.OAuthServerConfiguration; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import javax.servlet.FilterChain; -import javax.servlet.ServletOutputStream; - -import static org.testng.Assert.assertEquals; - -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({IdentityCommonUtil.class, OAuthServerConfiguration.class}) -public class OBCIBASignatureAlgorithmEnforcementValidatorTest extends PowerMockTestCase { - - - - MockHttpServletResponse response; - MockHttpServletRequest request; - FilterChain filterChain; - - @BeforeMethod - public void beforeMethod() { - - request = new MockHttpServletRequest(); - response = new MockHttpServletResponse(); - filterChain = Mockito.spy(FilterChain.class); - - } - - //Enable this test if you are building CIBA components along with this. - @Test(description = "Test when registered algorithm and signed algorithm differ") - public void fapiUnsupportedSignatureAlgorithmValidationTest() throws Exception { - Map configMap = new HashMap<>(); - PowerMockito.mockStatic(IdentityCommonUtil.class); - PowerMockito.mockStatic(OAuthServerConfiguration.class); - - OAuthServerConfiguration oAuthServerConfiguration = Mockito.mock(OAuthServerConfiguration.class); - configMap.put(IdentityCommonConstants.ENABLE_TRANSPORT_CERT_AS_HEADER, true); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - OBCIBASignatureAlgorithmEnforcementValidator validator = - Mockito.spy(OBCIBASignatureAlgorithmEnforcementValidator.class); - - request.setParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION, TestConstants.CLIENT_ASSERTION); - request.addHeader(TestConstants.CERTIFICATE_HEADER, TestConstants.CERTIFICATE_CONTENT); - - Mockito.doReturn("RS256").when(validator) - .getRegisteredSigningAlgorithm("iYpRm64b2vmvmKDhdL6KZD9z6fca"); - Mockito.doReturn("RS256").when(validator) - .getRequestSigningAlgorithm(TestConstants.CLIENT_ASSERTION); - - List validators = new ArrayList<>(); - validators.add(validator); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - Mockito.doReturn(validators).when(filter).getValidators(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("iYpRm64b2vmvmKDhdL6KZD9z6fca")) - .thenReturn(true); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - PowerMockito.when(OAuthServerConfiguration.getInstance()).thenReturn(oAuthServerConfiguration); - Mockito.when(oAuthServerConfiguration.isFapiCiba()).thenReturn(true); - filter.doFilter(request, response, filterChain); - - Map responseMap = getResponse(response.getOutputStream()); - assertEquals(response.getStatus(), HttpStatus.SC_UNAUTHORIZED); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR), "invalid_client"); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION), - "FAPI unsupported signing algorithm RS256 used to sign the JWT"); - } - - public static Map getResponse(ServletOutputStream outputStream) { - - Map response = new HashMap<>(); - JSONObject outputStreamMap = new JSONObject(outputStream); - JSONObject targetStream = new JSONObject(outputStreamMap.get(TestConstants.TARGET_STREAM).toString()); - response.put(IdentityCommonConstants.OAUTH_ERROR, - targetStream.get(IdentityCommonConstants.OAUTH_ERROR).toString()); - response.put(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION, - targetStream.get(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION).toString()); - return response; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/test/java/com/wso2/openbanking/accelerator/ciba/TestConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/test/java/com/wso2/openbanking/accelerator/ciba/TestConstants.java deleted file mode 100644 index adbe8927..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/test/java/com/wso2/openbanking/accelerator/ciba/TestConstants.java +++ /dev/null @@ -1,66 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.ciba; - -public class TestConstants { - - public static final String CLIENT_ASSERTION = "eyJraWQiOiJqeVJVY3l0MWtWQ2xjSXZsVWxjRHVrVlozdFUiLCJhbGciOiJQUzI1" + - "NiJ9.eyJzdWIiOiJpWXBSbTY0YjJ2bXZtS0RoZEw2S1pEOXo2ZmNhIiwiYXVkIjoiaHR0cHM6Ly9sb2NhbGhvc3Q6OTQ0My9vYXV0a" + - "DIvdG9rZW4iLCJpc3MiOiJpWXBSbTY0YjJ2bXZtS0RoZEw2S1pEOXo2ZmNhIiwiZXhwIjoxNjEwNjMxNDEyLCJpYXQiOjE2MTA2MDE" + - "0MTIsImp0aSI6IjE2MTA2MDE0MTI5MDAifQ.tmMTlCL-VABhFTA6QQ6UPvUydKuzynidepAa8oZGEBfVyAsiW5IF01NKYD0ynpXXJC" + - "Q6hcbWK0FEGity67p6DeI9LT-xAnaKwZY7H8rbuxWye2vhanM0jVa1vggsmwWYyOR4k55ety9lP1MkcGZpaK48qoaqsX_X7GCSGXzq" + - "BncTEPYfCpVUQtS4ctwoCl06TFbY2Lfm9E24z1rfmU9xPc7au6LpKRLMMHQ8QXuc-FhnWdgEFv_3tAai2ovVmrqEfwj6Z6Ew5bFeI9" + - "jtCR4TSol47hzDwldx5rH7m2OPUx66yEtGrM7UU62fC-4nxplZ69fjlHN4KQ62PxEaCQs0_A"; - - public static final String CERTIFICATE_HEADER = "x-wso2-mutual-auth-cert"; - public static final String CERTIFICATE_CONTENT = "-----BEGIN CERTIFICATE-----" + - "MIIFODCCBCCgAwIBAgIEWcWGxDANBgkqhkiG9w0BAQsFADBTMQswCQYDVQQGEwJH" + - "QjEUMBIGA1UEChMLT3BlbkJhbmtpbmcxLjAsBgNVBAMTJU9wZW5CYW5raW5nIFBy" + - "ZS1Qcm9kdWN0aW9uIElzc3VpbmcgQ0EwHhcNMTkwNTE2MDg0NDQ2WhcNMjAwNjE2" + - "MDkxNDQ2WjBhMQswCQYDVQQGEwJHQjEUMBIGA1UEChMLT3BlbkJhbmtpbmcxGzAZ" + - "BgNVBAsTEjAwMTU4MDAwMDFIUVFyWkFBWDEfMB0GA1UEAxMWc0Zna2k3Mk9pcXda" + - "TkZPWmc2T2FqaTCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEBANoVwx4E" + - "iWnQs89lj8vKSy/xTbZU2AHS9tFNz7wVa+rkpFyLVPtQW8AthG4hlfrBYMne7/P9" + - "c1Fi/q+n7eomWvJJo44GV44GJhegM6yyRaIcQdpxe9x9G4twWK4cY+VU3TfE6Dbd" + - "DdmAt7ai4KFbbpB33N8RwXoeGZdwxZFNPmfaoZZbz5p9+aSMQf1UyExcdlPXah77" + - "PDZDwAnyy5kYXUPS59S78+p4twqZXyZu9hd+Su5Zod5UObRJ4F5LQzZPS1+KzBje" + - "JM0o8qoRRZTZkLNnmmQw503KXp/LCLrSbFU2ZLGy3bQpKFFc5I6tZiy67ELNzLWo" + - "DzngEbApwhX+jtsCAwEAAaOCAgQwggIAMA4GA1UdDwEB/wQEAwIHgDAgBgNVHSUB" + - "Af8EFjAUBggrBgEFBQcDAQYIKwYBBQUHAwIwgeAGA1UdIASB2DCB1TCB0gYLKwYB" + - "BAGodYEGAWQwgcIwKgYIKwYBBQUHAgEWHmh0dHA6Ly9vYi50cnVzdGlzLmNvbS9w" + - "b2xpY2llczCBkwYIKwYBBQUHAgIwgYYMgYNVc2Ugb2YgdGhpcyBDZXJ0aWZpY2F0" + - "ZSBjb25zdGl0dXRlcyBhY2NlcHRhbmNlIG9mIHRoZSBPcGVuQmFua2luZyBSb290" + - "IENBIENlcnRpZmljYXRpb24gUG9saWNpZXMgYW5kIENlcnRpZmljYXRlIFByYWN0" + - "aWNlIFN0YXRlbWVudDBtBggrBgEFBQcBAQRhMF8wJgYIKwYBBQUHMAGGGmh0dHA6" + - "Ly9vYi50cnVzdGlzLmNvbS9vY3NwMDUGCCsGAQUFBzAChilodHRwOi8vb2IudHJ1" + - "c3Rpcy5jb20vb2JfcHBfaXNzdWluZ2NhLmNydDA6BgNVHR8EMzAxMC+gLaArhilo" + - "dHRwOi8vb2IudHJ1c3Rpcy5jb20vb2JfcHBfaXNzdWluZ2NhLmNybDAfBgNVHSME" + - "GDAWgBRQc5HGIXLTd/T+ABIGgVx5eW4/UDAdBgNVHQ4EFgQU5eqvEZ6ZdQS5bq/X" + - "dzP5XY/fUXUwDQYJKoZIhvcNAQELBQADggEBAIg8bd/bIh241ewS79lXU058VjCu" + - "JC+4QtcI2XiGV3dBpg10V6Kb6E/h8Gru04uVZW1JK52ivVb5NYs6r8txRsTBIaA8" + - "Cr03LJqEftclL9NbkPZnpEkUfqCBfujNQF8XWaQgXIIA+io1UzV1TG3K9XCa/w2S" + - "sTANKfF8qK5kRsy6z9OGPUE+Oi3DUt+E9p5LCq6n5Bkp9YRGmyYRPs8JMkJmq3sf" + - "wtXOy27LE4exJRuZsF1CA78ObaRytuE3DJcnIRdhOcjWieS/MxZD7bzuuAPu5ySX" + - "i2/qxT3AlWtHtxrz0mKSC3rlgYAHCzCAHoASWKpf5tnB3TodPVZ6DYOu7oI=" + - "-----END CERTIFICATE-----"; - - public static final String TARGET_STREAM = "targetStream"; - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/test/resources/testng.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/test/resources/testng.xml deleted file mode 100644 index 5b2f8ba3..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.ciba/src/test/resources/testng.xml +++ /dev/null @@ -1,29 +0,0 @@ - - - - - - - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/pom.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/pom.xml deleted file mode 100644 index 870e3fdd..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/pom.xml +++ /dev/null @@ -1,306 +0,0 @@ - - - - - 4.0.0 - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../pom.xml - - - com.wso2.openbanking.accelerator.common - bundle - WSO2 Open Banking - Common component - - - org.wso2.orbit.org.bouncycastle - bcpkix-jdk18on - - - org.wso2.orbit.org.bouncycastle - bcprov-jdk18on - - - org.apache.ws.commons.axiom.wso2 - axiom - - - commons-logging - commons-logging - - - org.wso2.securevault - org.wso2.securevault - - - org.wso2.carbon.identity.inbound.auth.oauth2 - org.wso2.carbon.identity.oauth - - - org.wso2.orbit.org.bouncycastle - bcprov-jdk15on - - - - - org.wso2.carbon.identity.framework - org.wso2.carbon.identity.application.mgt - - - org.wso2.orbit.org.bouncycastle - bcprov-jdk15on - - - - - org.hibernate - hibernate-validator - - - commons-beanutils - commons-beanutils - - - - org.wso2.eclipse.osgi - org.eclipse.osgi.services - - - org.eclipse.osgi - org.eclipse.osgi - - - org.wso2.carbon - org.wso2.carbon.core - - - org.wso2.orbit.com.hazelcast - hazelcast - - - org.wso2.orbit.org.bouncycastle - bcprov-jdk15on - - - - - org.testng - testng - test - - - org.mockito - mockito-all - test - - - org.powermock - powermock-module-testng - - - org.powermock - powermock-api-mockito - - - io.jsonwebtoken - jjwt - - - org.wso2.orbit.com.nimbusds - nimbus-jose-jwt - - - net.minidev - json-smart - - - - - - com.github.spotbugs - spotbugs-maven-plugin - 4.2.3 - - Max - Low - false - true - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - org.apache.maven.plugins - maven-compiler-plugin - - - org.jacoco - jacoco-maven-plugin - - - - **/*Constants.class - **/*Component.class - **/*DataHolder.class - **/*Enum.class - **/*Exception.class - **/*Discoverer.class - **/*AudienceValidator.class - **/*OpenBankingBaseCache.class - **/*OpenBankingBaseCacheKey.class - **/*OpenBankingIdentityBaseCache.class - **/*JWKSetCache.class - **/*JWKSetCacheKey.class - **/*ApplicationIdentityService.class - **/*JWKRetriever.class - **/*OpenBankingErrorCodes.class - **/*IdentityConstants.class - **/*ServerIdentityRetriever.class - - **/DatabaseUtil.class - **/JDBCPersistenceManager.class - **/CertValidationErrors.class - **/JDBCRetentionDataPersistenceManager.class - **/*Type*/** - - - - - default-prepare-agent - - prepare-agent - - - - default-prepare-agent-integration - - prepare-agent-integration - - - - default-report - - report - - - - default-report-integration - - report-integration - - - - default-check - - check - - - - - BUNDLE - - - INSTRUCTION - COVEREDRATIO - 0.77 - - - - - - - - - - org.apache.maven.plugins - maven-surefire-plugin - - - src/test/resources/testng.xml - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-exclude.xml - - - - analyze-compile - compile - - check - - - - - - org.apache.felix - maven-bundle-plugin - true - - - - ${project.artifactId} - - - com.wso2.openbanking.accelerator.common.internal - - - org.osgi.framework;version="${osgi.framework.imp.pkg.version.range}", - org.osgi.service.component;version="${osgi.service.component.imp.pkg.version.range}" - - - !com.wso2.openbanking.accelerator.common.internal, - com.wso2.openbanking.accelerator.common.*;version="${project.version}", - - * - <_dsannotations>* - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/caching/OpenBankingBaseCache.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/caching/OpenBankingBaseCache.java deleted file mode 100644 index 1c41b245..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/caching/OpenBankingBaseCache.java +++ /dev/null @@ -1,212 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.caching; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.concurrent.TimeUnit; - -import javax.cache.Cache; -import javax.cache.CacheBuilder; -import javax.cache.CacheConfiguration; -import javax.cache.CacheManager; -import javax.cache.Caching; - -/** - * Abstract cache manager for Open Banking. - * - * @param Extended Cache Key - * @param Cache Value - */ -public abstract class OpenBankingBaseCache { - - private static final String BASE_CACHE_KEY = "OB_BASE_CACHE"; - private final String cacheName; - - private static final Log log = LogFactory.getLog(OpenBankingBaseCache.class); - - /** - * On Demand Retriever for caching. - */ - public interface OnDemandRetriever { - - Object get() throws OpenBankingException; - } - - /** - * Initialize With unique cache name. - * - * @param cacheName unique cache name. - */ - public OpenBankingBaseCache(String cacheName) { - - this.cacheName = cacheName; - - if (log.isDebugEnabled()) { - log.debug(String.format("Base Cache initialized for %s", cacheName.replaceAll("[\r\n]", ""))); - } - } - - /** - * Get from cache or invoke ondemand retriever and store. - * - * @param key cache key. - * @param onDemandRetriever on demand retriever. - * @return cached object. - * @throws OpenBankingException if an error occurs while retrieving the object - */ - public V getFromCacheOrRetrieve(K key, OnDemandRetriever onDemandRetriever) throws OpenBankingException { - - Cache cache = getBaseCache(); - - if (cache.containsKey(key)) { - - if (log.isDebugEnabled()) { - log.debug(String.format("Found cache entry `%s` in cache %s", - key.toString().replaceAll("[\r\n]", ""), cacheName.replaceAll("[\r\n]", ""))); - } - return cache.get(key); - } else { - - if (log.isDebugEnabled()) { - log.debug(String.format("Cache Entry `%s` not available in cache %s", - key.toString().replaceAll("[\r\n]", ""), cacheName.replaceAll("[\r\n]", ""))); - } - - V value = (V) onDemandRetriever.get(); - - if (log.isDebugEnabled()) { - log.debug(String.format("On demand retrieved `%s` for %s", - key.toString().replaceAll("[\r\n]", ""), cacheName.replaceAll("[\r\n]", ""))); - } - - removeFromCache(key); - addToCache(key, value); - return value; - } - - } - - /** - * Get from cache. - * - * @param key cache key. - * @return cached object. - */ - public V getFromCache(K key) { - - Cache cache = getBaseCache(); - - if (cache.containsKey(key)) { - - if (log.isDebugEnabled()) { - log.debug(String.format("Found cache entry `%s` in cache %s", - key.toString().replaceAll("[\r\n]", ""), cacheName.replaceAll("[\r\n]", ""))); - } - return cache.get(key); - } else { - - return null; - } - - } - - /** - * Add Object to cache. - * - * @param key cache key. - * @param value cache value. - */ - public void addToCache(K key, V value) { - - if (log.isDebugEnabled()) { - log.debug(String.format("`%s` added into cache %s", key.toString().replaceAll("[\r\n]", ""), - cacheName.replaceAll("[\r\n]", ""))); - } - - Cache cache = getBaseCache(); - cache.put(key, value); - } - - /** - * Remove Object from Cache. - * - * @param key cache key. - */ - public void removeFromCache(K key) { - - if (log.isDebugEnabled()) { - log.debug(String.format("`%s` removed from cache %s", key.toString().replaceAll("[\r\n]", ""), - cacheName.replaceAll("[\r\n]", ""))); - } - - Cache cache = getBaseCache(); - cache.remove(key); - } - - /** - * Get Cache for instance. - * - * @return - */ - private Cache getBaseCache() { - - CacheManager cacheManager = Caching.getCacheManager(BASE_CACHE_KEY); - - Iterable> availableCaches = cacheManager.getCaches(); - for (Cache cache : availableCaches) { - if (cache.getName().equalsIgnoreCase( - cache.getName().startsWith("$__local__$.") ? "$__local__$." + cacheName : cacheName)) { - return cacheManager.getCache(cacheName); - } - } - - CacheConfiguration.Duration accessExpiry = new CacheConfiguration.Duration(TimeUnit.MINUTES, - getCacheAccessExpiryMinutes()); - - CacheConfiguration.Duration modifiedExpiry = new CacheConfiguration.Duration(TimeUnit.MINUTES, - getCacheModifiedExpiryMinutes()); - - // Build Cache on OB base cache. - CacheBuilder cacheBuilder = cacheManager.createCacheBuilder(cacheName); - - return cacheBuilder.setExpiry(CacheConfiguration.ExpiryType.ACCESSED, accessExpiry) - .setExpiry(CacheConfiguration.ExpiryType.MODIFIED, modifiedExpiry) - .build(); - - } - - - /** - * Get Cache expiry time upon access in minutes. - * - * @return integer denoting number of minutes. - */ - public abstract int getCacheAccessExpiryMinutes(); - - /** - * Get Cache expiry time upon modification in minutes. - * - * @return integer denoting number of minutes. - */ - public abstract int getCacheModifiedExpiryMinutes(); - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/caching/OpenBankingBaseCacheKey.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/caching/OpenBankingBaseCacheKey.java deleted file mode 100644 index b6ca634d..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/caching/OpenBankingBaseCacheKey.java +++ /dev/null @@ -1,38 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.caching; - -/** - * Abstract class for Open Banking Cache Key. - */ -public class OpenBankingBaseCacheKey { - - public OpenBankingBaseCacheKey() { - - } - - public OpenBankingBaseCacheKey(String cacheKey) { - - } - - public static OpenBankingBaseCacheKey of(String cacheKey) { - return new OpenBankingBaseCacheKey(cacheKey); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/config/OpenBankingConfigParser.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/config/OpenBankingConfigParser.java deleted file mode 100644 index 32486797..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/config/OpenBankingConfigParser.java +++ /dev/null @@ -1,1483 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.config; - -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingRuntimeException; -import com.wso2.openbanking.accelerator.common.util.CarbonUtils; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import org.apache.axiom.om.OMElement; -import org.apache.axiom.om.OMException; -import org.apache.axiom.om.impl.builder.StAXOMBuilder; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.securevault.SecretResolver; -import org.wso2.securevault.SecretResolverFactory; - -import java.io.File; -import java.io.FileInputStream; -import java.io.FileNotFoundException; -import java.io.IOException; -import java.io.InputStream; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Collections; -import java.util.HashMap; -import java.util.Iterator; -import java.util.LinkedHashMap; -import java.util.List; -import java.util.Map; -import java.util.Stack; -import java.util.stream.Collectors; - -import javax.xml.namespace.QName; -import javax.xml.stream.XMLStreamException; - -import static java.util.Map.Entry.comparingByKey; - -/** - * Config parser for open-banking.xml. - */ -public class OpenBankingConfigParser { - - // To enable attempted thread-safety using double-check locking - private static final Object lock = new Object(); - private static final Log log = LogFactory.getLog(OpenBankingConfigParser.class); - private static final Map configuration = new HashMap<>(); - private static final Map> obExecutors = new HashMap<>(); - private static final Map> dataPublishingStreams = new HashMap<>(); - private static final Map> dataPublishingValidationMap = new HashMap<>(); - private static final Map> dcrRegistrationConfigs = new HashMap<>(); - private static final Map> authorizeSteps = new HashMap<>(); - private static final Map> allowedScopes = new HashMap<>(); - private static final Map> allowedAPIs = new HashMap<>(); - private static final Map revocationValidators = new HashMap<>(); - private static final List serviceActivatorSubscribers = new ArrayList<>(); - private static final Map> keyManagerAdditionalProperties - = new HashMap<>(); - private static Map obEventExecutors = new HashMap<>(); - private static OpenBankingConfigParser parser; - private static String configFilePath; - private static SecretResolver secretResolver; - private OMElement rootElement; - - private Map authWorkerConfig = new HashMap<>(); - - /** - * Private Constructor of config parser. - */ - private OpenBankingConfigParser() { - - buildConfiguration(); - } - - /** - * Singleton getInstance method to create only one object. - * - * @return OpenBankingConfigParser object - */ - public static OpenBankingConfigParser getInstance() { - - if (parser == null) { - synchronized (lock) { - if (parser == null) { - parser = new OpenBankingConfigParser(); - } - } - } - return parser; - } - - /** - * Method to get an instance of ConfigParser when custom file path is provided. - * - * This method is deprecated as it allows custom absolute file paths which could result in - * path traversal attacks. Do not use this method unless the custom path is trusted. - * - * @param filePath Custom file path - * @return OpenBankingConfigParser object - * @Deprecated use OpenBankingConfigParser.getInstance() - */ - @Deprecated - public static OpenBankingConfigParser getInstance(String filePath) { - - configFilePath = filePath; - return getInstance(); - } - - /** - * Method to obtain map of configs. - * - * @return Config map - */ - public Map getConfiguration() { - - return configuration; - } - - /** - * Method to read the configuration (in a recursive manner) as a model and put them in the configuration map. - */ - @SuppressFBWarnings("PATH_TRAVERSAL_IN") - // Suppressed content - new FileInputStream(configFilePath) - // Suppression reason - False Positive : Method for passing configFilePath is deprecated and is used for testing - // purposes only. Therefore, it can be assumed that configFilePath is a trusted filepath - // Suppressed warning count - 1 - private void buildConfiguration() { - - InputStream inStream = null; - StAXOMBuilder builder; - String warningMessage = ""; - try { - if (configFilePath != null) { - File openBankingConfigXml = new File(configFilePath); - if (openBankingConfigXml.exists()) { - inStream = new FileInputStream(openBankingConfigXml); - } - } else { - File openBankingConfigXml = new File(CarbonUtils.getCarbonConfigDirPath(), - OpenBankingConstants.OB_CONFIG_FILE); - if (openBankingConfigXml.exists()) { - inStream = new FileInputStream(openBankingConfigXml); - } - } - if (inStream == null) { - String message = - "open-banking configuration not found at: " + configFilePath + " . Cause - " + warningMessage; - if (log.isDebugEnabled()) { - log.debug(message.replaceAll("[\r\n]", "")); - } - throw new FileNotFoundException(message); - } - builder = new StAXOMBuilder(inStream); - builder.setDoDebug(false); - rootElement = builder.getDocumentElement(); - Stack nameStack = new Stack<>(); - secretResolver = SecretResolverFactory.create(rootElement, true); - readChildElements(rootElement, nameStack); - buildOBExecutors(); - buildDataPublishingStreams(); - buildDCRParameters(); - buildConsentAuthSteps(); - buildAllowedScopes(); - buildAllowedSubscriptions(); - buildServiceActivatorSubscribers(); - buildKeyManagerProperties(); - buildOBEventExecutors(); - buildWorkers(); - } catch (IOException | XMLStreamException | OMException e) { - throw new OpenBankingRuntimeException("Error occurred while building configuration from open-banking.xml", - e); - } finally { - try { - if (inStream != null) { - inStream.close(); - } - } catch (IOException e) { - log.error("Error closing the input stream for open-banking.xml", e); - } - } - } - - private void buildOBExecutors() { - - OMElement gatewayElement = rootElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, - OpenBankingConstants.GATEWAY_CONFIG_TAG)); - - if (gatewayElement != null) { - - OMElement openBankingGatewayExecutors = gatewayElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, - OpenBankingConstants.GATEWAY_EXECUTOR_CONFIG_TAG)); - - if (openBankingGatewayExecutors != null) { - //obtaining each consent type element under OpenBankingGatewayExecutors tag - Iterator consentTypeElement = openBankingGatewayExecutors.getChildElements(); - while (consentTypeElement.hasNext()) { - OMElement consentType = (OMElement) consentTypeElement.next(); - String consentTypeName = consentType.getLocalName(); - Map executors = new HashMap<>(); - //obtaining each Executor element under each consent type - Iterator obExecutor = consentType.getChildrenWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, OpenBankingConstants.EXECUTOR_CONFIG_TAG)); - if (obExecutor != null) { - while (obExecutor.hasNext()) { - OMElement executorElement = obExecutor.next(); - //Retrieve class name and priority from executor config - String obExecutorClass = executorElement.getAttributeValue(new QName("class")); - String obExecutorPriority = executorElement.getAttributeValue(new QName("priority")); - - if (StringUtils.isEmpty(obExecutorClass)) { - //Throwing exceptions since we cannot proceed without invalid executor names - throw new OpenBankingRuntimeException("Executor class is not defined " + - "correctly in open-banking.xml"); - } - int priority = Integer.MAX_VALUE; - if (!StringUtils.isEmpty(obExecutorPriority)) { - priority = Integer.parseInt(obExecutorPriority); - } - executors.put(priority, obExecutorClass); - } - } - //Ordering the executors based on the priority number - LinkedHashMap priorityMap = executors.entrySet() - .stream() - .sorted(comparingByKey()) - .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue, (e1, e2) -> e2, - LinkedHashMap::new)); - obExecutors.put(consentTypeName, priorityMap); - } - } - } - } - - protected void buildKeyManagerProperties() { - - OMElement keyManagerElement = rootElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, - OpenBankingConstants.KEY_MANAGER_CONFIG_TAG)); - - if (keyManagerElement != null) { - OMElement keyManagerProperties = keyManagerElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, - OpenBankingConstants.KEY_MANAGER_ADDITIONAL_PROPERTIES_CONFIG_TAG)); - - if (keyManagerProperties != null) { - Iterator properties = keyManagerProperties.getChildrenWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, OpenBankingConstants.PROPERTY_CONFIG_TAG)); - if (properties != null) { - while (properties.hasNext()) { - OMElement propertyElement = properties.next(); - - //Retrieve attributes from key manager config - Map property = new HashMap<>(); - property.put("priority", propertyElement.getAttributeValue(new QName("priority"))); - property.put("label", propertyElement.getAttributeValue(new QName("label"))); - property.put("type", propertyElement.getAttributeValue(new QName("type"))); - property.put("tooltip", propertyElement.getAttributeValue(new QName("tooltip"))); - property.put("default", propertyElement.getAttributeValue(new QName("default"))); - property.put("required", propertyElement.getAttributeValue(new QName("required"))); - property.put("mask", propertyElement.getAttributeValue(new QName("mask"))); - property.put("multiple", propertyElement.getAttributeValue(new QName("multiple"))); - property.put("values", propertyElement.getAttributeValue(new QName("values"))); - String propertyName = propertyElement.getAttributeValue(new QName("name")); - - if (StringUtils.isBlank(propertyName)) { - //Throwing exceptions since we cannot proceed without property names - throw new OpenBankingRuntimeException("Additional property name is not defined " + - "correctly in open-banking.xml"); - } - - keyManagerAdditionalProperties.put(propertyName, property); - } - } - } - } - } - - protected void buildDataPublishingStreams() { - - OMElement dataPublishingElement = rootElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, - OpenBankingConstants.DATA_PUBLISHING_CONFIG_TAG)); - - if (dataPublishingElement != null) { - OMElement thriftElement = dataPublishingElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, - OpenBankingConstants.THRIFT_CONFIG_TAG)); - - if (thriftElement != null) { - OMElement streams = thriftElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, - OpenBankingConstants.STREAMS_CONFIG_TAG)); - - if (streams != null) { - Iterator dataStreamElement = streams.getChildElements(); - while (dataStreamElement.hasNext()) { - OMElement dataStream = (OMElement) dataStreamElement.next(); - String dataStreamName = dataStream.getLocalName(); - Map attributes = new HashMap<>(); - //obtaining attributes under each stream - Iterator attribute = dataStream.getChildrenWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, - OpenBankingConstants.ATTRIBUTE_CONFIG_TAG)); - if (attribute != null) { - while (attribute.hasNext()) { - OMElement attributeElement = attribute.next(); - //Retrieve attribute name and priority from config - String attributeName = attributeElement.getAttributeValue(new QName("name")); - String attributePriority = attributeElement.getAttributeValue(new QName("priority")); - String isRequired = attributeElement.getAttributeValue(new QName("required")); - String type = attributeElement.getAttributeValue(new QName("type")); - - if (StringUtils.isEmpty(attributeName)) { - //Throwing exceptions since we cannot proceed without valid attribute names - throw new OpenBankingRuntimeException( - "Data publishing attribute name is not defined " + - "correctly in open-banking.xml"); - } - int priority = Integer.MAX_VALUE; - if (!StringUtils.isEmpty(attributePriority)) { - priority = Integer.parseInt(attributePriority); - } - boolean required = false; - if (!StringUtils.isEmpty(isRequired)) { - required = Boolean.parseBoolean(isRequired); - } - - String attributeType = "string"; - if (!StringUtils.isEmpty(type)) { - attributeType = type; - } - - Map metadata = new HashMap<>(); - metadata.put(OpenBankingConstants.REQUIRED, required); - metadata.put(OpenBankingConstants.ATTRIBUTE_TYPE, attributeType); - - attributes.put(priority, attributeName); - String attributeKey = dataStreamName + "_" + attributeName; - dataPublishingValidationMap.put(attributeKey, metadata); - } - } - //Ordering the attributes based on the priority number - LinkedHashMap priorityMap = attributes.entrySet() - .stream() - .sorted(comparingByKey()) - .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue, (e1, e2) -> e2, - LinkedHashMap::new)); - dataPublishingStreams.put(dataStreamName, priorityMap); - } - } - } - } - } - - private void buildDCRParameters() { - - OMElement dcrElement = rootElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, OpenBankingConstants.DCR_CONFIG_TAG)); - - if (dcrElement != null) { - OMElement registrationElement = dcrElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, OpenBankingConstants.DCR_REGISTRATION_CONFIG_TAG)); - - if (registrationElement != null) { - //obtaining each parameter type element under RegistrationRequestPrams tag - Iterator parameterTypeElement = registrationElement.getChildElements(); - while (parameterTypeElement.hasNext()) { - OMElement parameterType = (OMElement) parameterTypeElement.next(); - String parameterTypeName = parameterType.getLocalName(); - Map parameterValues = new HashMap<>(); - //obtaining each element under each parameter type - Iterator childValues = parameterType.getChildElements(); - while (childValues.hasNext()) { - OMElement child = (OMElement) childValues.next(); - if (OpenBankingConstants.DCR_REGISTRATION_PARAM_ALLOWED_VALUE_TAG - .equalsIgnoreCase(child.getLocalName())) { - - OMElement allowedValuesElement = parameterType.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, - OpenBankingConstants.DCR_REGISTRATION_PARAM_ALLOWED_VALUE_TAG)); - - List values = new ArrayList<>(); - if (allowedValuesElement != null) { - Iterator allowedValues = allowedValuesElement.getChildElements(); - while (allowedValues.hasNext()) { - OMElement value = (OMElement) allowedValues.next(); - values.add(value.getText()); - } - parameterValues.put(child.getLocalName(), values); - } - } else { - parameterValues.put(child.getLocalName(), child.getText()); - } - } - dcrRegistrationConfigs.put(parameterTypeName, parameterValues); - } - } - } - - } - - private void buildConsentAuthSteps() { - - OMElement consentElement = rootElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, - OpenBankingConstants.CONSENT_CONFIG_TAG)); - - if (consentElement != null) { - OMElement consentAuthorizeSteps = consentElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, - OpenBankingConstants.AUTHORIZE_STEPS_CONFIG_TAG)); - - if (consentAuthorizeSteps != null) { - //obtaining each step type element under AuthorizeSteps tag - Iterator stepTypeElement = consentAuthorizeSteps.getChildElements(); - while (stepTypeElement.hasNext()) { - OMElement stepType = (OMElement) stepTypeElement.next(); - String consentTypeName = stepType.getLocalName(); - Map executors = new HashMap<>(); - //obtaining each step under each consent type - Iterator obExecutor = stepType.getChildrenWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, OpenBankingConstants.STEP_CONFIG_TAG)); - if (obExecutor != null) { - while (obExecutor.hasNext()) { - OMElement executorElement = obExecutor.next(); - //Retrieve class name and priority from executor config - String obExecutorClass = executorElement.getAttributeValue(new QName("class")); - String obExecutorPriority = executorElement.getAttributeValue(new QName("priority")); - - if (StringUtils.isEmpty(obExecutorClass)) { - //Throwing exceptions since we cannot proceed without invalid executor names - throw new OpenBankingRuntimeException("Executor class is not defined " + - "correctly in open-banking.xml"); - } - int priority = Integer.MAX_VALUE; - if (!StringUtils.isEmpty(obExecutorPriority)) { - priority = Integer.parseInt(obExecutorPriority); - } - executors.put(priority, obExecutorClass); - } - } - //Ordering the executors based on the priority number - LinkedHashMap priorityMap = executors.entrySet() - .stream() - .sorted(comparingByKey()) - .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue, (e1, e2) -> e2, - LinkedHashMap::new)); - authorizeSteps.put(consentTypeName, priorityMap); - } - } - } - } - - /** - * Method to read text configs from xml when root element is given. - * - * @param serverConfig XML root element object - * @param nameStack stack of config names - */ - private void readChildElements(OMElement serverConfig, Stack nameStack) { - - for (Iterator childElements = serverConfig.getChildElements(); childElements.hasNext(); ) { - OMElement element = (OMElement) childElements.next(); - nameStack.push(element.getLocalName()); - if (elementHasText(element)) { - String key = getKey(nameStack); - Object currentObject = configuration.get(key); - String value = replaceSystemProperty(element.getText()); - if (secretResolver != null && secretResolver.isInitialized() && - secretResolver.isTokenProtected(key)) { - value = secretResolver.resolve(key); - } - if (currentObject == null) { - configuration.put(key, value); - } else if (currentObject instanceof ArrayList) { - ArrayList list = (ArrayList) currentObject; - if (!list.contains(value)) { - list.add(value); - configuration.put(key, list); - } - } else { - if (!value.equals(currentObject)) { - ArrayList arrayList = new ArrayList<>(2); - arrayList.add(currentObject); - arrayList.add(value); - configuration.put(key, arrayList); - } - } - } else if (OpenBankingConstants.REVOCATION_VALIDATORS_CONFIG_TAG.equalsIgnoreCase(element.getLocalName())) { - Iterator environmentIterator = element - .getChildrenWithLocalName(OpenBankingConstants.REVOCATION_VALIDATOR_CONFIG_TAG); - - while (environmentIterator.hasNext()) { - OMElement environmentElem = (OMElement) environmentIterator.next(); - String revocationType = environmentElem.getAttributeValue(new QName("type")); - Integer priority; - try { - priority = Integer.parseInt(environmentElem.getAttributeValue(new QName("priority"))); - } catch (NumberFormatException e) { - log.warn("Consent retrieval RevocationValidator " + revocationType.replaceAll("[\r\n]", "") - + " priority invalid. Hence skipped"); - continue; - } - revocationValidators.put(priority, revocationType); - } - } - readChildElements(element, nameStack); - nameStack.pop(); - } - } - - private void buildAllowedScopes() { - OMElement gatewayElement = rootElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, OpenBankingConstants.GATEWAY_CONFIG_TAG)); - - if (gatewayElement != null) { - OMElement tppManagementElement = gatewayElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, OpenBankingConstants.TPP_MANAGEMENT_CONFIG_TAG)); - - if (tppManagementElement != null) { - OMElement allowedScopesElement = tppManagementElement.getFirstChildWithName(new QName( - OpenBankingConstants.OB_CONFIG_QNAME, OpenBankingConstants.ALLOWED_SCOPES_CONFIG_TAG)); - - //obtaining each scope under allowed scopes - Iterator environmentIterator = - allowedScopesElement.getChildrenWithLocalName(OpenBankingConstants.SCOPE_CONFIG_TAG); - - while (environmentIterator.hasNext()) { - OMElement scopeElem = (OMElement) environmentIterator.next(); - String scopeName = scopeElem.getAttributeValue(new QName("name")); - String rolesStr = scopeElem.getAttributeValue(new QName("roles")); - if (StringUtils.isNotEmpty(rolesStr)) { - List rolesList = Arrays.stream(rolesStr.split(",")) - .map(String::trim) - .collect(Collectors.toList()); - allowedScopes.put(scopeName, rolesList); - } - } - } - } - } - - private void buildAllowedSubscriptions() { - - OMElement dcrElement = rootElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, OpenBankingConstants.DCR_CONFIG_TAG)); - - if (dcrElement != null) { - OMElement regulatoryAPIs = dcrElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, OpenBankingConstants.REGULATORY_API_NAMES)); - - if (regulatoryAPIs != null) { - - //obtaining each regulatory API under allowed regulatory APIs - Iterator environmentIterator = - regulatoryAPIs.getChildrenWithLocalName(OpenBankingConstants.REGULATORY_API); - - while (environmentIterator.hasNext()) { - OMElement regulatoryAPIElem = (OMElement) environmentIterator.next(); - String regulatoryAPIName = regulatoryAPIElem.getAttributeValue(new QName( - OpenBankingConstants.API_NAME)); - String rolesStr = regulatoryAPIElem.getAttributeValue(new QName( - OpenBankingConstants.API_ROLE)); - if (StringUtils.isNotEmpty(rolesStr)) { - List rolesList = Arrays.stream(rolesStr.split(",")) - .map(String::trim) - .collect(Collectors.toList()); - allowedAPIs.put(regulatoryAPIName, rolesList); - } else { - allowedAPIs.put(regulatoryAPIName, Collections.emptyList()); - } - } - } - } - } - - private void buildOBEventExecutors() { - - OMElement eventElement = rootElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, - OpenBankingConstants.EVENT_CONFIG_TAG)); - - if (eventElement != null) { - - OMElement openBankingEventExecutors = eventElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, - OpenBankingConstants.EVENT_EXECUTOR_CONFIG_TAG)); - - if (openBankingEventExecutors != null) { - //obtaining each executor element under EventExecutors tag - //Ordering the executors based on the priority number - Iterator eventExecutor = openBankingEventExecutors.getChildrenWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, OpenBankingConstants.EXECUTOR_CONFIG_TAG)); - if (eventExecutor != null) { - while (eventExecutor.hasNext()) { - OMElement executorElement = eventExecutor.next(); - //Retrieve class name and priority from executor config - String obExecutorClass = executorElement.getAttributeValue(new QName("class")); - String obExecutorPriority = executorElement.getAttributeValue(new QName("priority")); - - if (StringUtils.isEmpty(obExecutorClass)) { - //Throwing exceptions since we cannot proceed without invalid executor names - throw new OpenBankingRuntimeException("Event Executor class is not defined " + - "correctly in open-banking.xml"); - } - int priority = Integer.MAX_VALUE; - if (!StringUtils.isEmpty(obExecutorPriority)) { - priority = Integer.parseInt(obExecutorPriority); - } - obEventExecutors.put(priority, obExecutorClass); - } - } - //Ordering the executors based on the priority number - obEventExecutors = obEventExecutors.entrySet() - .stream() - .sorted(comparingByKey()) - .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue, (e1, e2) -> e2, - LinkedHashMap::new)); - } - } - } - - /** - * Method to build configurations for Authentication Worker Extension point. - */ - private void buildWorkers() { - - OMElement workersOMEList = rootElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, OpenBankingConstants.AUTHENTICATION_WORKER_LIST_TAG)); - - if (workersOMEList != null) { - Iterator workerConfigs = workersOMEList.getChildrenWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, OpenBankingConstants.AUTHENTICATION_WORKER_TAG)); - if (workerConfigs != null) { - while (workerConfigs.hasNext()) { - OMElement executorElement = workerConfigs.next(); - //Retrieve class name and implementation from executor config - String workerClass = executorElement.getAttributeValue(new QName("class")); - String workerName = executorElement.getAttributeValue(new QName("name")); - - if (StringUtils.isEmpty(workerClass) || StringUtils.isEmpty(workerName)) { - //Throwing exceptions since we cannot proceed without invalid worker names - throw new OpenBankingRuntimeException("Authentication worker class is not defined " + - "correctly in open-banking.xml"); - } - authWorkerConfig.put(workerName, workerClass); - } - } - } - } - - /** - * Method to obtain config key from stack. - * - * @param nameStack Stack of strings with names. - * @return key as a String - */ - private String getKey(Stack nameStack) { - - StringBuilder key = new StringBuilder(); - for (int index = 0; index < nameStack.size(); index++) { - String name = nameStack.elementAt(index); - key.append(name).append("."); - } - key.deleteCharAt(key.lastIndexOf(".")); - return key.toString(); - } - - /** - * Method to replace system properties in configs. - * - * @param text String that may require modification - * @return modified string - */ - private String replaceSystemProperty(String text) { - - int indexOfStartingChars = -1; - int indexOfClosingBrace; - - // The following condition deals with properties. - // Properties are specified as ${system.property}, - // and are assumed to be System properties - StringBuilder textBuilder = new StringBuilder(text); - while (indexOfStartingChars < textBuilder.indexOf("${") - && (indexOfStartingChars = textBuilder.indexOf("${")) != -1 - && (indexOfClosingBrace = textBuilder.indexOf("}")) != -1) { // Is a property used? - String sysProp = textBuilder.substring(indexOfStartingChars + 2, indexOfClosingBrace); - String propValue = System.getProperty(sysProp); - if (propValue != null) { - textBuilder = new StringBuilder(textBuilder.substring(0, indexOfStartingChars) + propValue - + textBuilder.substring(indexOfClosingBrace + 1)); - } - if (sysProp.equals(OpenBankingConstants.CARBON_HOME) && - System.getProperty(OpenBankingConstants.CARBON_HOME).equals(".")) { - textBuilder.insert(0, new File(".").getAbsolutePath() + File.separator); - } - } - return textBuilder.toString(); - } - - /** - * Method to check whether config element has text value. - * - * @param element root element as a object - * @return availability of text in the config - */ - private boolean elementHasText(OMElement element) { - - String text = element.getText(); - return text != null && text.trim().length() != 0; - } - - public Map> getOpenBankingExecutors() { - - return obExecutors; - } - - public Map getOpenBankingEventExecutors() { - - return obEventExecutors; - } - - public Map> getDataPublishingStreams() { - - return dataPublishingStreams; - } - - public Map> getDataPublishingValidationMap() { - - return dataPublishingValidationMap; - } - - public Map> getConsentAuthorizeSteps() { - - return authorizeSteps; - } - - public Map> getKeyManagerAdditionalProperties() { - - return keyManagerAdditionalProperties; - } - - /** - * Returns the element with the provided key. - * - * @param key local part name - * @return Corresponding value for key - */ - public Object getConfigElementFromKey(String key) { - - return configuration.get(key); - } - - public String getDataSourceName() { - - return getConfigElementFromKey(OpenBankingConstants.JDBC_PERSISTENCE_CONFIG) == null ? "" : - ((String) getConfigElementFromKey(OpenBankingConstants.JDBC_PERSISTENCE_CONFIG)).trim(); - } - - /** - * Returns the database connection verification timeout in seconds configured in open-banking.xml. - * - * @return 1 if nothing is configured - */ - public int getConnectionVerificationTimeout() { - - return getConfigElementFromKey(OpenBankingConstants.DB_CONNECTION_VERIFICATION_TIMEOUT) == null ? 1 : - Integer.parseInt(getConfigElementFromKey( - OpenBankingConstants.DB_CONNECTION_VERIFICATION_TIMEOUT).toString().trim()); - } - - /** - * Returns the retention datasource name configured in open-banking.xml. - * @return retention datasource name or empty string if nothing is configured - */ - public String getRetentionDataSourceName() { - - return getConfigElementFromKey(OpenBankingConstants.JDBC_RETENTION_DATA_PERSISTENCE_CONFIG) == null ? "" : - ((String) getConfigElementFromKey(OpenBankingConstants.JDBC_RETENTION_DATA_PERSISTENCE_CONFIG)).trim(); - } - - /** - * Returns the retention database connection verification timeout in seconds configured in open-banking.xml. - * - * @return 1 if nothing is configured - */ - public int getRetentionDataSourceConnectionVerificationTimeout() { - - return getConfigElementFromKey(OpenBankingConstants.RETENTION_DATA_DB_CONNECTION_VERIFICATION_TIMEOUT) - == null ? 1 : Integer.parseInt(getConfigElementFromKey( - OpenBankingConstants.RETENTION_DATA_DB_CONNECTION_VERIFICATION_TIMEOUT).toString().trim()); - } - - /** - * Method to get isEnabled config for consent data retention feature. - * @return consent data retention is enabled - */ - public boolean isConsentDataRetentionEnabled() { - - return getConfigElementFromKey(OpenBankingConstants.IS_CONSENT_DATA_RETENTION_ENABLED) == null ? false : - (Boolean.parseBoolean(getConfigElementFromKey( - OpenBankingConstants.IS_CONSENT_DATA_RETENTION_ENABLED).toString().trim())); - } - - - /** - * Method to get isEnabled config for consent data retention periodical job. - * @return consent data retention is enabled - */ - public boolean isRetentionDataDBSyncEnabled() { - - return getConfigElementFromKey(OpenBankingConstants.IS_CONSENT_RETENTION_DATA_DB_SYNC_ENABLED) == null ? false : - (Boolean.parseBoolean(getConfigElementFromKey( - OpenBankingConstants.IS_CONSENT_RETENTION_DATA_DB_SYNC_ENABLED).toString().trim())); - } - - - /** - * Method to get configs for data retention db sync periodical job's cron value. - * @return data retention job's cron string - */ - public String getRetentionDataDBSyncCronExpression() { - - return getConfigElementFromKey(OpenBankingConstants.CONSENT_RETENTION_DATA_DB_SYNC_CRON) == null - ? OpenBankingConstants.DEFAULT_MIDNIGHT_CRON : - ((String) getConfigElementFromKey(OpenBankingConstants.CONSENT_RETENTION_DATA_DB_SYNC_CRON)).trim(); - } - - /** - * Truststore dynamic loading interval. - * - * @return truststore dynamic loading time in seconds - */ - public Long getTruststoreDynamicLoadingInterval() { - try { - Object truststoreDynamicLoadingInterval = - getConfigElementFromKey(OpenBankingConstants.TRUSTSTORE_DYNAMIC_LOADING_INTERVAL); - if (truststoreDynamicLoadingInterval != null) { - return Long.parseLong((String) truststoreDynamicLoadingInterval); - } else { - return Long.parseLong("86400"); - } - } catch (NumberFormatException e) { - throw new NumberFormatException("Error occurred while reading the truststore dynamic loading interval " + - "value in open-banking.xml. " + e.getMessage()); - } - } - - /** - * Returns the revocation validators map. - *

- * The revocation validator map contains revocation type (OCSP/CRL) and its executing priority. - * The default priority value has set as 1 for OCSP type, as OCSP validation is faster than the CRL validation - * - * @return certificate revocation validators map - */ - public Map getCertificateRevocationValidators() { - return revocationValidators; - } - - public Map> getOpenBankingDCRRegistrationParams() { - return dcrRegistrationConfigs; - } - - public String getAuthServletExtension() { - return getConfigElementFromKey(OpenBankingConstants.AUTH_SERVLET_EXTENSION) == null ? "" : - ((String) getConfigElementFromKey(OpenBankingConstants.AUTH_SERVLET_EXTENSION)).trim(); - } - - public String getCibaServletExtension() { - return getConfigElementFromKey(OpenBankingConstants.CIBA_SERVLET_EXTENSION) == null ? "" : - ((String) getConfigElementFromKey(OpenBankingConstants.CIBA_SERVLET_EXTENSION)).trim(); - } - - public String getJWKSConnectionTimeOut() { - - return getConfigElementFromKey(OpenBankingConstants.DCR_JWKS_CONNECTION_TIMEOUT) == null ? "3000" : - ((String) getConfigElementFromKey(OpenBankingConstants.DCR_JWKS_CONNECTION_TIMEOUT)).trim(); - } - - public String getJWKSReadTimeOut() { - - return getConfigElementFromKey(OpenBankingConstants.DCR_JWKS_READ_TIMEOUT) == null ? "3000" : - ((String) getConfigElementFromKey(OpenBankingConstants.DCR_JWKS_READ_TIMEOUT)).trim(); - } - - public String getSPMetadataFilterExtension() { - return getConfigElementFromKey(OpenBankingConstants.SP_METADATA_FILTER_EXTENSION) == null ? "" : - ((String) getConfigElementFromKey(OpenBankingConstants.SP_METADATA_FILTER_EXTENSION)).trim(); - } - - public Map> getAllowedScopes() { - return allowedScopes; - } - - public Map> getAllowedAPIs() { - return allowedAPIs; - } - - /** - * Method to get configs for periodical consent expiration job's cron value. - * @return consent expiration job's cron string - */ - public String getConsentExpiryCronExpression() { - - return getConfigElementFromKey(OpenBankingConstants.CONSENT_PERIODICAL_EXPIRATION_CRON) == null - ? OpenBankingConstants.DEFAULT_MIDNIGHT_CRON : - ((String) getConfigElementFromKey(OpenBankingConstants.CONSENT_PERIODICAL_EXPIRATION_CRON)).trim(); - } - - /** - * Method to get statue for expired consents. - * @return statue for expired consents - */ - public String getStatusWordingForExpiredConsents() { - - return getConfigElementFromKey(OpenBankingConstants.STATUS_FOR_EXPIRED_CONSENT) == null - ? OpenBankingConstants.DEFAULT_STATUS_FOR_EXPIRED_CONSENTS : - ((String) getConfigElementFromKey(OpenBankingConstants.STATUS_FOR_EXPIRED_CONSENT)).trim(); - } - - /** - * Method to get eligible statues for evaluate expiration logic. - * @return eligible statues for evaluate expiration logic - */ - public String getEligibleStatusesForConsentExpiry() { - - return getConfigElementFromKey(OpenBankingConstants.ELIGIBLE_STATUSES_FOR_CONSENT_EXPIRY) == null ? "" : - ((String) getConfigElementFromKey(OpenBankingConstants.ELIGIBLE_STATUSES_FOR_CONSENT_EXPIRY)).trim(); - } - - /** - * Method to get isEnabled config for periodical consent expiration job. - * @return consent expiration job is enabled - */ - public boolean isConsentExpirationPeriodicalJobEnabled() { - - return getConfigElementFromKey(OpenBankingConstants.IS_CONSENT_PERIODICAL_EXPIRATION_ENABLED) == null ? false : - (Boolean.parseBoolean(getConfigElementFromKey( - OpenBankingConstants.IS_CONSENT_PERIODICAL_EXPIRATION_ENABLED).toString().trim())); - } - - public boolean isConsentAmendmentHistoryEnabled() { - - return getConfigElementFromKey(OpenBankingConstants.IS_CONSENT_AMENDMENT_HISTORY_ENABLED) == null ? false : - (Boolean.parseBoolean(getConfigElementFromKey( - OpenBankingConstants.IS_CONSENT_AMENDMENT_HISTORY_ENABLED).toString().trim())); - } - - public String getOBKeyManagerExtensionImpl() { - return getConfigElementFromKey(OpenBankingConstants.OB_KEYMANAGER_EXTENSION_IMPL) == null ? "" : - ((String) getConfigElementFromKey(OpenBankingConstants.OB_KEYMANAGER_EXTENSION_IMPL)) - .trim(); - } - - /** - * ConnectionPool maximum connection count. - * - * @return maximum connections count, default value is 2000 - */ - public int getConnectionPoolMaxConnections() { - try { - Object maxConnectionsCount = - getConfigElementFromKey(OpenBankingConstants.CONNECTION_POOL_MAX_CONNECTIONS); - if (maxConnectionsCount != null) { - return Integer.parseInt(String.valueOf(maxConnectionsCount)); - } else { - return 2000; - } - } catch (NumberFormatException e) { - throw new NumberFormatException("Error occurred while reading the MaxConnections " + - "value in open-banking.xml. " + e.getMessage()); - } - } - - /** - * ConnectionPool maximum connection per route count. - * - * @return maximum connections per route value, default value is 1500 - */ - public int getConnectionPoolMaxConnectionsPerRoute() { - try { - Object maxConnectionsPerRouteCount = - getConfigElementFromKey(OpenBankingConstants.CONNECTION_POOL_MAX_CONNECTIONS_PER_ROUTE); - if (maxConnectionsPerRouteCount != null) { - return Integer.parseInt(String.valueOf(maxConnectionsPerRouteCount)); - } else { - return 1500; - } - } catch (NumberFormatException e) { - throw new NumberFormatException("Error occurred while reading the MaxConnectionsPerRoute " + - "value in open-banking.xml. " + e.getMessage()); - } - } - - private void buildServiceActivatorSubscribers() { - OMElement serviceActivatorElement = rootElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, OpenBankingConstants.SERVICE_ACTIVATOR_TAG)); - - if (serviceActivatorElement != null) { - OMElement subscribers = serviceActivatorElement.getFirstChildWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, OpenBankingConstants.SA_SUBSCRIBERS_TAG)); - - if (subscribers != null) { - Iterator subscriber = subscribers.getChildrenWithName( - new QName(OpenBankingConstants.OB_CONFIG_QNAME, OpenBankingConstants.SA_SUBSCRIBER_TAG)); - if (subscriber != null) { - while (subscriber.hasNext()) { - OMElement executorElement = subscriber.next(); - //Retrieve subscriber class name from service activator configs - final String subscriberClass = executorElement.getText(); - - if (!StringUtils.isEmpty(subscriberClass)) { - serviceActivatorSubscribers.add(subscriberClass); - } - } - } - } - } - } - - /** - * Returns a list of FQNs of the OBServiceObserver interface implementations. - * - * @return ServiceActivator subscribers FQNs. - */ - public List getServiceActivatorSubscribers() { - return serviceActivatorSubscribers; - } - - //Event notifications configurations. - public String getEventNotificationTokenIssuer() { - - return getConfigElementFromKey(OpenBankingConstants.TOKEN_ISSUER) == null ? "www.wso2.com" : - ((String) getConfigElementFromKey(OpenBankingConstants.TOKEN_ISSUER)).trim(); - } - - public int getNumberOfSetsToReturn() { - - return getConfigElementFromKey(OpenBankingConstants.MAX_SETS_TO_RETURN) == null ? 5 : - Integer.parseInt((String) getConfigElementFromKey(OpenBankingConstants.MAX_SETS_TO_RETURN)); - } - - public boolean isSubClaimIncluded() { - - return getConfigElementFromKey(OpenBankingConstants.IS_SUB_CLAIM_INCLUDED) == null ? false : - (Boolean.parseBoolean(getConfigElementFromKey( - OpenBankingConstants.IS_SUB_CLAIM_INCLUDED).toString().trim())); - } - - public boolean isToeClaimIncluded() { - return getConfigElementFromKey(OpenBankingConstants.IS_TOE_CLAIM_INCLUDED) == null ? false : - (Boolean.parseBoolean(getConfigElementFromKey( - OpenBankingConstants.IS_TOE_CLAIM_INCLUDED).toString().trim())); - } - - public boolean isTxnClaimIncluded() { - return getConfigElementFromKey(OpenBankingConstants.IS_TXN_CLAIM_INCLUDED) == null ? false : - (Boolean.parseBoolean(getConfigElementFromKey( - OpenBankingConstants.IS_TXN_CLAIM_INCLUDED).toString().trim())); - } - - /** - * Returns the expiry time for cache modification. - * - * @return String Expiry time. - */ - public String getCommonCacheModifiedExpiryTime() { - - return getConfigElementFromKey(OpenBankingConstants.COMMON_IDENTITY_CACHE_MODIFY_EXPIRY) == null ? "60" : - ((String) getConfigElementFromKey(OpenBankingConstants.COMMON_IDENTITY_CACHE_MODIFY_EXPIRY)).trim(); - } - - /** - * Returns the expiry time for cache access. - * - * @return String Expiry time. - */ - public String getCommonCacheAccessExpiryTime() { - return getConfigElementFromKey(OpenBankingConstants.COMMON_IDENTITY_CACHE_ACCESS_EXPIRY) == null ? "60" : - ((String) getConfigElementFromKey(OpenBankingConstants.COMMON_IDENTITY_CACHE_ACCESS_EXPIRY)).trim(); - } - - /** - * Alias of the signing certificate in Production Environment. - * - * @return signing certificate alias - */ - public String getOBIdnRetrieverSigningCertificateAlias() { - - return getConfigElementFromKey(OpenBankingConstants.OB_IDN_RETRIEVER_SIG_ALIAS) == null ? "wso2carbon" : - ((String) getConfigElementFromKey(OpenBankingConstants.OB_IDN_RETRIEVER_SIG_ALIAS)).trim(); - } - - /** - * Alias of the signing certificate in Sandbox Environment. - * - * @return signing certificate alias - */ - public String getOBIdnRetrieverSandboxSigningCertificateAlias() { - - return getConfigElementFromKey(OpenBankingConstants.OB_IDN_RETRIEVER_SANDBOX_SIG_ALIAS) == null ? "wso2carbon" : - ((String) getConfigElementFromKey(OpenBankingConstants.OB_IDN_RETRIEVER_SANDBOX_SIG_ALIAS)).trim(); - } - - /** - * Key ID of the public key of the corresponding private key used for signing. - * - * @return signing certificate Kid in Production environment - */ - public String getOBIdnRetrieverSigningCertificateKid() { - - return getConfigElementFromKey(OpenBankingConstants.OB_IDN_RETRIEVER_SIG_KID) == null ? "1234" : - ((String) getConfigElementFromKey(OpenBankingConstants.OB_IDN_RETRIEVER_SIG_KID)).trim(); - } - - /** - * Key ID of the public key of the corresponding private key used for signing. - * - * @return signing certificate Kid in sandbox environment - */ - public String getOBIdnRetrieverSandboxCertificateKid() { - - return getConfigElementFromKey(OpenBankingConstants.OB_IDN_RETRIEVER_SANDBOX_KID) == null ? "5678" : - ((String) getConfigElementFromKey(OpenBankingConstants.OB_IDN_RETRIEVER_SANDBOX_KID)).trim(); - } - - /** - * JWKS Retriever Size Limit for JWS Signature Handling. - * - * @return JWKS Retriever Size Limit - */ - public String getJwksRetrieverSizeLimit() { - - return getConfigElementFromKey(OpenBankingConstants.JWKS_RETRIEVER_SIZE_LIMIT) == null ? "51200" : - ((String) getConfigElementFromKey(OpenBankingConstants.JWKS_RETRIEVER_SIZE_LIMIT)).trim(); - } - - /** - * JWKS Retriever Connection Timeout for JWS Signature Handling. - * - * @return JWKS Retriever Connection Timeout - */ - public String getJwksRetrieverConnectionTimeout() { - - return getConfigElementFromKey(OpenBankingConstants.JWKS_RETRIEVER_CONN_TIMEOUT) == null ? "2000" : - ((String) getConfigElementFromKey(OpenBankingConstants.JWKS_RETRIEVER_CONN_TIMEOUT)).trim(); - } - - /** - * JWKS Retriever Read Timeout for JWS Signature Handling. - * - * @return JWKS Retriever Read Timeout - */ - public String getJwksRetrieverReadTimeout() { - - return getConfigElementFromKey(OpenBankingConstants.JWKS_RETRIEVER_READ_TIMEOUT) == null ? "2000" : - ((String) getConfigElementFromKey(OpenBankingConstants.JWKS_RETRIEVER_READ_TIMEOUT)).trim(); - } - - /** - * Check if Jws Signature Validation is enabled. - * - * @return if Jws Signature Validation is enabled - */ - public boolean isJwsSignatureValidationEnabled() { - - return getConfigElementFromKey(OpenBankingConstants.JWS_SIG_VALIDATION_ENABLE) != null && - Boolean.parseBoolean(((String) getConfigElementFromKey(OpenBankingConstants.JWS_SIG_VALIDATION_ENABLE)) - .trim()); - } - - /** - * Check if Jws Response signing is enabled. - * - * @return if Jws message Response is enabled - */ - public boolean isJwsResponseSigningEnabled() { - - return getConfigElementFromKey(OpenBankingConstants.JWS_RESP_SIGNING_ENABLE) != null && - Boolean.parseBoolean(((String) getConfigElementFromKey(OpenBankingConstants.JWS_RESP_SIGNING_ENABLE)) - .trim()); - } - - /** - * Jws Request Signing allowed algorithms. - * - * @return Jws Request Signing allowed algorithms - */ - public List getJwsRequestSigningAlgorithms() { - - Object allowedAlgorithmsElement = getConfigElementFromKey( - OpenBankingConstants.JWS_SIG_VALIDATION_ALGO) == null ? new String[] {"PS256"} : - (getConfigElementFromKey(OpenBankingConstants.JWS_SIG_VALIDATION_ALGO)); - List allowedAlgorithmsList = new ArrayList<>(); - if (allowedAlgorithmsElement instanceof ArrayList) { - allowedAlgorithmsList.addAll((ArrayList) allowedAlgorithmsElement); - } else if (allowedAlgorithmsElement instanceof String) { - allowedAlgorithmsList.add((String) allowedAlgorithmsElement); - } - return allowedAlgorithmsList.isEmpty() ? Arrays.asList("PS256") : allowedAlgorithmsList; - } - - /** - * Jws Response Signing allowed algorithm. - * - * @return Jws Response Signing allowed algorithm - */ - public String getJwsResponseSigningAlgorithm() { - - return getConfigElementFromKey(OpenBankingConstants.JWS_RESP_SIGNING_ALGO) == null ? "PS256" : - ((String) getConfigElementFromKey(OpenBankingConstants.JWS_RESP_SIGNING_ALGO)).trim(); - } - - public Map getAuthWorkerConfig() { - return authWorkerConfig; - } - - /** - * Method to check if the Dispute Resolution feature is enabled. - * @return true if Dispute Resolution is enabled. - */ - public boolean isDisputeResolutionEnabled() { - - return getConfigElementFromKey(OpenBankingConstants.IS_DISPUTE_RESOLUTION_ENABLED) == null ? false : - (Boolean.parseBoolean(getConfigElementFromKey( - OpenBankingConstants.IS_DISPUTE_RESOLUTION_ENABLED).toString().trim())); - } - - /** - * Method to check if the Dispute Resolution feature is enabled for Non Error Scenarios. - * @return true if Dispute Resolution feature is enabled for Non Error scenarios - */ - public boolean isNonErrorDisputeDataPublishingEnabled() { - - return getConfigElementFromKey(OpenBankingConstants.PUBLISH_NON_ERROR_DISPUTE_DATA) == null ? false : - (Boolean.parseBoolean(getConfigElementFromKey( - OpenBankingConstants.PUBLISH_NON_ERROR_DISPUTE_DATA).toString().trim())); - } - - /** - * Method to get maximum length for publish response body in Dispute Resolution Feature. - * @return maximum length for response body. - */ - public int getMaxResponseBodyLength() { - - return getConfigElementFromKey(OpenBankingConstants.MAX_RESPONSE_BODY_LENGTH) - == null ? 4096 : (Integer.parseInt(getConfigElementFromKey( - OpenBankingConstants.MAX_RESPONSE_BODY_LENGTH).toString().trim())); - } - - /** - * Method to get maximum length for publish request body in Dispute Resolution Feature. - * @return maximum length for request body. - */ - public int getMaxRequestBodyLength() { - - return getConfigElementFromKey(OpenBankingConstants.MAX_REQUEST_BODY_LENGTH) - == null ? 4096 : (Integer.parseInt(getConfigElementFromKey( - OpenBankingConstants.MAX_REQUEST_BODY_LENGTH).toString().trim())); - } - - /** - *Method to get maximum length for publish headers in Dispute Resolution Feature. - * @return maximum length for headers. - */ - public int getMaxHeaderLength() { - - return getConfigElementFromKey(OpenBankingConstants.MAX_HEADER_LENGTH) - == null ? 2048 : (Integer.parseInt(getConfigElementFromKey( - OpenBankingConstants.MAX_HEADER_LENGTH).toString().trim())); - } - - /** - * Method to determine real-time event notification feature is enabled or not from the configurations. - * - * @return boolean value indicating the state - */ - public boolean isRealtimeEventNotificationEnabled() { - return getConfigElementFromKey(OpenBankingConstants.REALTIME_EVENT_NOTIFICATION_ENABLED) != null - && (Boolean.parseBoolean(getConfigElementFromKey( - OpenBankingConstants.REALTIME_EVENT_NOTIFICATION_ENABLED).toString().trim())); - } - - /** - * Method to get periodic Cron expression config for realtime event notifications scheduler. - * - * @return String Cron expression to trigger the Cron job for real-time event notification - */ - public String getRealtimeEventNotificationSchedulerCronExpression() { - return getConfigElementFromKey(OpenBankingConstants.PERIODIC_CRON_EXPRESSION) - == null ? "0 0/1 0 ? * * *" : (String) getConfigElementFromKey( - OpenBankingConstants.PERIODIC_CRON_EXPRESSION); - } - - /** - * Method to get TIMEOUT_IN_SECONDS config for realtime event notifications. - * - * @return integer timeout for the HTTP Client's POST requests - */ - public int getRealtimeEventNotificationTimeoutInSeconds() { - return getConfigElementFromKey(OpenBankingConstants.TIMEOUT_IN_SECONDS) - == null ? 60 : (Integer.parseInt(getConfigElementFromKey( - OpenBankingConstants.TIMEOUT_IN_SECONDS).toString().trim())); - } - - /** - * Method to get MAX_RETRIES config for realtime event notifications. - * - * @return integer maximum number of retries to the retry policy in real-time notification sender - */ - public int getRealtimeEventNotificationMaxRetries() { - return getConfigElementFromKey(OpenBankingConstants.MAX_RETRIES) - == null ? 5 : (Integer.parseInt(getConfigElementFromKey( - OpenBankingConstants.MAX_RETRIES).toString().trim())); - } - - /** - * Method to get INITIAL_BACKOFF_TIME_IN_SECONDS config for realtime event notifications. - * - * @return integer start waiting time for the retry policy before the first retry - */ - public int getRealtimeEventNotificationInitialBackoffTimeInSeconds() { - return getConfigElementFromKey(OpenBankingConstants.INITIAL_BACKOFF_TIME_IN_SECONDS) - == null ? 60 : (Integer.parseInt(getConfigElementFromKey( - OpenBankingConstants.INITIAL_BACKOFF_TIME_IN_SECONDS).toString().trim())); - } - - /** - * Method to get BACKOFF_FUNCTION config for realtime event notifications. - * Function name should be "EX", "CONSTANT" or "LINEAR". - * - * @return string indicating the retry function - */ - public String getRealtimeEventNotificationBackoffFunction() { - return getConfigElementFromKey(OpenBankingConstants.BACKOFF_FUNCTION) - == null ? "EX" : (String) getConfigElementFromKey( - OpenBankingConstants.BACKOFF_FUNCTION); - } - - /** - * Method to get CIRCUIT_BREAKER_OPEN_TIMEOUT_IN_SECONDS config for realtime event notifications. - * - * @return integer timeout to break the retrying process and make that notification as ERR - */ - public int getRealtimeEventNotificationCircuitBreakerOpenTimeoutInSeconds() { - return getConfigElementFromKey(OpenBankingConstants.CIRCUIT_BREAKER_OPEN_TIMEOUT_IN_SECONDS) - == null ? 600 : (Integer.parseInt(getConfigElementFromKey( - OpenBankingConstants.CIRCUIT_BREAKER_OPEN_TIMEOUT_IN_SECONDS).toString().trim())); - } - - /** - * Method to get EVENT_NOTIFICATION_THREADPOOL_SIZE config for realtime event notifications. - * - * @return integer fix size to set the Thread Pool size in the real-time event notification sender - */ - public int getEventNotificationThreadpoolSize() { - return getConfigElementFromKey(OpenBankingConstants.EVENT_NOTIFICATION_THREADPOOL_SIZE) - == null ? 20 : (Integer.parseInt(getConfigElementFromKey( - OpenBankingConstants.EVENT_NOTIFICATION_THREADPOOL_SIZE).toString().trim())); - } - - /** - * Method to get EVENT_NOTIFICATION_GENERATOR config for event notifications. - * - * @return String class name of the event notification generator to generate the event notification payload - */ - public String getEventNotificationGenerator() { - return getConfigElementFromKey(OpenBankingConstants.EVENT_NOTIFICATION_GENERATOR) == null ? - "com.wso2.openbanking.accelerator.event.notifications.service.service.DefaultEventNotificationGenerator" - : (String) getConfigElementFromKey(OpenBankingConstants.EVENT_NOTIFICATION_GENERATOR); - } - - /** - * Method to get REALTIME_EVENT_NOTIFICATION_REQUEST_GENERATOR config for realtime event notifications. - * - * @return String class path of the realtime event notification payload generator - */ - public String getRealtimeEventNotificationRequestGenerator() { - return getConfigElementFromKey(OpenBankingConstants.REALTIME_EVENT_NOTIFICATION_REQUEST_GENERATOR) == null ? - "com.wso2.openbanking.accelerator.event.notifications.service." + - "realtime.service.DefaultRealtimeEventNotificationRequestGenerator" - : (String) getConfigElementFromKey(OpenBankingConstants.REALTIME_EVENT_NOTIFICATION_REQUEST_GENERATOR); - } - - /** - * Method to get software environment identification SSA property name. - * - * @return String software environment identification SSA property name. - */ - public String getSoftwareEnvIdentificationSSAPropertyName() { - return getConfigElementFromKey(OpenBankingConstants.DCR_SOFTWARE_ENV_IDENTIFICATION_PROPERTY_NAME) == null ? - OpenBankingConstants.SOFTWARE_ENVIRONMENT : (String) getConfigElementFromKey( - OpenBankingConstants.DCR_SOFTWARE_ENV_IDENTIFICATION_PROPERTY_NAME); - } - - /** - * Method to get software environment identification value for sandbox in SSA. - * - * @return String software environment identification value for sandbox. - */ - public String getSoftwareEnvIdentificationSSAPropertyValueForSandbox() { - return getConfigElementFromKey(OpenBankingConstants.DCR_SOFTWARE_ENV_IDENTIFICATION_VALUE_FOR_SANDBOX) == null ? - "sandbox" : (String) getConfigElementFromKey( - OpenBankingConstants.DCR_SOFTWARE_ENV_IDENTIFICATION_VALUE_FOR_SANDBOX); - } - - /** - * Method to get software environment identification value for production in SSA. - * - * @return String software environment identification value for production. - */ - public String getSoftwareEnvIdentificationSSAPropertyValueForProduction() { - return getConfigElementFromKey( - OpenBankingConstants.DCR_SOFTWARE_ENV_IDENTIFICATION_VALUE_FOR_PRODUCTION) == null ? - "production" : (String) getConfigElementFromKey( - OpenBankingConstants.DCR_SOFTWARE_ENV_IDENTIFICATION_VALUE_FOR_PRODUCTION); - } - - /** - * Get config related for checking whether PSU is a federated user or not. - * - * @return Boolean value indicating whether PSU is a federated user or not - */ - public boolean isPSUFederated() { - - Object isPSUFederated = getConfigElementFromKey(OpenBankingConstants.IS_PSU_FEDERATED); - if (isPSUFederated != null) { - return Boolean.parseBoolean((String) isPSUFederated); - } else { - return false; - } - } - - /** - * Get Federated PSU IDP Name. - * - * @return String Federated IDP name - */ - public String getFederatedIDPName() { - - return getConfigElementFromKey(OpenBankingConstants.PSU_FEDERATED_IDP_NAME) == null ? "" : - ((String) getConfigElementFromKey(OpenBankingConstants.PSU_FEDERATED_IDP_NAME)).trim(); - } - - /** - * Method to get the value Idempotency enable configuration. - * @return Whether Idempotency is enabled or not - */ - public boolean isIdempotencyValidationEnabled() { - return getConfigElementFromKey(OpenBankingConstants.IDEMPOTENCY_IS_ENABLED) != null && - Boolean.parseBoolean(((String) - getConfigElementFromKey(OpenBankingConstants.IDEMPOTENCY_IS_ENABLED)).trim()); - } - - /** - * Method to get the value Idempotency allowed time configuration. - * @return Idempotency allowed time - */ - public String getIdempotencyAllowedTime() { - return getConfigElementFromKey(OpenBankingConstants.IDEMPOTENCY_ALLOWED_TIME) == null ? "1440" : - (String) getConfigElementFromKey(OpenBankingConstants.IDEMPOTENCY_ALLOWED_TIME); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/config/OpenBankingConfigurationService.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/config/OpenBankingConfigurationService.java deleted file mode 100644 index 0e526caa..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/config/OpenBankingConfigurationService.java +++ /dev/null @@ -1,47 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.config; - -import java.util.List; -import java.util.Map; - -/** - * Interface to expose Configurations as an OSGi Service. - */ -public interface OpenBankingConfigurationService { - - public Map getConfigurations(); - - public Map> getExecutors(); - - public Map> getDataPublishingStreams(); - - public Map> getDataPublishingValidationMap(); - - public Map> getDCRRegistrationConfigurations(); - - public Map> getAuthorizeSteps(); - - public Map> getAllowedScopes(); - - public Map> getAllowedAPIs(); - - public Map getAuthenticationWorkers(); - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/config/OpenBankingConfigurationServiceImpl.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/config/OpenBankingConfigurationServiceImpl.java deleted file mode 100644 index 00d0b929..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/config/OpenBankingConfigurationServiceImpl.java +++ /dev/null @@ -1,87 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.config; - -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.List; -import java.util.Map; - -/** - * Implementation of Open Banking Configuration Service. - */ -public class OpenBankingConfigurationServiceImpl implements OpenBankingConfigurationService { - - private static final OpenBankingConfigParser openBankingConfigParser = OpenBankingConfigParser.getInstance(); - private static final Log log = LogFactory.getLog(OpenBankingConfigurationServiceImpl.class); - - @Override - public Map getConfigurations() { - - return openBankingConfigParser.getConfiguration(); - } - - @Override - public Map> getExecutors() { - - return openBankingConfigParser.getOpenBankingExecutors(); - } - - @Override - public Map> getDataPublishingStreams() { - - return openBankingConfigParser.getDataPublishingStreams(); - } - - @Override - public Map> getDataPublishingValidationMap() { - - return openBankingConfigParser.getDataPublishingValidationMap(); - } - - @Override - public Map> getDCRRegistrationConfigurations() { - - return openBankingConfigParser.getOpenBankingDCRRegistrationParams(); - } - - @Override - public Map> getAuthorizeSteps() { - - return openBankingConfigParser.getConsentAuthorizeSteps(); - } - - @Override - public Map> getAllowedScopes() { - return openBankingConfigParser.getAllowedScopes(); - } - - @Override - public Map> getAllowedAPIs() { - return openBankingConfigParser.getAllowedAPIs(); - } - - @Override - public Map getAuthenticationWorkers() { - return openBankingConfigParser.getAuthWorkerConfig(); - } - - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/config/TextFileReader.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/config/TextFileReader.java deleted file mode 100644 index 790e3949..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/config/TextFileReader.java +++ /dev/null @@ -1,118 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.config; - -import com.wso2.openbanking.accelerator.common.util.CarbonUtils; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import org.apache.commons.io.FilenameUtils; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.io.BufferedInputStream; -import java.io.File; -import java.io.FileInputStream; -import java.io.IOException; -import java.io.InputStream; -import java.util.HashMap; -import java.util.Map; - -/** - * Common class to read a text file. - */ -public class TextFileReader { - - private String directoryPath; - private Map files = new HashMap<>(); - private static volatile TextFileReader textFileReader; - private static final Log logger = LogFactory.getLog(TextFileReader.class); - private static final Object lock = new Object(); - - private TextFileReader() { - - } - - public static TextFileReader getInstance() { - - if (textFileReader == null) { - synchronized (lock) { - if (textFileReader == null) { - textFileReader = new TextFileReader(); - } - } - } - return textFileReader; - } - - public String getDirectoryPath() { - - return directoryPath; - } - - public void setDirectoryPath(String directoryPath) { - - this.directoryPath = directoryPath; - } - - /** - * To read the auth textFile from the given file path. - * - * @param fileName Path of the file. - * @return file content as a String - * @throws IOException IO Exception. - */ - @SuppressFBWarnings({"WEAK_FILENAMEUTILS", "PATH_TRAVERSAL_IN"}) - // Suppressed content - FilenameUtils.getName() - // Suppression reason - - // WEAK_FILENAMEUTILS - False positive: The vulnerability is fixed from Java 7 update 40 and Java 8+ versions - // PATH_TRAVERSAL_IN - False positive: The user input value is only filename and it is secured using - // FilenameUtils. This could be a true positive if directory path is sent - // as a user input in the future. - // Suppressed warning count - 2 - public String readFile(String fileName) throws IOException { - - String filePath; - if (files.containsKey(fileName)) { - return files.get(FilenameUtils.getName(fileName)); - } - if (StringUtils.isNotEmpty(directoryPath)) { - filePath = directoryPath + File.separator + FilenameUtils.getName(fileName); - } else { - filePath = CarbonUtils.getCarbonConfigDirPath() + File.separator + FilenameUtils.getName(fileName); - } - File file = new File(filePath); - if (file.exists()) { - try (InputStream resourceAsStream = new FileInputStream(filePath); - BufferedInputStream bufferedInputStream = new BufferedInputStream(resourceAsStream)) { - - StringBuilder resourceFile = new StringBuilder(); - int c; - while ((c = bufferedInputStream.read()) != -1) { - char val = (char) c; - resourceFile.append(val); - } - files.put(fileName, resourceFile.toString()); - if (logger.isDebugEnabled()) { - logger.debug("File " + fileName.replaceAll("[\r\n]", "") + "read and stored in memory"); - } - return resourceFile.toString(); - } - } - return ""; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/constant/OpenBankingConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/constant/OpenBankingConstants.java deleted file mode 100644 index f04b3229..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/constant/OpenBankingConstants.java +++ /dev/null @@ -1,272 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.constant; - - -/** - * Class containing the constants for Open Banking Common module. - */ -public class OpenBankingConstants { - - public static final String OB_CONFIG_FILE = "open-banking.xml"; - public static final String CARBON_HOME = "carbon.home"; - - public static final String OB_CONFIG_QNAME = "http://wso2.org/projects/carbon/open-banking.xml"; - public static final String GATEWAY_CONFIG_TAG = "Gateway"; - public static final String GATEWAY_EXECUTOR_CONFIG_TAG = "OpenBankingGatewayExecutors"; - public static final String EVENT_CONFIG_TAG = "Event"; - public static final String EVENT_EXECUTOR_CONFIG_TAG = "EventExecutors"; - public static final String EXECUTOR_CONFIG_TAG = "Executor"; - public static final String DCR_CONFIG_TAG = "DCR"; - public static final String DCR_REGISTRATION_CONFIG_TAG = "RegistrationRequestParams"; - public static final String DCR_REGISTRATION_PARAM_ALLOWED_VALUE_TAG = "AllowedValues"; - public static final String REGULATORY = "regulatory"; - public static final String DATA_PUBLISHING_CONFIG_TAG = "DataPublishing"; - public static final String THRIFT_CONFIG_TAG = "Thrift"; - public static final String STREAMS_CONFIG_TAG = "Streams"; - public static final String ATTRIBUTE_CONFIG_TAG = "Attribute"; - public static final String REQUIRED = "required"; - public static final String ATTRIBUTE_TYPE = "type"; - public static final String DEFAULT_MIDNIGHT_CRON = "0 0 0 * * ?"; - public static final String DEFAULT_STATUS_FOR_EXPIRED_CONSENTS = "Expired"; - public static final String DEFAULT_STATUS_FOR_REVOKED_CONSENTS = "Revoked"; - public static final String IS_CONSENT_REVOCATION_FLOW = "IS_CONSENT_REVOCATION_FLOW"; - - public static final String SIGNATURE_ALGORITHMS = "SignatureValidation.AllowedAlgorithms.Algorithm"; - public static final String AUTH_SERVLET_EXTENSION = "Identity.AuthenticationWebApp.ServletExtension"; - public static final String COMMON_IDENTITY_CACHE_ACCESS_EXPIRY = "Common.Identity.Cache.CacheAccessExpiry"; - public static final String COMMON_IDENTITY_CACHE_MODIFY_EXPIRY = "Common.Identity.Cache.CacheModifiedExpiry"; - public static final String JWKS_ENDPOINT_NAME = "DCR.JWKSEndpointName"; - public static final String SP_METADATA_FILTER_EXTENSION = - "Identity.ApplicationInformationEndpoint.SPMetadataFilterExtension"; - public static final String CIBA_SERVLET_EXTENSION = "Identity.CIBAAuthenticationEndpointWebApp.ServletExtension"; - public static final String DCR_JWKS_CONNECTION_TIMEOUT = "DCR.JWKS-Retriever.ConnectionTimeout"; - public static final String DCR_JWKS_READ_TIMEOUT = "DCR.JWKS-Retriever.ReadTimeout"; - public static final String DCR_USE_SOFTWAREID_AS_APPNAME = "DCR.UseSoftwareIdAsAppName"; - public static final String DCR_JWKS_NAME = "DCR.JWKSEndpointName"; - public static final String DCR_APPLICATION_NAME_KEY = "DCR.ApplicationName"; - public static final String OB_KM_NAME = "KeyManagerName"; - public static final String DCR_SOFTWARE_ENV_IDENTIFICATION_PROPERTY_NAME = - "DCR.RegistrationRequestParams.SoftwareEnvironmentIdentification.PropertyName"; - public static final String DCR_SOFTWARE_ENV_IDENTIFICATION_VALUE_FOR_SANDBOX = - "DCR.RegistrationRequestParams.SoftwareEnvironmentIdentification.PropertyValueForSandbox"; - public static final String DCR_SOFTWARE_ENV_IDENTIFICATION_VALUE_FOR_PRODUCTION = - "DCR.RegistrationRequestParams.SoftwareEnvironmentIdentification.PropertyValueForProduction"; - - public static final String APIM_APPCREATION = "DCR.APIMRESTEndPoints.AppCreation"; - public static final String APIM_KEYGENERATION = "DCR.APIMRESTEndPoints.KeyGeneration"; - public static final String APIM_GETAPIS = "DCR.APIMRESTEndPoints.RetrieveAPIS"; - public static final String APIM_SUBSCRIBEAPIS = "DCR.APIMRESTEndPoints.SubscribeAPIs"; - public static final String APIM_GETSUBSCRIPTIONS = "DCR.APIMRESTEndPoints.RetrieveSubscribedAPIs"; - public static final String REGULATORY_API_NAMES = "RegulatoryAPINames"; - public static final String API_NAME = "name"; - public static final String API_ROLE = "roles"; - public static final String API_ID = "id"; - public static final String API_LIST = "list"; - public static final String REGULATORY_API = "API"; - public static final String SOFTWARE_ROLES = "software_roles"; - public static final String SOFTWARE_STATEMENT = "software_statement"; - public static final String SOFTWARE_ID = "software_id"; - public static final String JWT_BODY = "body"; - public static final String SOFTWARE_ENVIRONMENT = "software_environment"; - public static final String TOKEN_ENDPOINT = "DCR.TokenEndpoint"; - public static final String STORE_HOSTNAME = "PublisherURL"; - - public static final String JDBC_PERSISTENCE_CONFIG = "JDBCPersistenceManager.DataSource.Name"; - public static final String DB_CONNECTION_VERIFICATION_TIMEOUT = - "JDBCPersistenceManager.ConnectionVerificationTimeout"; - public static final String JDBC_RETENTION_DATA_PERSISTENCE_CONFIG = - "JDBCRetentionDataPersistenceManager.DataSource.Name"; - public static final String RETENTION_DATA_DB_CONNECTION_VERIFICATION_TIMEOUT = - "JDBCRetentionDataPersistenceManager.ConnectionVerificationTimeout"; - - public static final String TRUSTSTORE_CONF_TYPE_DEFAULT = "JKS"; - public static final String CLIENT_CERT_CACHE = "ClientCertCache"; - public static final String OB_CACHE_MANAGER = "OB_CERTIFICATE_CACHE"; - public static final String CERTIFICATE_REVOCATION_VALIDATION_RETRY_COUNT = "Gateway" + - ".CertificateManagement.CertificateRevocationValidationRetryCount"; - public static final String CERTIFICATE_REVOCATION_VALIDATION_CONNECT_TIMEOUT = "Gateway" + - ".CertificateManagement.CertificateRevocationValidationConnectTimeout"; - public static final String CERTIFICATE_REVOCATION_VALIDATION_CONNECTION_REQUEST_TIMEOUT = "Gateway" + - ".CertificateManagement.CertificateRevocationValidationConnectionRequestTimeout"; - public static final String CERTIFICATE_REVOCATION_VALIDATION_SOCKET_TIMEOUT = "Gateway" + - ".CertificateManagement.CertificateRevocationValidationSocketTimeout"; - public static final String CERTIFICATE_REVOCATION_VALIDATION_ENABLED = "Gateway" + - ".CertificateManagement.CertificateRevocationValidationEnabled"; - public static final String CERTIFICATE_REVOCATION_VALIDATION_EXCLUDED_ISSUERS = "Gateway" + - ".CertificateManagement.RevocationValidationExcludedIssuers.IssuerDN"; - public static final String TPP_VALIDATION_SERVICE_IMPL_CLASS = "Gateway" + - ".TPPManagement.TPPValidation.ServiceImplClass"; - public static final String TPP_VALIDATION_ENABLED = "Gateway" + - ".TPPManagement.TPPValidation.Enabled"; - public static final String PSD2_ROLE_VALIDATION_ENABLED = "Gateway" + - ".TPPManagement.PSD2RoleValidation.Enabled"; - public static final String CERTIFICATE_REVOCATION_PROXY_ENABLED = "Gateway" + - ".CertificateManagement.CertificateRevocationProxy.Enabled"; - public static final String CERTIFICATE_REVOCATION_PROXY_HOST = "Gateway" + - ".CertificateManagement.CertificateRevocationProxy.ProxyHost"; - public static final String CERTIFICATE_REVOCATION_PROXY_PORT = "Gateway" + - ".CertificateManagement.CertificateRevocationProxy.ProxyPort"; - public static final String TRANSPORT_CERT_ISSUER_VALIDATION_ENABLED = "Gateway" + - ".CertificateManagement.TransportCertIssuerValidationEnabled"; - public static final String TRUSTSTORE_DYNAMIC_LOADING_INTERVAL = "Gateway" + - ".CertificateManagement.TrustStoreDynamicLoadingInterval"; - public static final String CLIENT_CERTIFICATE_CACHE_EXPIRY = "Gateway" + - ".CertificateManagement.ClientCertificateCacheExpiry"; - public static final String TPP_VALIDATION_CACHE_EXPIRY = "Gateway" + - ".TPPManagement.TPPValidationCacheExpiry"; - public static final String TPP_VALIDATION_SERVICE_AISP_SCOPE_REGEX = "Gateway" + - ".CertificateManagement.TPPValidationService.ScopeRegexPatterns.AISP"; - public static final String TPP_VALIDATION_SERVICE_PISP_SCOPE_REGEX = "Gateway" + - ".CertificateManagement.TPPValidationService.ScopeRegexPatterns.PISP"; - public static final String TPP_VALIDATION_SERVICE_CBPII_SCOPE_REGEX = "Gateway" + - ".CertificateManagement.TPPValidationService.ScopeRegexPatterns.CBPII"; - public static final String CLIENT_TRANSPORT_CERT_HEADER_NAME = "Gateway" + - ".CertificateManagement.ClientTransportCertHeaderName"; - public static final String URL_ENCODE_CLIENT_TRANSPORT_CERT_HEADER_ENABLED = "Gateway" + - ".CertificateManagement.UrlEncodeClientTransportCertHeaderEnabled"; - public static final int PAGINATION_LIMIT_DEFAULT = 25; - public static final int PAGINATION_OFFSET_DEFAULT = 0; - public static final String CONSENT_CONFIG_TAG = "Consent"; - public static final String AUTHORIZE_STEPS_CONFIG_TAG = "AuthorizeSteps"; - public static final String STEP_CONFIG_TAG = "Step"; - public static final String ALLOWED_SCOPES_CONFIG_TAG = "AllowedScopes"; - public static final String SCOPE_CONFIG_TAG = "Scope"; - public static final String REVOCATION_VALIDATORS_CONFIG_TAG = "RevocationValidators"; - public static final String REVOCATION_VALIDATOR_CONFIG_TAG = "RevocationValidator"; - public static final String TPP_MANAGEMENT_CONFIG_TAG = "TPPManagement"; - public static final String CONNECTION_POOL_MAX_CONNECTIONS = "HTTPConnectionPool.MaxConnections"; - public static final String CONNECTION_POOL_MAX_CONNECTIONS_PER_ROUTE = "HTTPConnectionPool.MaxConnectionsPerRoute"; - public static final String PUSH_AUTH_EXPIRY_TIME = "PushAuthorisation.ExpiryTime"; - public static final String PUSH_AUTH_REQUEST_URI_SUBSTRING = "PushAuthorisation.RequestUriSubString"; - - public static final String CONSENT_PERIODICAL_EXPIRATION_CRON = "Consent.PeriodicalExpiration.CronValue"; - public static final String STATUS_FOR_EXPIRED_CONSENT = "Consent.PeriodicalExpiration.ExpiredConsentStatusValue"; - public static final String IS_CONSENT_PERIODICAL_EXPIRATION_ENABLED = "Consent.PeriodicalExpiration.Enabled"; - public static final String IS_CONSENT_AMENDMENT_HISTORY_ENABLED = "Consent.AmendmentHistory.Enabled"; - public static final String ELIGIBLE_STATUSES_FOR_CONSENT_EXPIRY = - "Consent.PeriodicalExpiration.EligibleStatuses"; - public static final String CONSENT_ID_CLAIM_NAME = "Identity.ConsentIDClaimName"; - - public static final String EVENT_QUEUE_SIZE = "Event.QueueSize"; - public static final String EVENT_WORKER_THREAD_COUNT = "Event.WorkerThreadCount"; - public static final String EVENT_EXECUTOR = "Event.EventExecutor"; - - // Data Retention Constants - public static final String IS_CONSENT_DATA_RETENTION_ENABLED = "Consent.DataRetention.Enabled"; - public static final String IS_CONSENT_RETENTION_DATA_DB_SYNC_ENABLED = "Consent.DataRetention.DBSyncEnabled"; - public static final String CONSENT_RETENTION_DATA_DB_SYNC_CRON = "Consent.DataRetention.CronValue"; - - // Service Activator Constants - public static final String SERVICE_ACTIVATOR_TAG = "ServiceActivator"; - public static final String SA_SUBSCRIBERS_TAG = "Subscribers"; - public static final String SA_SUBSCRIBER_TAG = "Subscriber"; - - //JWS handling related constants - public static final String JWS_SIG_VALIDATION_ENABLE = "JwsSignatureConfiguration.SignatureValidation.Enable"; - public static final String JWS_SIG_VALIDATION_ALGO = - "JwsSignatureConfiguration.SignatureValidation.AllowedAlgorithms"; - public static final String JWS_RESP_SIGNING_ENABLE = "JwsSignatureConfiguration.ResponseSigning.Enable"; - public static final String JWS_RESP_SIGNING_ALGO = "JwsSignatureConfiguration.ResponseSigning.AllowedAlgorithm"; - - // Open Banking Identity Manager - public static final String OB_IDN_RETRIEVER_SIG_ALIAS = "OBIdentityRetriever.Server.SigningCertificateAlias"; - public static final String OB_IDN_RETRIEVER_SANDBOX_SIG_ALIAS = - "OBIdentityRetriever.Server.SandboxSigningCertificateAlias"; - public static final String OB_IDN_RETRIEVER_SIG_KID = "OBIdentityRetriever.Server.SigningCertificateKid"; - public static final String OB_IDN_RETRIEVER_SANDBOX_KID = "OBIdentityRetriever.Server.SandboxCertificateKid"; - public static final String JWKS_RETRIEVER_SIZE_LIMIT = "OBIdentityRetriever.JWKSRetriever.SizeLimit"; - public static final String JWKS_RETRIEVER_CONN_TIMEOUT = "OBIdentityRetriever.JWKSRetriever.ConnectionTimeout"; - public static final String JWKS_RETRIEVER_READ_TIMEOUT = "OBIdentityRetriever.JWKSRetriever.ReadTimeout"; - - // Key Manager Additional Property Configs - public static final String KEY_MANAGER_CONFIG_TAG = "KeyManager"; - public static final String KEY_MANAGER_ADDITIONAL_PROPERTIES_CONFIG_TAG = "KeyManagerAdditionalProperties"; - public static final String PROPERTY_CONFIG_TAG = "Property"; - public static final String OB_KEYMANAGER_EXTENSION_IMPL = - "KeyManager.KeyManagerExtensionImpl"; - - //OB Event Notifications Constants - public static final String TOKEN_ISSUER = "OBEventNotifications.TokenIssuer"; - public static final String MAX_SETS_TO_RETURN = "OBEventNotifications.NumberOfSetsToReturn"; - public static final String SIGNING_ALIAS = "OBEventNotifications.SigningAlias"; - public static final String IS_SUB_CLAIM_INCLUDED = "OBEventNotifications.PollingResponseParams.IsSubClaimAvailable"; - public static final String IS_TXN_CLAIM_INCLUDED = "OBEventNotifications.PollingResponseParams.IsTxnClaimAvailable"; - public static final String IS_TOE_CLAIM_INCLUDED = "OBEventNotifications.PollingResponseParams.IsToeClaimAvailable"; - public static final String EVENT_CREATION_HANDLER = "OBEventNotifications.EventCreationHandler"; - public static final String EVENT_POLLING_HANDLER = "OBEventNotifications.EventPollingHandler"; - public static final String EVENT_SUBSCRIPTION_HANDLER = "OBEventNotifications.EventSubscriptionHandler"; - public static final String EVENT_NOTIFICATION_GENERATOR = "OBEventNotifications.NotificationGenerator"; - public static final String AUTHENTICATION_WORKER_LIST_TAG = "AuthenticationWorkers"; - public static final String AUTHENTICATION_WORKER_TAG = "AuthenticationWorker"; - - // Dispute Resolution Implementation Constants - public static final String IS_DISPUTE_RESOLUTION_ENABLED = "DataPublishing.DisputeResolution.Enabled"; - public static final String PUBLISH_NON_ERROR_DISPUTE_DATA = "DataPublishing" + - ".DisputeResolution.PublishNonErrorDisputeResolutionData"; - public static final String MAX_REQUEST_BODY_LENGTH = "DataPublishing.DisputeResolution.MaxRequestBodyLength"; - public static final String MAX_RESPONSE_BODY_LENGTH = "DataPublishing.DisputeResolution.MaxResponseBodyLength"; - public static final String MAX_HEADER_LENGTH = "DataPublishing.DisputeResolution.MaxHeaderLength"; - public static final String DISPUTE_RESOLUTION_STREAM_NAME = "DisputeResolutionStream"; - public static final String DISPUTE_RESOLUTION_STREAM_VERSION = "1.0.0"; - public static final String REQUEST_BODY = "requestBody"; - public static final String HTTP_METHOD = "httpMethod"; - public static final String STATUS_CODE = "statusCode"; - public static final String RESPONSE_BODY = "responseBody"; - public static final String ELECTED_RESOURCE = "electedResource"; - public static final String HEADERS = "headers"; - public static final String TIMESTAMP = "timestamp"; - - public static final String CUTOFF_DATE_ENABLED = "ConsentManagement.PaymentRestrictions.CutOffDateTime.Enabled"; - public static final String CUTOFF_DATE_POLICY = "ConsentManagement.PaymentRestrictions.CutOffDateTime" + - ".CutOffDateTimePolicy"; - public static final String ZONE_ID = "ZoneId"; - public static final String DAILY_CUTOFF = "ConsentManagement.PaymentRestrictions.CutOffDateTime" + - ".DailyCutOffTime"; - public static final String EXPECTED_EXECUTION_TIME = "ConsentManagement.PaymentRestrictions.CutOffDateTime" + - ".ExpectedExecutionTime"; - public static final String EXPECTED_SETTLEMENT_TIME = "ConsentManagement.PaymentRestrictions.CutOffDateTime" + - ".ExpectedSettlementTime"; - - // Realtime Event Notification Constants - public static final String REALTIME_EVENT_NOTIFICATION_ENABLED = "RealtimeEventNotification.Enable"; - public static final String PERIODIC_CRON_EXPRESSION = "RealtimeEventNotification.PeriodicCronExpression"; - public static final String TIMEOUT_IN_SECONDS = "RealtimeEventNotification.TimeoutInSeconds"; - public static final String MAX_RETRIES = "RealtimeEventNotification.MaxRetries"; - public static final String INITIAL_BACKOFF_TIME_IN_SECONDS - = "RealtimeEventNotification.InitialBackoffTimeInSeconds"; - public static final String BACKOFF_FUNCTION = "RealtimeEventNotification.BackoffFunction"; - public static final String CIRCUIT_BREAKER_OPEN_TIMEOUT_IN_SECONDS - = "RealtimeEventNotification.CircuitBreakerOpenTimeoutInSeconds"; - public static final String EVENT_NOTIFICATION_THREADPOOL_SIZE - = "RealtimeEventNotification.EventNotificationThreadPoolSize"; - public static final String REALTIME_EVENT_NOTIFICATION_REQUEST_GENERATOR - = "RealtimeEventNotification.RequestGenerator"; - public static final String CONTENT_TYPE_TAG = "Content-Type"; - public static final String JSON_CONTENT_TYPE = "application/json"; - public static final String SP_API_PATH = "/stores/query"; - public static final String APP_NAME_CC = "appName"; - public static final String QUERY = "query"; - public static final String IS_PSU_FEDERATED = "PSUFederatedAuthentication.Enabled"; - public static final String PSU_FEDERATED_IDP_NAME = "PSUFederatedAuthentication.IDPName"; - public static final String IDEMPOTENCY_IS_ENABLED = "Consent.Idempotency.Enabled"; - public static final String IDEMPOTENCY_ALLOWED_TIME = "Consent.Idempotency.AllowedTimeDuration"; - public static final String DOT_SEPARATOR = "."; -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/error/OpenBankingErrorCodes.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/error/OpenBankingErrorCodes.java deleted file mode 100644 index 1588349b..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/error/OpenBankingErrorCodes.java +++ /dev/null @@ -1,60 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.error; - -/** - * Class containing the error codes for Open Banking. - */ -public class OpenBankingErrorCodes { - - public static final String BAD_REQUEST_CODE = "400"; - public static final String UNAUTHORIZED_CODE = "401"; - public static final String FORBIDDEN_CODE = "403"; - public static final String NOT_FOUND_CODE = "404"; - public static final String NOT_ALLOWED_CODE = "405"; - public static final String NOT_ACCEPTABLE_CODE = "406"; - public static final String UNSUPPORTED_MEDIA_TYPE_CODE = "415"; - public static final String SERVER_ERROR_CODE = "500"; - - public static final String INVALID_GRANT_TYPE_CODE = "200001"; - public static final String CONSENT_VALIDATION_REQUEST_FAILURE = "200002"; - public static final String INVALID_MTLS_CERT_CODE = "200003"; - public static final String TPP_VALIDATION_FAILED_CODE = "200004"; - public static final String INVALID_SIGNATURE = "200005"; - public static final String SCP_USER_VALIDATION_FAILED_CODE = "200006"; - public static final String MISSING_MTLS_CERT_CODE = "200007"; - public static final String EXPIRED_MTLS_CERT_CODE = "200008"; - public static final String REVOKED_MTLS_CERT_CODE = "200009"; - public static final String INVALID_SIGNATURE_CODE = "200010"; - public static final String MISSING_CONTENT_TYPE = "200011"; - public static final String INVALID_CONTENT_TYPE = "200012"; - public static final String MISSING_REQUEST_PAYLOAD = "200013"; - public static final String INVALID_CHARS_IN_HEADER_ERROR = "200014"; - public static final String MISSING_HEADER_PARAM_CLIENT_ID = "200015"; - public static final String ERROR_IN_EVENT_POLLING_REQUEST = "200016"; - - // Error titles - public static final String UNSUPPORTED_MEDIA_TYPE = "Unsupported Media Type"; - - public static final String REGISTRATION_INTERNAL_ERROR = "Error occurred while registering application"; - public static final String REGISTATION_DELETE_ERROR = "Error occurred while deleting application"; - public static final String REGISTRATION_UPDATE_ERROR = "Error occurred while updating application"; - - public static final String EXECUTOR_JWS_SIGNATURE_NOT_FOUND = "Error occurred in JWS Executor"; - public static final String JWS_SIGNATURE_HANDLE_ERROR = "Error occurred while validating JWS Signature"; -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/DefaultOBEventExecutor.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/DefaultOBEventExecutor.java deleted file mode 100644 index 0c2540a3..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/DefaultOBEventExecutor.java +++ /dev/null @@ -1,33 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.event.executor; - -import com.wso2.openbanking.accelerator.common.event.executor.model.OBEvent; - -/** - * Open banking event executor default implementation. - */ -public class DefaultOBEventExecutor implements OBEventExecutor { - - @Override - public void processEvent(OBEvent obEvent) { - - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/OBEventExecutor.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/OBEventExecutor.java deleted file mode 100644 index d2175cdb..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/OBEventExecutor.java +++ /dev/null @@ -1,34 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.event.executor; - -import com.wso2.openbanking.accelerator.common.event.executor.model.OBEvent; - -/** - * Open banking event executor interface. - */ -public interface OBEventExecutor { - - /** - * This method is used to process events. - * - * @param obEvent OBEvent which holds event related data - */ - public void processEvent(OBEvent obEvent); -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/OBEventQueue.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/OBEventQueue.java deleted file mode 100644 index 5e45ffd8..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/OBEventQueue.java +++ /dev/null @@ -1,66 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.event.executor; - -import com.wso2.openbanking.accelerator.common.event.executor.model.OBEvent; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.concurrent.ArrayBlockingQueue; -import java.util.concurrent.BlockingQueue; -import java.util.concurrent.ExecutorService; -import java.util.concurrent.Executors; -import java.util.concurrent.RejectedExecutionException; - - -/** - * Open Banking event queue wrapper class wrapping the ArrayBlockingQueue. - */ -public class OBEventQueue { - - private static final Log log = LogFactory.getLog(OBEventQueue.class); - private final BlockingQueue eventQueue; - private final ExecutorService executorService; - - public OBEventQueue(int queueSize, int workerThreadCount) { - - // Note : Using a fixed worker thread pool and a bounded queue to control the load on the server - executorService = Executors.newFixedThreadPool(workerThreadCount); - eventQueue = new ArrayBlockingQueue<>(queueSize); - } - - public void put(OBEvent obEvent) { - - try { - if (eventQueue.offer(obEvent)) { - executorService.submit(new OBQueueWorker(eventQueue, executorService)); - } else { - log.error("Event queue is full. Starting to drop events."); - } - } catch (RejectedExecutionException e) { - log.warn("Task submission failed. Task queue might be full", e); - } - } - - @Override - protected void finalize() throws Throwable { - executorService.shutdown(); - super.finalize(); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/OBQueueWorker.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/OBQueueWorker.java deleted file mode 100644 index 09aa7f45..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/OBQueueWorker.java +++ /dev/null @@ -1,69 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.event.executor; - -import com.wso2.openbanking.accelerator.common.event.executor.model.OBEvent; -import com.wso2.openbanking.accelerator.common.internal.OpenBankingCommonDataHolder; -import com.wso2.openbanking.accelerator.common.util.OpenBankingUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.List; -import java.util.Map; -import java.util.concurrent.BlockingQueue; -import java.util.concurrent.ExecutorService; -import java.util.concurrent.ThreadPoolExecutor; -import java.util.stream.Collectors; - -/** - * Open Banking Queue worker implementation to execute events in queue. - */ -public class OBQueueWorker implements Runnable { - - private BlockingQueue eventQueue; - private ExecutorService executorService; - private static final Log log = LogFactory.getLog(OBQueueWorker.class); - - public OBQueueWorker(BlockingQueue queue, ExecutorService executorService) { - - this.eventQueue = queue; - this.executorService = executorService; - } - - @Override - public void run() { - - ThreadPoolExecutor threadPoolExecutor = ((ThreadPoolExecutor) executorService); - - do { - OBEvent event = eventQueue.poll(); - if (event != null) { - Map obEventExecutors = OpenBankingCommonDataHolder.getInstance().getOBEventExecutors(); - List executorList = obEventExecutors.keySet().stream() - .map(integer -> (OBEventExecutor) OpenBankingUtils - .getClassInstanceFromFQN(obEventExecutors.get(integer))).collect(Collectors.toList()); - for (OBEventExecutor obEventExecutor : executorList) { - obEventExecutor.processEvent(event); - } - } else { - log.error("OB Event is null"); - } - } while (threadPoolExecutor.getActiveCount() == 1 && eventQueue.size() != 0); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/model/OBEvent.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/model/OBEvent.java deleted file mode 100644 index 30db41b0..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/event/executor/model/OBEvent.java +++ /dev/null @@ -1,46 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.event.executor.model; - -import java.util.Map; - -/** - * Open Banking event model class. - */ -public class OBEvent { - - private String eventType; - private Map eventData; - - public OBEvent(String eventType, Map eventData) { - - this.eventType = eventType; - this.eventData = eventData; - } - - public String getEventType() { - - return eventType; - } - - public Map getEventData() { - - return eventData; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/CertificateValidationException.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/CertificateValidationException.java deleted file mode 100644 index 4f13b630..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/CertificateValidationException.java +++ /dev/null @@ -1,83 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.exception; - -import com.wso2.openbanking.accelerator.common.error.OpenBankingErrorCodes; - -/** - * Certificate Validation exception class. - */ -public class CertificateValidationException extends Exception { - - private String errorCode; - private String errorPayload; - - public CertificateValidationException(String message) { - - super(message); - this.errorCode = OpenBankingErrorCodes.INVALID_MTLS_CERT_CODE; - this.errorPayload = ""; - } - - public CertificateValidationException(String message, String errorCode, String errorPayload) { - - super(message); - this.errorCode = errorCode; - this.errorPayload = errorPayload; - } - - public CertificateValidationException(String message, Throwable cause) { - - super(message, cause); - } - - public CertificateValidationException(Throwable cause, String errorCode, String errorPayload) { - - super(cause); - this.errorCode = errorCode; - this.errorPayload = errorPayload; - } - - public CertificateValidationException(String message, Throwable cause, String errorCode, String errorPayload) { - - super(message, cause); - this.errorCode = errorCode; - this.errorPayload = errorPayload; - } - - public String getErrorCode() { - - return errorCode; - } - - public void setErrorCode(String errorCode) { - - this.errorCode = errorCode; - } - - public String getErrorPayload() { - - return errorPayload; - } - - public void setErrorPayload(String errorPayload) { - - this.errorPayload = errorPayload; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/ConsentManagementException.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/ConsentManagementException.java deleted file mode 100644 index 308b762d..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/ConsentManagementException.java +++ /dev/null @@ -1,33 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.exception; - -/** - * Used for handling exceptions in consent management component. - */ -public class ConsentManagementException extends OpenBankingException { - - public ConsentManagementException(String message) { - super(message); - } - - public ConsentManagementException(String message, Throwable e) { - super(message, e); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/ConsentManagementRuntimeException.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/ConsentManagementRuntimeException.java deleted file mode 100644 index b43ba5b6..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/ConsentManagementRuntimeException.java +++ /dev/null @@ -1,35 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.exception; - -/** - * Used for runtime exceptions in consent management component. - */ -public class ConsentManagementRuntimeException extends OpenBankingRuntimeException { - - public ConsentManagementRuntimeException(String errorCode, Throwable cause) { - - super(errorCode, cause); - } - - public ConsentManagementRuntimeException(String errorCode) { - - super(errorCode); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/OBThrottlerException.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/OBThrottlerException.java deleted file mode 100644 index 78d0485e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/OBThrottlerException.java +++ /dev/null @@ -1,34 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.exception; - -/** - * Used for handling exceptions in ob throttler component. - */ -public class OBThrottlerException extends OpenBankingException { - - public OBThrottlerException(String message) { - - super(message); - } - - public OBThrottlerException(String message, Throwable e) { - - super(message, e); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/OpenBankingException.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/OpenBankingException.java deleted file mode 100644 index 2dbf5362..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/OpenBankingException.java +++ /dev/null @@ -1,34 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.exception; - -/** - * Used for exceptions in Open Banking components. - */ -public class OpenBankingException extends Exception { - - public OpenBankingException(String message) { - super(message); - } - - public OpenBankingException(String message, Throwable e) { - super(message, e); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/OpenBankingRuntimeException.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/OpenBankingRuntimeException.java deleted file mode 100644 index ad1effc2..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/OpenBankingRuntimeException.java +++ /dev/null @@ -1,50 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.exception; - -/** - * Used for creating runtime exceptions for Open-banking modules. - */ -public class OpenBankingRuntimeException extends RuntimeException { - - private static final long serialVersionUID = -5686395831712095972L; - private String errorCode; - - public OpenBankingRuntimeException(String errorCode, Throwable cause) { - - super(cause); - this.errorCode = errorCode; - } - - public OpenBankingRuntimeException(String errorCode) { - super(); - this.errorCode = errorCode; - } - - public String getErrorCode() { - - return errorCode; - } - - public void setErrorCode(String errorCode) { - - this.errorCode = errorCode; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/TPPValidationException.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/TPPValidationException.java deleted file mode 100644 index bca1a9a5..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/exception/TPPValidationException.java +++ /dev/null @@ -1,33 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.exception; - -/** - * TPPValidationException class. - */ -public class TPPValidationException extends Exception { - - public TPPValidationException(String message) { - super(message); - } - - public TPPValidationException(String message, Throwable e) { - super(message, e); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/ApplicationIdentityService.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/ApplicationIdentityService.java deleted file mode 100644 index 8dd7171a..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/ApplicationIdentityService.java +++ /dev/null @@ -1,66 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.identity; - -import com.nimbusds.jose.jwk.JWKSet; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.identity.cache.JWKSetCache; -import com.wso2.openbanking.accelerator.common.identity.cache.JWKSetCacheKey; -import com.wso2.openbanking.accelerator.common.identity.retriever.JWKRetriever; -import org.apache.commons.lang.StringUtils; - -import java.net.URL; - -/** - * Class to handle retrieving JWKSet from jwksUri. - */ -public class ApplicationIdentityService { - - /** - * Get JWKSet for application. - * First checks to get from cache, else retrieve the JWKSet from the URL by calling - * a method in JWKRetriever - * @param applicationName Application Name - * @param jwksUrl URL of the JWKSet - * @param useCache Use cache or not - * @return JWKSet - * @throws OpenBankingException if an error occurs while retrieving the JWKSet - */ - public JWKSet getPublicJWKSet(String applicationName, URL jwksUrl, - boolean useCache) throws OpenBankingException { - - if (StringUtils.isEmpty(applicationName)) { - throw new OpenBankingException("Application Name is required"); - } - - // Get JWK Set - if (useCache) { - JWKSetCache jwkSetCache = new JWKSetCache(); - try { - return jwkSetCache.getFromCacheOrRetrieve(JWKSetCacheKey.of(applicationName), - () -> new JWKRetriever().updateJWKSetFromURL(jwksUrl)); - } catch (OpenBankingException e) { - throw new OpenBankingException(String.format("Unable to retrieve JWKSet for %s", - applicationName), e); - } - } else { - return new JWKRetriever().updateJWKSetFromURL(jwksUrl); - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/IdentityConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/IdentityConstants.java deleted file mode 100644 index 5f06d6c0..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/IdentityConstants.java +++ /dev/null @@ -1,90 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.identity; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; - -import java.util.Collections; -import java.util.EnumMap; -import java.util.Map; -import java.util.Optional; - -/** - * Constants required for Server Identity Retriever. - */ -public class IdentityConstants { - - public static final String PRODUCTION = "PRODUCTION"; - public static final String SANDBOX = "SANDBOX"; - - public static final String KEYSTORE_LOCATION_CONF_KEY = "Security.KeyStore.Location"; - public static final String KEYSTORE_PASS_CONF_KEY = "Security.KeyStore.Password"; - - /** - * CertificateType enum. - */ - public enum CertificateType { - TRANSPORT, SIGNING - } - - /** - * EnvironmentType enum. - */ - public enum EnvironmentType { - SANDBOX, PRODUCTION, DEFAULT - } - - /** - * Use values for JWKS key set retrieval. - * - * Default values defined by specification - * @see RFC7517 Key Use Values - */ - public static final Map USE_TYPE_VALUE_MAP; - - static { - Map useMap = new EnumMap<>(CertificateType.class); - - useMap.put(CertificateType.SIGNING, new String[]{"sig"}); - useMap.put(CertificateType.TRANSPORT, new String[]{"enc", "tls"}); - - USE_TYPE_VALUE_MAP = Collections.unmodifiableMap(useMap); - } - - /** - * Custom Configurations. - * defines and loads custom configurations from xml. - */ - public static final Optional PRIMARY_SIGNING_CERT_ALIAS; - public static final Optional SANDBOX_SIGNING_CERT_ALIAS; - public static final Optional PRIMARY_SIGNING_CERT_KID; - public static final Optional SANDBOX_SIGNING_CERT_KID; - - static { - - PRIMARY_SIGNING_CERT_ALIAS = Optional - .ofNullable(OpenBankingConfigParser.getInstance().getOBIdnRetrieverSigningCertificateAlias()); - SANDBOX_SIGNING_CERT_ALIAS = Optional - .ofNullable(OpenBankingConfigParser.getInstance().getOBIdnRetrieverSandboxSigningCertificateAlias()); - PRIMARY_SIGNING_CERT_KID = Optional - .ofNullable(OpenBankingConfigParser.getInstance().getOBIdnRetrieverSigningCertificateKid()); - SANDBOX_SIGNING_CERT_KID = Optional - .ofNullable(OpenBankingConfigParser.getInstance().getOBIdnRetrieverSandboxCertificateKid()); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/cache/JWKSetCache.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/cache/JWKSetCache.java deleted file mode 100644 index 0be4a853..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/cache/JWKSetCache.java +++ /dev/null @@ -1,42 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.identity.cache; - -import com.nimbusds.jose.jwk.JWKSet; -import com.wso2.openbanking.accelerator.common.identity.cache.base.OpenBankingIdentityBaseCache; - -/** - * Cache Manager for Public Certificates. - */ -public class JWKSetCache extends OpenBankingIdentityBaseCache { - - private static final String DEFAULT_CACHE_NAME = "OB_IDN_JWKS_CACHE"; - - public JWKSetCache() { - - super(DEFAULT_CACHE_NAME); - } - - public JWKSetCache(String cacheName) { - - super(cacheName); - } - -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/cache/JWKSetCacheKey.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/cache/JWKSetCacheKey.java deleted file mode 100644 index 87d18af7..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/cache/JWKSetCacheKey.java +++ /dev/null @@ -1,63 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.identity.cache; - -import com.wso2.openbanking.accelerator.common.caching.OpenBankingBaseCacheKey; - -import java.io.Serializable; -import java.util.Objects; - -/** - * Cache Key for JWKSet cache. - */ -public class JWKSetCacheKey extends OpenBankingBaseCacheKey implements Serializable { - - static final long serialVersionUID = 42L; - private String applicationName; - - public JWKSetCacheKey(String applicationName) { - - this.applicationName = applicationName; - } - - public static JWKSetCacheKey of(String applicationName) { - - return new JWKSetCacheKey(applicationName); - } - - @Override - public boolean equals(Object o) { - - if (this == o) { - return true; - } - if (o == null || getClass() != o.getClass()) { - return false; - } - JWKSetCacheKey that = (JWKSetCacheKey) o; - return Objects.equals(applicationName, that.applicationName); - } - - @Override - public int hashCode() { - - return Objects.hash(applicationName); - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/cache/base/OpenBankingIdentityBaseCache.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/cache/base/OpenBankingIdentityBaseCache.java deleted file mode 100644 index 2f8776da..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/cache/base/OpenBankingIdentityBaseCache.java +++ /dev/null @@ -1,77 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.identity.cache.base; - -import com.wso2.openbanking.accelerator.common.caching.OpenBankingBaseCache; -import com.wso2.openbanking.accelerator.common.caching.OpenBankingBaseCacheKey; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.internal.OpenBankingCommonDataHolder; - -/** - * Cache definition to store objects in open banking iam component implementations. - * @param Extended Cache key - * @param Cache value - */ -public class OpenBankingIdentityBaseCache extends OpenBankingBaseCache { - - private static final String cacheName = "OPEN_BANKING_IDENTITY_CACHE"; - - private Integer accessExpiryMinutes; - private Integer modifiedExpiryMinutes; - - /** - * Initialize with unique cache name. - * @param cacheName Unique cache name - */ - public OpenBankingIdentityBaseCache(String cacheName) { - - super(cacheName); - this.accessExpiryMinutes = setAccessExpiryMinutes(); - this.modifiedExpiryMinutes = setModifiedExpiryMinutes(); - } - - @Override - public int getCacheAccessExpiryMinutes() { - return accessExpiryMinutes; - } - - @Override - public int getCacheModifiedExpiryMinutes() { - return modifiedExpiryMinutes; - } - - public int setAccessExpiryMinutes() { - - return OpenBankingCommonDataHolder.getInstance().getCommonCacheAccessExpiry(); - } - - public int setModifiedExpiryMinutes() { - - return OpenBankingCommonDataHolder.getInstance().getCommonCacheModifiedExpiry(); - } - - public V getFromCacheOrRetrieve(K key, OnDemandRetriever onDemandRetriever) throws OpenBankingException { - - try { - return super.getFromCacheOrRetrieve(key, onDemandRetriever); - } catch (OpenBankingException e) { - throw new OpenBankingException("Unable to retrieve from cache", e); - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/retriever/JWKRetriever.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/retriever/JWKRetriever.java deleted file mode 100644 index f0eff1db..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/retriever/JWKRetriever.java +++ /dev/null @@ -1,127 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.identity.retriever; - -import com.nimbusds.jose.jwk.JWKSet; -import com.nimbusds.jose.util.DefaultResourceRetriever; -import com.nimbusds.jose.util.Resource; -import com.nimbusds.jose.util.ResourceRetriever; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.identity.ApplicationIdentityService; -import com.wso2.openbanking.accelerator.common.identity.cache.JWKSetCache; -import com.wso2.openbanking.accelerator.common.identity.cache.JWKSetCacheKey; - -import java.io.IOException; -import java.net.URL; -import java.text.ParseException; - -/** - * Retrieve JWK set using nimbus retriever. - */ -public class JWKRetriever { - - private volatile JWKRetriever instance = null; - - private static final int jwksSizeLimit; - private static final int jwksConnectionTimeout; - private static final int jwksReadTimeout; - - /** - * The JWK set retriever. - */ - private static final ResourceRetriever resourceRetriever; - - static { - - jwksSizeLimit = Integer.parseInt(OpenBankingConfigParser.getInstance().getJwksRetrieverSizeLimit()); - jwksConnectionTimeout = Integer.parseInt(OpenBankingConfigParser.getInstance() - .getJwksRetrieverConnectionTimeout()); - jwksReadTimeout = Integer.parseInt(OpenBankingConfigParser.getInstance().getJwksRetrieverReadTimeout()); - resourceRetriever = new DefaultResourceRetriever(jwksReadTimeout, jwksConnectionTimeout, jwksSizeLimit); - } - - /** - * Get instance of JWKRetriever. - * - * @return JWKRetriever instance - */ - public JWKRetriever getInstance() { - - if (instance == null) { - synchronized (this) { - if (instance == null) { - instance = new JWKRetriever(); - } - } - } - return instance; - } - - /** - * Get JWK Set from remote resource retriever. - * - * @param jwksURL jwksURL in URL format - * @return JWKSet - * @throws OpenBankingException if an error occurs while retrieving resource - */ - public JWKSet updateJWKSetFromURL(URL jwksURL) throws OpenBankingException { - - JWKSet jwkSet; - Resource res; - try { - res = resourceRetriever.retrieveResource(jwksURL); - } catch (IOException e) { - throw new OpenBankingException("Couldn't retrieve remote JWK set: " + e.getMessage(), e); - } - try { - jwkSet = JWKSet.parse(res.getContent()); - } catch (ParseException e) { - throw new OpenBankingException("Couldn't parse remote JWK set: " + e.getMessage(), e); - } - - return jwkSet; - } - - /** - * Get JWK Set from cache or retrieve from onDemand retriever. - * - * @param jwksURL jwksURL in URL format - * @param applicationName application name as a string - * @return jwkSet - * @throws OpenBankingException if an error occurs while getting JWK set - */ - public JWKSet getJWKSet(URL jwksURL , String applicationName) throws OpenBankingException { - - try { - JWKSetCache jwkSetCache = new JWKSetCache(); - ApplicationIdentityService applicationIdentityService = new ApplicationIdentityService(); - JWKSet jwkSet = jwkSetCache.getFromCacheOrRetrieve(JWKSetCacheKey.of(applicationName), () - ->applicationIdentityService.getPublicJWKSet( - applicationName, jwksURL, true)); - - if (jwkSet == null) { - jwkSet = updateJWKSetFromURL(jwksURL); - } - return jwkSet; - } catch (OpenBankingException e) { - throw new OpenBankingException("Couldn't get remote JWK set: " + e.getMessage(), e); - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/retriever/ServerIdentityRetriever.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/retriever/ServerIdentityRetriever.java deleted file mode 100644 index 59d3f3b6..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/retriever/ServerIdentityRetriever.java +++ /dev/null @@ -1,153 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.identity.retriever; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.identity.IdentityConstants; -import com.wso2.openbanking.accelerator.common.util.HTTPClientUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.base.ServerConfiguration; - -import java.security.Key; -import java.security.KeyStore; -import java.security.KeyStoreException; -import java.security.NoSuchAlgorithmException; -import java.security.UnrecoverableKeyException; -import java.security.cert.Certificate; -import java.util.Optional; - -/** - * Utility to retrieve ASPSP certificates. - */ -public class ServerIdentityRetriever { - - private static KeyStore keyStore = null; - // Internal KeyStore Password. - private static char[] keyStorePassword; - - private static final Log log = LogFactory.getLog(ServerIdentityRetriever.class); - - static { - // Static Initialize Internal Keystore. - String keyStoreLocation = ServerConfiguration.getInstance() - .getFirstProperty(IdentityConstants.KEYSTORE_LOCATION_CONF_KEY); - String keyStorePassword = ServerConfiguration.getInstance() - .getFirstProperty(IdentityConstants.KEYSTORE_PASS_CONF_KEY); - - try { - ServerIdentityRetriever.keyStore = HTTPClientUtils.loadKeyStore(keyStoreLocation, keyStorePassword); - ServerIdentityRetriever.keyStorePassword = keyStorePassword.toCharArray(); - } catch (OpenBankingException e) { - log.error("Unable to load InternalKeyStore", e); - } - } - - /** - * Returns the signing key using the signing Certificate. - * @param certificateType Signing certificate - * @param environmentType Sandbox or Production environment - * @return Key The signing key - * @throws OpenBankingException throws at KeyStoreException, NoSuchAlgorithmException, UnrecoverableKeyException - */ - public static Optional getPrimaryCertificate(IdentityConstants.CertificateType certificateType, - IdentityConstants.EnvironmentType environmentType) - throws OpenBankingException { - Optional certAlias; - - if (certificateType.equals(IdentityConstants.CertificateType.SIGNING)) { - - certAlias = getCertAlias(certificateType, environmentType); - - if (certAlias.isPresent()) { - try { - // The requested key, or - // null if the given alias does not exist or does not identify a key-related entry. - return Optional.of(keyStore.getKey(certAlias.get(), keyStorePassword)); - } catch (KeyStoreException | NoSuchAlgorithmException | UnrecoverableKeyException e) { - throw new OpenBankingException("Unable to retrieve certificate", e); - } - } - - } - return Optional.empty(); - } - - /** - * Returns signing key used at production environment. - * @param certificateType signing certificate - * @return Key signing key - * @throws OpenBankingException throws OpenBankingException - */ - public static Optional getPrimaryCertificate(IdentityConstants.CertificateType certificateType) - throws OpenBankingException { - - return getPrimaryCertificate(certificateType, IdentityConstants.EnvironmentType.PRODUCTION); - } - - /** - * Get certificate from keystore with the given alias. - * Used in toolkits to get public signing certificate from keystore to retrieve the issuer. - * - * @param alias alias of the signing certificate of the bank - * @return signing certificate - * @throws KeyStoreException throw a generic KeyStore exception - */ - public static Certificate getCertificate(String alias) throws KeyStoreException { - - return keyStore.getCertificate(alias); - } - - /** - * Returns Signing certificate alias at Production environment. - * @param certificateType Signing - * @return String Certificate alias - * @throws OpenBankingException when there is an exception while retrieving the alias - */ - public static Optional getCertAlias(IdentityConstants.CertificateType certificateType) - throws OpenBankingException { - return getCertAlias(certificateType, IdentityConstants.EnvironmentType.PRODUCTION); - } - - /** - * Returns Signing certificate alias. - * @param certificateType signing - * @param environmentType Production or Sandbox - * @return Signing certificate alias - * @throws OpenBankingException throws OpenBankingException - */ - public static Optional getCertAlias(IdentityConstants.CertificateType certificateType, - IdentityConstants.EnvironmentType environmentType) - throws OpenBankingException { - Optional certAlias = Optional.empty(); - - if (certificateType.equals(IdentityConstants.CertificateType.SIGNING)) { - if (keyStore == null) { - throw new OpenBankingException("Internal Key Store not initialized"); - } - - if (environmentType == IdentityConstants.EnvironmentType.SANDBOX) { - certAlias = IdentityConstants.SANDBOX_SIGNING_CERT_ALIAS; - } else { - certAlias = IdentityConstants.PRIMARY_SIGNING_CERT_ALIAS; - } - } - return certAlias; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/retriever/sp/CommonServiceProviderRetriever.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/retriever/sp/CommonServiceProviderRetriever.java deleted file mode 100644 index 0281130d..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/identity/retriever/sp/CommonServiceProviderRetriever.java +++ /dev/null @@ -1,82 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.identity.retriever.sp; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.internal.OpenBankingCommonDataHolder; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.common.util.ServiceProviderUtils; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.application.common.IdentityApplicationManagementException; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.application.common.model.ServiceProviderProperty; -import org.wso2.carbon.identity.application.common.util.IdentityApplicationConstants; - -import java.util.Arrays; -import java.util.Optional; -import java.util.stream.Collectors; - -/** - * Class to retrieve Service Provider Data. - */ -public class CommonServiceProviderRetriever { - - private static final Log log = LogFactory.getLog(CommonServiceProviderRetriever.class); - - /** - * Utility method get the application property from SP Meta Data. - * - * @param clientId ClientId of the application - * @param property Property of the application - * @return the property value from SP metadata - * @throws OpenBankingException if an error occurs while retrieving the property - */ - @Generated(message = "Excluding from code coverage since it requires a service call") - public String getAppPropertyFromSPMetaData(String clientId, String property) throws OpenBankingException { - - String spProperty = null; - - if (StringUtils.isNotEmpty(clientId)) { - Optional serviceProvider; - try { - serviceProvider = Optional.ofNullable(OpenBankingCommonDataHolder.getInstance() - .getApplicationManagementService().getServiceProviderByClientId(clientId, - IdentityApplicationConstants.OAuth2.NAME, - ServiceProviderUtils.getSpTenantDomain(clientId))); - if (serviceProvider.isPresent()) { - spProperty = Arrays.stream(serviceProvider.get().getSpProperties()) - .collect(Collectors.toMap(ServiceProviderProperty::getName, - ServiceProviderProperty::getValue)).get(property); - } - } catch (IdentityApplicationManagementException e) { - log.error(String.format("Error occurred while retrieving OAuth2 application data for clientId %s", - clientId.replaceAll("[\r\n]" , "")) , e); - throw new OpenBankingException("Error occurred while retrieving OAuth2 application data for clientId" - , e); - } - } else { - log.error("Client id not found"); - throw new OpenBankingException("Client id not found"); - } - - return spProperty; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/internal/OpenBankingCommonDataHolder.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/internal/OpenBankingCommonDataHolder.java deleted file mode 100644 index 44dfd82f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/internal/OpenBankingCommonDataHolder.java +++ /dev/null @@ -1,126 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.internal; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.event.executor.OBEventQueue; -import org.wso2.carbon.identity.application.mgt.ApplicationManagementService; - -import java.util.Map; - -/** - * Data holder for Open Banking Common module. - */ -public class OpenBankingCommonDataHolder { - - private static volatile OpenBankingCommonDataHolder instance; - private OBEventQueue obEventQueue; - private Map obEventExecutors; - private ApplicationManagementService applicationManagementService; - private int commonCacheAccessExpiry; - private int commonCacheModifiedExpiry; - - private OpenBankingCommonDataHolder() { - - int queueSize = Integer.parseInt((String) OpenBankingConfigParser.getInstance().getConfiguration() - .get(OpenBankingConstants.EVENT_QUEUE_SIZE)); - int workerThreadCount = - Integer.parseInt((String) OpenBankingConfigParser.getInstance().getConfiguration() - .get(OpenBankingConstants.EVENT_WORKER_THREAD_COUNT)); - obEventQueue = new OBEventQueue(queueSize, workerThreadCount); - obEventExecutors = OpenBankingConfigParser.getInstance().getOpenBankingEventExecutors(); - setCommonCacheAccessExpiry((String) OpenBankingConfigParser.getInstance().getConfiguration() - .get(OpenBankingConstants.COMMON_IDENTITY_CACHE_ACCESS_EXPIRY)); - setCommonCacheModifiedExpiry((String) OpenBankingConfigParser.getInstance().getConfiguration() - .get(OpenBankingConstants.COMMON_IDENTITY_CACHE_MODIFY_EXPIRY)); - } - - public static OpenBankingCommonDataHolder getInstance() { - - if (instance == null) { - synchronized (OpenBankingCommonDataHolder.class) { - if (instance == null) { - instance = new OpenBankingCommonDataHolder(); - } - } - } - return instance; - } - - public Map getOBEventExecutors() { - - return obEventExecutors; - } - - public void setOBEventExecutor(Map obEventExecutors) { - - this.obEventExecutors = obEventExecutors; - } - - public OBEventQueue getOBEventQueue() { - - return obEventQueue; - } - - public void setOBEventQueue(OBEventQueue obEventQueue) { - - this.obEventQueue = obEventQueue; - } - - /** - * To get the instance of {@link ApplicationManagementService}. - * - * @return applicationManagementService - */ - public ApplicationManagementService getApplicationManagementService() { - - return applicationManagementService; - } - - /** - * To set the ApplicationManagementService. - * - * @param applicationManagementService instance of {@link ApplicationManagementService} - */ - public void setApplicationManagementService(ApplicationManagementService applicationManagementService) { - - this.applicationManagementService = applicationManagementService; - } - - public int getCommonCacheAccessExpiry() { - - return commonCacheAccessExpiry; - } - - public void setCommonCacheAccessExpiry(String expTime) { - - this.commonCacheAccessExpiry = expTime == null ? 60 : Integer.parseInt(expTime); - } - - public int getCommonCacheModifiedExpiry() { - - return commonCacheModifiedExpiry; - } - - public void setCommonCacheModifiedExpiry(String expTime) { - - this.commonCacheModifiedExpiry = expTime == null ? 60 : Integer.parseInt(expTime); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/internal/OpenBankingCommonServiceComponent.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/internal/OpenBankingCommonServiceComponent.java deleted file mode 100644 index d34bc661..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/internal/OpenBankingCommonServiceComponent.java +++ /dev/null @@ -1,84 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.internal; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationServiceImpl; -import com.wso2.openbanking.accelerator.common.event.executor.OBEventQueue; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.osgi.service.component.ComponentContext; -import org.osgi.service.component.annotations.Activate; -import org.osgi.service.component.annotations.Component; -import org.osgi.service.component.annotations.Deactivate; -import org.osgi.service.component.annotations.Reference; -import org.osgi.service.component.annotations.ReferenceCardinality; -import org.osgi.service.component.annotations.ReferencePolicy; -import org.wso2.carbon.identity.application.mgt.ApplicationManagementService; - -/** - * Method to register Open Banking common OSGi Services. - */ -@Component( - name = "com.wso2.open.banking.common", - immediate = true -) -public class OpenBankingCommonServiceComponent { - - private static final Log log = LogFactory.getLog(OpenBankingCommonServiceComponent.class); - - @Activate - protected void activate(ComponentContext context) { - - OpenBankingConfigurationService openBankingConfigurationService - = new OpenBankingConfigurationServiceImpl(); - OpenBankingCommonDataHolder openBankingCommonDataHolder = OpenBankingCommonDataHolder.getInstance(); - context.getBundleContext().registerService(OpenBankingConfigurationService.class.getName(), - openBankingConfigurationService, null); - context.getBundleContext().registerService(OBEventQueue.class.getName(), - openBankingCommonDataHolder.getOBEventQueue(), null); - context.getBundleContext().registerService(ApplicationManagementService.class, - ApplicationManagementService.getInstance(), null); - - log.debug("Open banking common component is activated successfully"); - } - - @Reference( - name = "ApplicationManagementService", - service = ApplicationManagementService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetApplicationManagementService" - ) - protected void setApplicationManagementService(ApplicationManagementService mgtService) { - - OpenBankingCommonDataHolder.getInstance().setApplicationManagementService(mgtService); - } - - protected void unsetApplicationManagementService(ApplicationManagementService mgtService) { - - OpenBankingCommonDataHolder.getInstance().setApplicationManagementService(null); - } - - @Deactivate - protected void deactivate(ComponentContext context) { - - log.debug("Open banking common component is deactivated"); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/model/PSD2RoleEnum.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/model/PSD2RoleEnum.java deleted file mode 100644 index 441dacef..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/model/PSD2RoleEnum.java +++ /dev/null @@ -1,49 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.model; - -/** - * PSD2 role enum class. - */ -public enum PSD2RoleEnum { - - AISP("aisp"), PISP("pisp"), CBPII("cbpii"), ASPSP("aspsp"); - - private String value; - - PSD2RoleEnum(String value) { - - this.value = value; - } - - public String toString() { - - return value; - } - - public static PSD2RoleEnum fromValue(String text) { - - for (PSD2RoleEnum apiTypeEnum : PSD2RoleEnum.values()) { - if (String.valueOf(apiTypeEnum.value).equalsIgnoreCase(text)) { - return apiTypeEnum; - } - } - return null; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/persistence/JDBCPersistenceManager.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/persistence/JDBCPersistenceManager.java deleted file mode 100644 index 2f947f0c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/persistence/JDBCPersistenceManager.java +++ /dev/null @@ -1,160 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.persistence; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementRuntimeException; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.SQLException; - -import javax.naming.Context; -import javax.naming.InitialContext; -import javax.naming.NamingException; -import javax.sql.DataSource; - -/** - * This class is used for handling Open banking consent data persistence in the JDBC Store. During the server - * start-up, it checks whether the database is created, if not it creates one. It reads the data source properties - * from the open-banking.xml. This is implemented as a singleton. An instance of this class can be obtained through - * JDBCPersistenceManager.getInstance() method. - */ -public class JDBCPersistenceManager { - - private static volatile JDBCPersistenceManager instance; - private static volatile DataSource dataSource; - private static Log log = LogFactory.getLog(JDBCPersistenceManager.class); - - private JDBCPersistenceManager() { - - initDataSource(); - } - - /** - * Get an instance of the JDBCPersistenceManager. It implements a double checked locking initialization. - * - * @return JDBCPersistenceManager instance - */ - public static synchronized JDBCPersistenceManager getInstance() { - if (instance == null) { - synchronized (JDBCPersistenceManager.class) { - if (instance == null) { - instance = new JDBCPersistenceManager(); - } - } - } - return instance; - } - - /** - * Initialize the data source. - */ - @SuppressFBWarnings("LDAP_INJECTION") - // Suppressed content - context.lookup(dataSourceName) - // Suppression reason - False Positive : Since the dataSourceName is taken from the deployment.toml, it can be - // trusted - // Suppressed warning count - 1 - private void initDataSource() { - - if (dataSource != null) { - return; - } - synchronized (JDBCPersistenceManager.class) { - try { - String dataSourceName = OpenBankingConfigParser.getInstance().getDataSourceName(); - if (StringUtils.isNotBlank(dataSourceName)) { - Context context = new InitialContext(); - dataSource = (DataSource) context.lookup(dataSourceName); - } else { - throw new ConsentManagementRuntimeException("Persistence Manager configuration for Open Banking " + - "is not available in open-banking.xml file. Terminating the JDBC persistence manager " + - "initialization."); - } - } catch (NamingException e) { - throw new ConsentManagementRuntimeException("Error when looking up the Consent Management Data Source.", - e); - } - } - } - - /** - * Returns an database connection for Consent Management data source. - * - * @return Database connection. - * @throws ConsentManagementRuntimeException Exception occurred when getting the data source. - */ - public Connection getDBConnection() throws ConsentManagementRuntimeException { - - try { - Connection dbConnection = dataSource.getConnection(); - dbConnection.setAutoCommit(false); - log.debug("Returning database connection for Consent Management data source"); - return dbConnection; - } catch (SQLException e) { - throw new ConsentManagementRuntimeException("Error when getting a database connection object from the " + - "consent management data source.", e); - } - } - - /** - * Returns Consent Management data source. - * - * @return Data source. - */ - public DataSource getDataSource() { - - return dataSource; - } - - /** - * Revoke the transaction when catch then sql transaction errors. - * - * @param dbConnection database connection. - */ - public void rollbackTransaction(Connection dbConnection) { - - try { - if (dbConnection != null) { - dbConnection.rollback(); - } - } catch (SQLException e) { - log.error("An error occurred while rolling back transactions. ", e); - } - } - - /** - * Commit the transaction. - * - * @param dbConnection database connection. - */ - public void commitTransaction(Connection dbConnection) { - - try { - if (dbConnection != null) { - dbConnection.commit(); - } - } catch (SQLException e) { - log.error("An error occurred while commit transactions. ", e); - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/persistence/JDBCRetentionDataPersistenceManager.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/persistence/JDBCRetentionDataPersistenceManager.java deleted file mode 100644 index 22060fc1..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/persistence/JDBCRetentionDataPersistenceManager.java +++ /dev/null @@ -1,161 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.persistence; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementRuntimeException; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.SQLException; - -import javax.naming.Context; -import javax.naming.InitialContext; -import javax.naming.NamingException; -import javax.sql.DataSource; - -/** - * This class is used for handling open banking retention data (if enabled) persistence in the JDBC Store. - * During the server start-up, it checks whether the database is created, It reads the data source properties - * from the open-banking.xml. This is implemented as a singleton. An instance of this class can be obtained through - * JDBCRetentionDataPersistenceManager.getInstance() method. - */ -public class JDBCRetentionDataPersistenceManager { - - private static volatile JDBCRetentionDataPersistenceManager instance; - private static volatile DataSource dataSource; - private static Log log = LogFactory.getLog(JDBCRetentionDataPersistenceManager.class); - - private JDBCRetentionDataPersistenceManager() { - - initDataSource(); - } - - /** - * Get an instance of the JDBCRetentionDataPersistenceManager. It implements a double checked locking initialization - * - * @return JDBCRetentionDataPersistenceManager instance - */ - public static synchronized JDBCRetentionDataPersistenceManager getInstance() { - - if (instance == null) { - synchronized (JDBCRetentionDataPersistenceManager.class) { - if (instance == null) { - instance = new JDBCRetentionDataPersistenceManager(); - } - } - } - return instance; - } - - /** - * Initialize the data source. - */ - @SuppressFBWarnings("LDAP_INJECTION") - // Suppressed content - context.lookup(dataSourceName) - // Suppression reason - False Positive : Since the dataSourceName is taken from the deployment.toml, it can be - // trusted - // Suppressed warning count - 1 - private void initDataSource() { - - if (dataSource != null) { - return; - } - synchronized (JDBCRetentionDataPersistenceManager.class) { - try { - String dataSourceName = OpenBankingConfigParser.getInstance().getRetentionDataSourceName(); - if (StringUtils.isNotBlank(dataSourceName)) { - Context context = new InitialContext(); - dataSource = (DataSource) context.lookup(dataSourceName); - } else { - throw new ConsentManagementRuntimeException("Persistence Manager configuration for " + - "retention datasource is not available in open-banking.xml file. Terminating the " + - "JDBC retention data persistence manager initialization."); - } - } catch (NamingException e) { - throw new ConsentManagementRuntimeException("Error when looking up the Consent Retention " + - "Data Source.", e); - } - } - } - - /** - * Returns an database connection for retention data source. - * - * @return Database connection. - * @throws ConsentManagementRuntimeException Exception occurred when getting the data source. - */ - public Connection getDBConnection() throws ConsentManagementRuntimeException { - - try { - Connection dbConnection = dataSource.getConnection(); - dbConnection.setAutoCommit(false); - log.debug("Returning database connection for retention data source"); - return dbConnection; - } catch (SQLException e) { - throw new ConsentManagementRuntimeException("Error when getting a database connection object from the " + - "retention data source.", e); - } - } - - /** - * Returns retention data source. - * - * @return Data source. - */ - public DataSource getDataSource() { - - return dataSource; - } - - /** - * Revoke the transaction when catch then sql transaction errors. - * - * @param dbConnection database connection. - */ - public void rollbackTransaction(Connection dbConnection) { - - try { - if (dbConnection != null) { - dbConnection.rollback(); - } - } catch (SQLException e) { - log.error("An error occurred while rolling back transactions. ", e); - } - } - - /** - * Commit the transaction. - * - * @param dbConnection database connection. - */ - public void commitTransaction(Connection dbConnection) { - - try { - if (dbConnection != null) { - dbConnection.commit(); - } - } catch (SQLException e) { - log.error("An error occurred while commit transactions. ", e); - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/AnalyticsLogsUtils.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/AnalyticsLogsUtils.java deleted file mode 100644 index 8a8c64de..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/AnalyticsLogsUtils.java +++ /dev/null @@ -1,59 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util; - -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.databind.ObjectMapper; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.Map; - -/** - * Open Banking common utility class to publish analytics logs. - */ -public class AnalyticsLogsUtils { - - private static final Log log = LogFactory.getLog(AnalyticsLogsUtils.class); - private static final String LOG_FORMAT = "Data Stream : %s , Data Stream Version : %s , Data : {\"payload\":%s}"; - private static final String DATA_PROCESSING_ERROR = "Error occurred while processing the analytics dataset"; - - /** - * Method to add analytics logs to the OB analytics log file. - * - * @param logFile Name of the logger which is used to log analytics data to the log file - * @param dataStream Name of the data stream to which the data belongs - * @param dataVersion Version of the data stream to which the data belongs - * @param analyticsData Data which belongs to the given data stream that needs to be logged via the given logger - * @throws OpenBankingException if an error occurs while processing the analytics data - */ - public static void addAnalyticsLogs(String logFile, String dataStream, String dataVersion, Map analyticsData) throws OpenBankingException { - Log customLog = LogFactory.getLog(logFile); - try { - customLog.info(String.format(LOG_FORMAT, dataStream, - dataVersion, new ObjectMapper().writeValueAsString(analyticsData))); - } catch (JsonProcessingException e) { - log.error(DATA_PROCESSING_ERROR); - throw new OpenBankingException(DATA_PROCESSING_ERROR, e); - } - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/CarbonUtils.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/CarbonUtils.java deleted file mode 100644 index d25f6461..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/CarbonUtils.java +++ /dev/null @@ -1,97 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util; - -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import org.wso2.carbon.base.ServerConfiguration; - -import java.io.File; - -/** - * Utility Class for WSO2 Carbon related functions. - */ -public class CarbonUtils { - - private static final String HTTPS = "https://"; - private static final String COLON = ":"; - - /** - * Method to obtain config directory of any carbon product. - * - * @return String indicating the location of conf directory. - */ - public static String getCarbonConfigDirPath() { - - String carbonConfigDirPath = System.getProperty("carbon.config.dir.path"); - if (carbonConfigDirPath == null) { - carbonConfigDirPath = System.getenv("CARBON_CONFIG_DIR_PATH"); - if (carbonConfigDirPath == null) { - return getCarbonHome() + File.separator + "repository" + File.separator + "conf"; - } - } - return carbonConfigDirPath; - } - - /** - * Method to obtain home location of the carbon product. - * - * @return String indicating the home location of conf directory. - */ - public static String getCarbonHome() { - - String carbonHome = System.getProperty(OpenBankingConstants.CARBON_HOME); - if (carbonHome == null) { - carbonHome = System.getenv("CARBON_HOME"); - System.setProperty(OpenBankingConstants.CARBON_HOME, carbonHome); - } - return carbonHome; - } - - /** - * Method to obtain hostname of the carbon product. - * - * @return String indicating the hostname of the server. - */ - @Generated(message = "Ignoring because ServerConfiguration cannot be mocked") - public static String getCarbonPort() { - - int offset = Integer.parseInt(ServerConfiguration.getInstance().getFirstProperty("Offset")); - return String.valueOf(9443 + offset); - } - - /** - * Method to obtain port of the carbon product. - * - * @return String indicating the port of the server. - */ - @Generated(message = "Ignoring because ServerConfiguration cannot be mocked") - public static String getCarbonHostname() { - return ServerConfiguration.getInstance().getFirstProperty("HostName"); - } - - /** - * Method to obtain server url of the carbon product. - * - * @return String indicating the url of the server. - */ - @Generated(message = "Ignoring because ServerConfiguration cannot be mocked") - public static String getCarbonServerUrl() { - return HTTPS + getCarbonHostname() + COLON + getCarbonPort(); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/CertificateUtils.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/CertificateUtils.java deleted file mode 100644 index 65906695..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/CertificateUtils.java +++ /dev/null @@ -1,105 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.io.ByteArrayInputStream; -import java.security.cert.CertificateException; -import java.security.cert.CertificateFactory; -import java.security.cert.X509Certificate; -import java.util.Base64; - -/** - * Common utilities related to Certificates. - */ -public class CertificateUtils { - - private static final Log log = LogFactory.getLog(CertificateUtils.class); - - private static final String BEGIN_CERT = "-----BEGIN CERTIFICATE-----"; - private static final String END_CERT = "-----END CERTIFICATE-----"; - private static final String X509_CERT_INSTANCE_NAME = "X.509"; - - /** - * Parse the certificate content. - * - * @param content the content to be passed - * @return the parsed certificate - * @throws OpenBankingException if an error occurs while parsing the certificate - */ - public static X509Certificate parseCertificate(String content) throws OpenBankingException { - - try { - if (StringUtils.isNotBlank(content)) { - // removing illegal base64 characters before decoding - content = removeIllegalBase64Characters(content); - byte[] bytes = Base64.getDecoder().decode(content); - - return (java.security.cert.X509Certificate) CertificateFactory.getInstance(X509_CERT_INSTANCE_NAME) - .generateCertificate(new ByteArrayInputStream(bytes)); - } - log.error("Certificate passed through the request is empty"); - return null; - } catch (CertificateException | IllegalArgumentException e) { - throw new OpenBankingException("Certificate passed through the request not valid", e); - } - } - - /** - * Remove illegal base64 characters from input string. - * - * @param value certificate as a string - * @return certificate without illegal base64 characters - */ - private static String removeIllegalBase64Characters(String value) { - if (value.contains(BEGIN_CERT) - && value.contains(END_CERT)) { - - // extracting certificate content - value = value.substring(value.indexOf(BEGIN_CERT) - + BEGIN_CERT.length(), - value.indexOf(END_CERT)); - } - // remove spaces, \r, \\r, \n, \\n, ], [ characters from certificate string - return value.replaceAll("\\\\r|\\\\n|\\r|\\n|\\[|]| ", StringUtils.EMPTY); - } - - /** - * Check whether the certificate is expired. - * - * @param peerCertificate the certificate to be checked - * @return true if the certificate is expired - */ - public static boolean isExpired(X509Certificate peerCertificate) { - try { - peerCertificate.checkValidity(); - } catch (CertificateException e) { - log.error("Certificate with the serial number " + - peerCertificate.getSerialNumber() + " issued by the CA " + - peerCertificate.getIssuerDN().toString() + " is expired. Caused by, " + e.getMessage()); - return true; - } - return false; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/DatabaseUtil.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/DatabaseUtil.java deleted file mode 100644 index 727f9f6c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/DatabaseUtil.java +++ /dev/null @@ -1,80 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingRuntimeException; -import com.wso2.openbanking.accelerator.common.persistence.JDBCPersistenceManager; -import com.wso2.openbanking.accelerator.common.persistence.JDBCRetentionDataPersistenceManager; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.SQLException; - -/** - * Utility class for database operations. - */ -public class DatabaseUtil { - - private static final Log log = LogFactory.getLog(DatabaseUtil.class); - - /** - * Get a database connection instance from the Consent Management Persistence Manager. - * - * @return Database Connection - * @throws OpenBankingRuntimeException Error when getting a database connection to Consent Management database - */ - public static Connection getDBConnection() throws OpenBankingRuntimeException { - - return JDBCPersistenceManager.getInstance().getDBConnection(); - } - - /** - * Get a database connection instance from the Retention Data Persistence Manager. - * - * @return Database Connection - * @throws OpenBankingRuntimeException Error when getting a database connection to retention database - */ - public static Connection getRetentionDBConnection() throws OpenBankingRuntimeException { - - return JDBCRetentionDataPersistenceManager.getInstance().getDBConnection(); - } - - public static void closeConnection(Connection dbConnection) { - - if (dbConnection != null) { - try { - dbConnection.close(); - } catch (SQLException e) { - log.error("Database error. Could not close statement. Continuing with others. - " - + e.getMessage().replaceAll("[\r\n]", ""), e); - } - } - } - - public static void rollbackTransaction(Connection dbConnection) { - - JDBCPersistenceManager.getInstance().rollbackTransaction(dbConnection); - } - - public static void commitTransaction(Connection dbConnection) { - - JDBCPersistenceManager.getInstance().commitTransaction(dbConnection); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/ErrorConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/ErrorConstants.java deleted file mode 100644 index 81106f21..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/ErrorConstants.java +++ /dev/null @@ -1,344 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - *

- * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util; - -/** - * Error Constant Class. - */ -public class ErrorConstants { - - //Error Response Structure constants - public static final String CODE = "Code"; - public static final String ID = "Id"; - public static final String ERRORS = "Errors"; - public static final String PATH = "Path"; - public static final String URL = "Url"; - public static final String ERROR = "error"; - - //Low level textual error code - public static final String FIELD_INVALID = "OB.Field.Invalid"; - public static final String FIELD_MISSING = "OB.Field.Missing"; - public static final String RESOURCE_INVALID_FORMAT = "OB.Resource.InvalidFormat"; - public static final String UNSUPPORTED_LOCAL_INSTRUMENTS = "OB.Unsupported.LocalInstrument"; - public static final String PATH_REQUEST_BODY = "Payload.Body"; - public static final String PATH_INSTRUCTED_AMOUNT = "Data.Initiation.InstructedAmount"; - public static final String PATH_CREDIT_ACCOUNT = "Data.Initiation.CreditorAccount"; - public static final String PATH_LOCAL_INSTRUMENT = "Data.Initiation.LocalInstrument"; - public static final String PATH_DEBTOR_ACCOUNT_NAME = "Data.Initiation.DebtorAccount.Name"; - public static final String PATH_DEBTOR_ACCOUNT_IDENTIFICATION = "Data.Initiation.DebtorAccount.Identification"; - public static final String PATH_DEBTOR_ACCOUNT_SCHEME = "Data.Initiation.DebtorAccount.SchemeName"; - public static final String PATH_CREDIT_ACCOUNT_SEC_IDENTIFICATION = "Data.Initiation.CreditorAccount" + - ".SecondaryIdentification"; - public static final String PATH_CREDIT_ACCOUNT_NAME = "Data.Initiation.CreditorAccount.Name"; - - public static final String PATH_CREDIT_ACCOUNT_IDENTIFICATION = "Data.Initiation.CreditorAccount.Identification"; - public static final String PATH_CREDIT_ACCOUNT_SCHEME = "Data.Initiation.CreditorAccount.SchemeName"; - public static final String PATH_INVALID = "Request path invalid"; - public static final String PAYLOAD_INVALID = "Consent validation failed due to invalid initiation payload"; - public static final String NOT_JSON_OBJECT_ERROR = "Payload is not a JSON object"; - public static final String PAYLOAD_FORMAT_ERROR = "Request Payload is not in correct JSON format"; - public static final String PAYLOAD_FORMAT_ERROR_VALID_TO_DATE = "Invalid valid_to_date parameter in the payload" + - "for valid to date"; - public static final String PAYLOAD_FORMAT_ERROR_DEBTOR_ACC = "Parameter Debtor Account does not exists "; - public static final String PAYLOAD_FORMAT_ERROR_CREDITOR_ACC = "Parameter Creditor Account " + - "does not exists "; - public static final String INVALID_REQ_PAYLOAD = "Invalid request payload"; - public static final String INVALID_REQ_PAYLOAD_INITIATION = "Invalid request payload in initiation key"; - public static final String INVALID_REQ_PAYLOAD_CONTROL_PARAMETERS = "Invalid request payload in " + - "control parameter key"; - public static final String MISSING_DEBTOR_ACC_SCHEME_NAME = "Mandatory parameter Debtor Account Scheme Name does " + - "not exists"; - public static final String MISSING_DEBTOR_ACC_IDENTIFICATION = "Mandatory parameter Debtor Account Identification" + - " does not exists"; - public static final String INVALID_DEBTOR_ACC_SCHEME_NAME = "Debtor Account Scheme Name does not match with the" + - " Scheme Names defined in the specification"; - public static final String INVALID_DEBTOR_ACC_IDENTIFICATION = "Debtor Account Identification should not exceed" + - " the max length of 256 characters defined in the specification"; - public static final String INVALID_DEBTOR_ACC_NAME = "Debtor Account Name should not exceed the max length of 70" + - " character defined in the specification"; - public static final String INVALID_DEBTOR_ACC_SEC_IDENTIFICATION = "Debtor Account Secondary Identification" + - " should not exceed the max length of 34 characters defined in the specification"; - public static final String NO_CONSENT_FOR_CLIENT_ERROR = "No valid consent found for given information"; - public static final String PAYMENT_INITIATION_HANDLE_ERROR = "Error occurred while handling the payment " + - "initiation request"; - public static final String MSG_ELAPSED_CUT_OFF_DATE_TIME = "{payment-order} consent / resource received after " + - "CutOffDateTime."; - public static final String MAX_INSTRUCTED_AMOUNT = "Instructed Amount specified exceed the Maximum Instructed " + - "Amount of the bank"; - public static final String INVALID_INSTRUCTED_AMOUNT = "Instructed Amount specified should be grater than zero"; - public static final String MSG_MISSING_CREDITOR_ACC = "Mandatory parameter CreditorAccount is missing in the" + - " payload."; - public static final String MISSING_CREDITOR_ACC_SCHEME_NAME = "Mandatory parameter Creditor Account Scheme Name" + - " does not exists"; - public static final String MISSING_CREDITOR_ACC_IDENTIFICATION = "Mandatory parameter Creditor Account " + - "Identification does not exists"; - public static final String INVALID_CREDITOR_ACC_SCHEME_NAME = "Creditor Account Scheme Name does not match with" + - " the Scheme Names defined in the specification"; - public static final String INVALID_CREDITOR_ACC_IDENTIFICATION = "Creditor Account Identification should not " + - "exceed the max length of 256 characters defined in the specification"; - public static final String INVALID_CREDITOR_ACC_NAME = "Creditor Account Name should not exceed the max length" + - " of 350 character defined in the specification"; - public static final String INVALID_CREDITOR_ACC_SEC_IDENTIFICATION = "Creditor Account Secondary Identification" + - " should not exceed the max length of 34 characters defined in the specification"; - public static final String INVALID_IDENTIFICATION = "Identification validation for SortCodeNumber Scheme failed."; - public static final String INVALID_LOCAL_INSTRUMENT = "The given local instrument value is not supported"; - public static final String INVALID_DEBTOR_ACC_SCHEME_NAME_LENGTH = "Debtor Account Scheme Name length does not " + - "match with the length defined in the specification"; - public static final String INVALID_CREDITOR_ACC_SCHEME_NAME_LENGTH = "Creditor Account Scheme Name length does" + - " not match with the length defined in the specification"; - public static final String IDEMPOTENCY_KEY_NOT_FOUND = "Idempotency related details should be submitted" + - " in order to proceed."; - public static final String MSG_INVALID_DEBTOR_ACC = "Mandatory parameter DebtorAccount object is invalid."; - public static final String PATH_DEBTOR_ACCOUNT = "Data.Initiation.DebtorAccount"; - public static final String COF_PATH_DEBTOR_ACCOUNT_SCHEME = "Data.DebtorAccount.SchemeName"; - public static final String COF_PATH_DEBTOR_ACCOUNT_IDENTIFICATION = "Data.DebtorAccount.Identification"; - public static final String COF_PATH_DEBTOR_ACCOUNT_NAME = "Data.DebtorAccount.Name"; - public static final String COF_PATH_DEBTOR_ACCOUNT_SECOND_IDENTIFICATION = - "Data.DebtorAccount.SecondaryIdentification"; - public static final String PATH_CUTOFF_DATE = "Data.CutOffDateTime"; - public static final String RULES_CUTOFF = "OB.Rules.AfterCutOffDateTime"; - public static final String PATH_CONSENT_ID = "Data.Initiation.Consent-id"; - public static final String PATH_DATA = "Data"; - public static final String PATH_INITIATION = "Data.Initiation"; - public static final String PATH_CONTROL_PARAMETERS = "Data.ControlParameters"; - public static final String PATH_RISK = "Data.Risk"; - public static final String PATH_URL = "Data.Url"; - public static final String PATH_EXPIRATION_DATE = "Data.Expiration-Date"; - public static final String MSG_MISSING_DEBTOR_ACC = "Mandatory parameter DebtorAccount is missing in the payload."; - public static final String REQUEST_OBJ_EXTRACT_ERROR = "Request object cannot be extracted"; - public static final String REQUEST_OBJ_NOT_SIGNED = "request object is not signed JWT"; - public static final String NOT_JSON_PAYLOAD = "Payload is not a JSON object"; - public static final String INTENT_ID_NOT_FOUND = "intent_id not found in request object"; - public static final String REQUEST_OBJ_PARSE_ERROR = "Error while parsing the request object."; - public static final String STATE_INVALID_ERROR = "Consent not in authorizable state"; - public static final String DATE_PARSE_MSG = "Parsed OffsetDateTime: %s, current OffsetDateTime: %s"; - public static final String EXP_DATE_PARSE_ERROR = "Error occurred while parsing the expiration date. "; - public static final String ACC_CONSENT_RETRIEVAL_ERROR = "Error occurred while retrieving the account initiation" + - " request details"; - public static final String CONSENT_EXPIRED = "Provided consent is expired"; - public static final String CONSENT_RETRIEVAL_ERROR = "Exception occurred while getting consent data"; - public static final String AUTH_CUT_OFF_DATE_ELAPSED = "Cut off time has elapsed"; - public static final String AUTH_TOKEN_REVOKE_ERROR = "Cutoff date time elapsed. Error while revoking the consent."; - public static final String ACCOUNT_ID_NOT_FOUND_ERROR = "Account IDs not available in persist request"; - public static final String ACCOUNT_ID_FORMAT_ERROR = "Account IDs format error in persist request"; - public static final String RESOURCE_CONSENT_MISMATCH = "OB.Resource.ConsentMismatch"; - public static final String INVALID_USER_ID = "Token received does not bound to the authorized user.:" - + ErrorConstants.PATH_ACCESS_TOKEN; - public static final String PATH_ACCESS_TOKEN = "Header.AccessToken"; - public static final String MSG_INVALID_CLIENT_ID = "The client Id related the consent does not match with the " + - "client id bound to token"; - public static final String PATH_CLIENT_ID = "Header.Client-id"; - public static final String UNEXPECTED_ERROR = "OB.UnexpectedError"; - public static final String INVALID_CONSENT_TYPE = "Invalid Consent Type found in the request"; - public static final String ACCOUNT_CONSENT_STATE_INVALID = "Account validation failed due to invalid consent" + - " state. :" + ErrorConstants.PATH_STATUS; - public static final String PATH_STATUS = "Payload.Status"; - public static final String RESOURCE_INVALID_CONSENT_STATUS = "OB.Resource.InvalidConsentStatus"; - public static final String INSTRUCTION_IDENTIFICATION_MISMATCH = "Instruction Identification does not match:" - + ErrorConstants.PATH_INSTRUCTION_IDENTIFICATION; - public static final String PATH_INSTRUCTION_IDENTIFICATION = "Data.Initiation.InstructionIdentification"; - public static final String END_TO_END_IDENTIFICATION_MISMATCH = "End to End Identification does not match:" - + ErrorConstants.PATH_ENDTOEND_IDENTIFICATION; - public static final String PATH_ENDTOEND_IDENTIFICATION = "Data.Initiation.EndToEndIdentification"; - public static final String END_TO_END_IDENTIFICATION_NOT_FOUND = "End to End Identification isn't present in " + - "the request or in the consent:" + ErrorConstants.PATH_ENDTOEND_IDENTIFICATION; - public static final String INSTRUCTED_AMOUNT_AMOUNT_MISMATCH = "Instructed Amount Amount does not match the " + - "initiated amount:" + ErrorConstants.PATH_INSTRUCTED_AMOUNT_AMOUNT; - public static final String PATH_INSTRUCTED_AMOUNT_AMOUNT = "Data.Initiation.InstructedAmount.Amount"; - public static final String INSTRUCTED_AMOUNT_AMOUNT_NOT_FOUND = "Instructed Amount Amount isn't present in the " + - "payload:"; - public static final String INSTRUCTED_AMOUNT_CURRENCY_MISMATCH = "Instructed Amount currency does not match the " + - "initiated amount or currency:" + ErrorConstants.PATH_INSTRUCTED_AMOUNT_CURRENCY; - public static final String PATH_INSTRUCTED_AMOUNT_CURRENCY = "Data.Initiation.InstructedAmount.Currency"; - public static final String INSTRUCTED_AMOUNT_CURRENCY_NOT_FOUND = "Instructed Amount Currency isn't present in " + - "the payload:"; - public static final String INSTRUCTED_AMOUNT_NOT_FOUND = "Instructed Amount isn't present in the payload"; - public static final String CREDITOR_ACC_SCHEME_NAME_MISMATCH = "Creditor Accounts Scheme does not match"; - public static final String CREDITOR_ACC_SCHEME_NAME_NOT_FOUND = "Creditor Accounts Scheme isn't present in the" + - " request or in the consent."; - public static final String CREDITOR_ACC_IDENTIFICATION_MISMATCH = "Creditor Account Identification does not match"; - public static final String CREDITOR_ACC_IDENTIFICATION_NOT_FOUND = "Creditor Account Identification isn't " + - "present in the request or in the consent."; - public static final String CREDITOR_ACC_NAME_MISMATCH = "Creditor Account Name does not match"; - public static final String CREDITOR_ACC_SEC_IDENTIFICATION_MISMATCH = "Creditor Account Secondary Identification" + - " does not match"; - - public static final String DEBTOR_ACC_SCHEME_NAME_MISMATCH = "Debtor Account Scheme name does not "; - public static final String DEBTOR_ACC_SCHEME_NAME_NOT_FOUND = "Debtor Account Scheme name isn't present in the " + - "request or in the consent"; - public static final String DEBTOR_ACC_IDENTIFICATION_MISMATCH = "Debtor Account Identification does " + - "not match:"; - public static final String DEBTOR_ACC_IDENTIFICATION_NOT_FOUND = "Debtor Account Identification isn't present " + - "in the request or in the consent"; - public static final String DEBTOR_ACC_NAME_MISMATCH = "Debtor Account Name does not match"; - public static final String DEBTOR_ACC_SEC_IDENTIFICATION_MISMATCH = "Debtor Account Secondary Identification" + - " does not match"; - public static final String PATH_DEBTOR_ACCOUNT_SECOND_IDENTIFICATION = - "Data.Initiation.DebtorAccount.SecondaryIdentification"; - public static final String CREDITOR_ACC_NOT_FOUND = "Creditor Account isn't present in the request."; - public static final String DEBTOR_ACC_MISMATCH = "Debtor Account mismatch"; - public static final String LOCAL_INSTRUMENT_MISMATCH = "Local Instrument Does Not Match" + - ErrorConstants.PATH_LOCAL_INSTRUMENT; - public static final String TOKEN_REVOKE_ERROR = "Token revocation unsuccessful. :" + - ErrorConstants.PATH_CUTOFF_DATE; - public static final String CUT_OFF_DATE_ELAPSED = "Cut off time has elapsed :" + - ErrorConstants.PATH_CUTOFF_DATE; - public static final String MSG_INVALID_CONSENT_ID = "The requested consent-Id does not match with the consent-Id" + - " bound to token"; - public static final String PAYMENT_CONSENT_STATE_INVALID = "Payment validation failed due to invalid consent" + - " state."; - public static final String VRP_CONSENT_STATUS_INVALID = "Validation failed due to invalid consent status."; - public static final String DATA_NOT_FOUND = "Data is not found or empty in the request."; - public static final String INITIATION_NOT_FOUND = "Initiation is not found or is empty in the request."; - public static final String RISK_MISMATCH = "RISK Does Not Match."; - public static final String INVALID_URI_ERROR = "Path requested is invalid. :" + ErrorConstants.PATH_URL; - public static final String COF_CONSENT_STATE_INVALID = "Confirmation of Funds validation failed due to invalid" + - " consent state.:" + ErrorConstants.PATH_STATUS; - public static final String CONSENT_EXPIRED_ERROR = "Provided consent is expired. :" - + ErrorConstants.PATH_EXPIRATION_DATE; - public static final String MSG_MISSING_CLIENT_ID = "Missing mandatory parameter x-wso2-client-id."; - public static final String RESOURCE_NOT_FOUND = "OB.Resource.NotFound"; - public static final String ACC_INITIATION_RETRIEVAL_ERROR = "Error occurred while handling the account initiation" + - " retrieval request"; - public static final String INVALID_CONSENT_ID = "Invalid Consent Id found in the request"; - public static final String CONSENT_ID_NOT_FOUND = "Consent ID not available in consent data"; - public static final String FIELD_INVALID_DATE = "OB.Field.InvalidDate"; - public static final String EXPIRED_DATE_ERROR = "The ExpirationDateTime value has to be a future date."; - public static final String CONSENT_ATTRIBUTE_RETRIEVAL_ERROR = "Error occurred while retrieving the consent " + - "attributes"; - - // VRP error constants - - public static final String VRP_INITIATION_HANDLE_ERROR = "Error occurred while handling the VRP " + - "initiation request"; - public static final String VRP_INITIATION_RETRIEVAL_ERROR = "Error occurred while handling the VRP initiation" + - " retrieval request"; - public static final String PAYLOAD_FORMAT_ERROR_VALID_FROM_DATE = "Request Payload is not in correct JSON format" + - " for valid from date"; - public static final String INVALID_VALID_TO_DATE_TIME = "Invalid date time format in validToDateTime"; - public static final String INVALID_VALID_FROM_DATE_TIME = "Invalid date time format in validFromDateTime"; - public static final String PAYLOAD_FORMAT_ERROR_CONTROL_PARAMETER = "Request Payload is not in correct JSON " + - "format for control parameter key"; - public static final String PAYLOAD_FORMAT_ERROR_MAXIMUM_INDIVIDUAL_AMOUNT = "Invalid maximum individual amount"; - public static final String MISSING_MAXIMUM_INDIVIDUAL_AMOUNT = "Missing mandatory parameter Maximum Individual" + - " Amount"; - public static final String PAYLOAD_FORMAT_ERROR_MAXIMUM_INDIVIDUAL_CURRENCY = "Invalid maximum individual amount" + - "currency"; - public static final String PAYLOAD_FORMAT_ERROR_INITIATION = "Missing mandatory parameter Initiation" + - " in the payload"; - public static final String PAYLOAD_FORMAT_ERROR_RISK = "Mandatory parameter Risk does not exists" + - " in the payload"; - public static final String INVALID_PERIOD_TYPE = "Invalid value for period type in PeriodicLimits"; - public static final String INVALID_PARAMETER = "Parameter passed in is null "; - public static final String INVALID_CLIENT_ID_MATCH = "Consent validation failed due to client ID mismatch"; - public static final String INVALID_DATE_TIME_FORMAT = "Date and Time is not in correct JSON " + - "ISO-8601 date-time format"; - public static final String MISSING_DATE_TIME_FORMAT = "The value is empty or the value is not a string"; - public static final String MISSING_VALID_TO_DATE_TIME = "Missing parameter ValidToDateTime"; - public static final String MISSING_VALID_FROM_DATE_TIME = "Missing parameter ValidFromDateTime"; - public static final String INVALID_PARAMETER_PERIODIC_LIMITS = "Parameter passed in is null , " + - "empty or not a JSONArray"; - public static final String MISSING_PERIOD_LIMITS = "Mandatory parameter " + - "periodic limits is missing in the payload"; - public static final String MISSING_PERIOD_TYPE = "Missing required parameter Period type"; - public static final String PAYLOAD_FORMAT_ERROR_PERIODIC_LIMITS_PERIOD_TYPE = "Value of period type is empty or " + - "the value passed in is not a string"; - public static final String PAYLOAD_FORMAT_ERROR_PERIODIC_LIMITS_ALIGNMENT = "Value of periodic alignment is empty" + - " or the value passed in is not a string"; - public static final String MISSING_PERIOD_ALIGNMENT = "Missing periodic alignment in periodic limits"; - public static final String INVALID_PERIOD_ALIGNMENT = "Invalid value for period alignment in PeriodicLimits"; - public static final String INVALID_PARAMETER_MESSAGE = "Parameter '%s' passed in is null, empty, or not a %s"; - public static final String DATE_INVALID_PARAMETER_MESSAGE = "Invalid date-time range for ValidToDateTime "; - public static final String INVALID_PERIODIC_LIMIT_SIZE = "Periodic limits exceed the allowed limits"; - public static final String DUPLICATE_PERIOD_TYPE = "Duplicate Period Types found in the request"; - public static final String CURRENCY_MISMATCH = "Currency does not match with the currency of the periodic limits"; - public static final int MAXIMUM_PERIODIC_LIMITS = 6; - public static final String INVALID_MAXIMUM_INDIVIDUAL_CURRENCY = "Invalid value for Currency in " + - "MaximumIndividualAmount"; - public static final String INVALID_PERIODIC_LIMIT_AMOUNT = "Invalid value for in Amount in Periodic Limits"; - public static final String INVALID_PERIODIC_LIMIT_CURRENCY = "Invalid value for Currency in Periodic Limits"; - - - // vrp path parameters - public static final String PATH_VALID_TO_DATE = "Data.ControlParameters.ValidToDateTime"; - public static final String PATH_VALID_FROM_DATE = "Data.ControlParameters.ValidFromDateTime"; - public static final String PATH_MAXIMUM_INDIVIDUAL_AMOUNT = "Data.ControlParameters.MaximumIndividualAmount"; - public static final String PATH_PERIOD_LIMIT = "Data.ControlParameters.PeriodicLimits"; - public static final String PATH_PERIOD_LIMIT_AMOUNT = "Data.ControlParameters.PeriodicLimits.Amount"; - public static final String PATH_PERIOD_LIMIT_CURRENCY = "Data.ControlParameters.PeriodicLimits.Currency"; - public static final String PATH_PERIOD_TYPE = "Data.ControlParameters.PeriodicLimits.PeriodType"; - public static final String PATH_PERIOD_ALIGNMENT = "Data.ControlParameters.PeriodicLimits.PeriodAlignment"; - - // VRP Authorization flow - public static final String CONTROL_PARAMETERS_MISSING_ERROR = "Missing mandatory parameter the ControlParameters"; - public static final String DATA_OBJECT_MISSING_ERROR = "Missing mandatory parameter the Data"; - public static final String MAX_AMOUNT_NOT_JSON_OBJECT_ERROR = "Parameter Maximum Individual Amount is" + - "not of type JSONObject"; - public static final String NOT_JSON_ARRAY_ERROR = "Parameter PeriodicLimits is not a JSON Array"; - public static final String PERIOD_ALIGNMENT_NOT_STRING_ERROR = "Parameter Period Alignment is not a String"; - public static final String PERIOD_TYPE_NOT_STRING_ERROR = "Parameter Period Type is not a String"; - public static final String NOT_STRING_ERROR = "Parameter amount or currency is not a String"; - - // VRP Submission flow - public static final String REMITTANCE_INFO_NOT_FOUND = "Remittance info is not present in the request."; - public static final String INSTRUCTION_IDENTIFICATION_NOT_FOUND = "Instruction Identification isn't present" + - " in the request"; - public static final String END_TO_END_IDENTIFICATION_PARAMETER_NOT_FOUND = "End to End Identification isn't" + - " present in the request"; - public static final String RISK_PARAMETER_MISMATCH = "RISK does not match"; - public static final String INSTRUCTED_AMOUNT_PARAMETER_NOT_FOUND = "Instructed Amount isn't present in the payload"; - public static final String INITIATION_REMITTANCE_INFO_PARAMETER_NOT_FOUND = "Remittance ifo present under" + - " initiation isn't present in the request"; - public static final String INSTRUCTION_REMITTANCE_INFO_PARAMETER_NOT_FOUND = "Remittance ifo present under" + - " instruction isn't present in the request"; - public static final String REMITTANCE_INFO_MISMATCH = "Remittance info does not match"; - public static final String REMITTANCE_UNSTRUCTURED_MISMATCH = "Remittance Information Unstructured does not " + - "match"; - public static final String INVALID_SUBMISSION_TYPE = "Value associated with INSTRUCTION_IDENTIFICATION key is " + - "not a String instance"; - public static final String INVALID_END_TO_END_IDENTIFICATION_TYPE = "Value associated with" + - " END_TO_END_IDENTIFICATION key is not a String instance"; - public static final String RISK_NOT_FOUND = "Risk is not found or empty in the request."; - public static final String RISK_NOT_JSON_ERROR = "Risk parameter is not in the correct JSON format"; - public static final String INSTRUCTION_NOT_FOUND = "Instruction is not found or empty in the request."; - public static final String INVALID_REQUEST_CONSENT_ID = "The consent-Id is not present in the request" + - " or it is not a String instance or there is a consentId mismatch"; - public static final String INSTRUCTION_CREDITOR_ACC_NOT_JSON_ERROR = "Creditor Account present under instruction" + - " isn't present in the correct JSON format in the request."; - public static final String INITIATION_CREDITOR_ACC_NOT_JSON_ERROR = "Creditor Account present under initiation" + - " isn't present in the correct JSON format in the request."; - public static final String DEBTOR_ACC_NOT_JSON_ERROR = "Debtor Account isn't present in the correct JSON format " + - "in the request."; - public static final String INITIATION_REMITTANCE_INFO_NOT_JSON_ERROR = "Remittance info of initiation is not " + - "present in the correct JSON format in the request."; - public static final String INSTRUCTION_REMITTANCE_INFO_NOT_JSON_ERROR = "Remittance info of instruction is not" + - " present in the correct JSON format in the request."; - public static final String DEBTOR_ACC_NOT_FOUND = "Debtor Account isn't present in the request."; - public static final String DATA_NOT_JSON_ERROR = "Data parameter is not in the correct JSON format in the request"; - public static final String INSTRUCTED_AMOUNT_NOT_STRING = "Value associated with Amount key is " + - "not a String instance"; - public static final String INSTRUCTED_AMOUNT_CURRENCY_NOT_STRING = "Value associated with Currency key is " + - "not a String instance"; - public static final String INSTRUCTED_AMOUNT_NOT_JSON_ERROR = "Instructed Amount is not in the correct JSON " + - "format in the request"; - public static final String INITIATION_NOT_JSON = "Initiation is not in the correct JSON " + - "format in the request"; - public static final String INSTRUCTION_NOT_JSON = "Instruction is not in the correct JSON format in the request"; -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/Generated.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/Generated.java deleted file mode 100644 index a3718e09..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/Generated.java +++ /dev/null @@ -1,28 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util; - -/** - * An annotation to make methods skip code coverage. Use only with a valid reason to skip - * code coverage. - */ -@java.lang.annotation.Retention(java.lang.annotation.RetentionPolicy.RUNTIME) -public @interface Generated { - String message(); -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/HTTPClientUtils.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/HTTPClientUtils.java deleted file mode 100644 index 5e84500f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/HTTPClientUtils.java +++ /dev/null @@ -1,205 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.config.Registry; -import org.apache.http.config.RegistryBuilder; -import org.apache.http.conn.socket.ConnectionSocketFactory; -import org.apache.http.conn.socket.PlainConnectionSocketFactory; -import org.apache.http.conn.ssl.SSLConnectionSocketFactory; -import org.apache.http.conn.ssl.SSLSocketFactory; -import org.apache.http.conn.ssl.TrustSelfSignedStrategy; -import org.apache.http.conn.ssl.X509HostnameVerifier; -import org.apache.http.impl.client.CloseableHttpClient; -import org.apache.http.impl.client.HttpClients; -import org.apache.http.impl.conn.PoolingHttpClientConnectionManager; -import org.apache.http.ssl.SSLContexts; -import org.wso2.carbon.base.ServerConfiguration; - -import java.io.FileInputStream; -import java.io.IOException; -import java.security.KeyManagementException; -import java.security.KeyStore; -import java.security.KeyStoreException; -import java.security.NoSuchAlgorithmException; -import java.security.cert.CertificateException; - -import javax.net.ssl.SSLContext; - -/** - * HTTP Client Utility methods. - */ -public class HTTPClientUtils { - - public static final String ALLOW_ALL = "AllowAll"; - public static final String STRICT = "Strict"; - public static final String HOST_NAME_VERIFIER = "httpclient.hostnameVerifier"; - public static final String HTTP_PROTOCOL = "http"; - public static final String HTTPS_PROTOCOL = "https"; - private static final String[] SUPPORTED_HTTP_PROTOCOLS = {"TLSv1.2"}; - private static final Log log = LogFactory.getLog(DatabaseUtil.class); - - /** - * Get closeable https client. - * - * @return Closeable https client - * @throws OpenBankingException OpenBankingException exception - */ - @Generated(message = "Ignoring because ServerConfiguration cannot be mocked") - public static CloseableHttpClient getHttpsClient() throws OpenBankingException { - - SSLConnectionSocketFactory sslsf = createSSLConnectionSocketFactory(); - - Registry socketFactoryRegistry = RegistryBuilder.create() - .register(HTTP_PROTOCOL, new PlainConnectionSocketFactory()) - .register(HTTPS_PROTOCOL, sslsf) - .build(); - - final PoolingHttpClientConnectionManager connectionManager = (socketFactoryRegistry != null) ? - new PoolingHttpClientConnectionManager(socketFactoryRegistry) : - new PoolingHttpClientConnectionManager(); - - // configuring default maximum connections - connectionManager.setMaxTotal(OpenBankingConfigParser.getInstance().getConnectionPoolMaxConnections()); - connectionManager.setDefaultMaxPerRoute(OpenBankingConfigParser.getInstance() - .getConnectionPoolMaxConnectionsPerRoute()); - - return HttpClients.custom().setConnectionManager(connectionManager).build(); - } - - /** - * Get closeable https client to send realtime event notifications. - * - * @return Closeable https client - * @throws OpenBankingException OpenBankingException exception - */ - @Generated(message = "Ignoring since method contains no logics") - public static CloseableHttpClient getRealtimeEventNotificationHttpsClient() throws OpenBankingException { - - SSLConnectionSocketFactory sslsf = createSSLConnectionSocketFactory(); - - Registry socketFactoryRegistry = RegistryBuilder.create() - .register(HTTP_PROTOCOL, new PlainConnectionSocketFactory()) - .register(HTTPS_PROTOCOL, sslsf) - .build(); - - final PoolingHttpClientConnectionManager connectionManager = (socketFactoryRegistry != null) ? - new PoolingHttpClientConnectionManager(socketFactoryRegistry) : - new PoolingHttpClientConnectionManager(); - - // configuring default maximum connections - connectionManager.setMaxTotal(OpenBankingConfigParser.getInstance() - .getRealtimeEventNotificationMaxRetries() + 1); - connectionManager.setDefaultMaxPerRoute(OpenBankingConfigParser.getInstance() - .getRealtimeEventNotificationMaxRetries() + 1); - - return HttpClients.custom().setConnectionManager(connectionManager).build(); - } - - /** - * create a SSL Connection Socket Factory. - * - * @return SSLConnectionSocketFactory - * @throws OpenBankingException - */ - @Generated(message = "Ignoring because ServerConfiguration cannot be mocked") - private static SSLConnectionSocketFactory createSSLConnectionSocketFactory() - throws OpenBankingException { - - KeyStore trustStore = null; - - trustStore = loadKeyStore( - ServerConfiguration.getInstance().getFirstProperty("Security.TrustStore.Location"), - ServerConfiguration.getInstance().getFirstProperty("Security.TrustStore.Password")); - - // Trust own CA and all self-signed certs - SSLContext sslcontext = null; - try { - sslcontext = SSLContexts.custom().loadTrustMaterial(trustStore, new TrustSelfSignedStrategy()).build(); - } catch (NoSuchAlgorithmException | KeyManagementException | KeyStoreException e) { - throw new OpenBankingException("Unable to create the ssl context", e); - } - - // Allow TLSv1 protocol only - return new SSLConnectionSocketFactory(sslcontext, SUPPORTED_HTTP_PROTOCOLS, - null, getX509HostnameVerifier()); - - } - - /** - * Load the keystore when the location and password is provided. - * - * @param keyStoreLocation Location of the keystore - * @param keyStorePassword Keystore password - * @return Keystore as an object - * @throws OpenBankingException when failed to load Keystore from given details - */ - @SuppressFBWarnings("PATH_TRAVERSAL_IN") - // Suppressed content - new FileInputStream(keyStoreLocation) - // Suppression reason - False Positive : Keystore location is obtained from deployment.toml. So it can be marked - // as a trusted filepath - // Suppressed warning count - 1 - public static KeyStore loadKeyStore(String keyStoreLocation, String keyStorePassword) - throws OpenBankingException { - - KeyStore keyStore; - - try (FileInputStream inputStream = new FileInputStream(keyStoreLocation)) { - keyStore = KeyStore.getInstance(KeyStore.getDefaultType()); - keyStore.load(inputStream, keyStorePassword.toCharArray()); - return keyStore; - } catch (KeyStoreException e) { - throw new OpenBankingException("Error while retrieving aliases from keystore", e); - } catch (IOException | CertificateException | NoSuchAlgorithmException e) { - throw new OpenBankingException("Error while loading keystore", e); - } - } - - /** - * Get the Hostname Verifier property in set in system properties. - * - * @return X509HostnameVerifier - */ - public static X509HostnameVerifier getX509HostnameVerifier() { - - String hostnameVerifierOption = System.getProperty(HOST_NAME_VERIFIER); - X509HostnameVerifier hostnameVerifier; - - if (ALLOW_ALL.equalsIgnoreCase(hostnameVerifierOption)) { - hostnameVerifier = SSLSocketFactory.ALLOW_ALL_HOSTNAME_VERIFIER; - } else if (STRICT.equalsIgnoreCase(hostnameVerifierOption)) { - hostnameVerifier = SSLSocketFactory.STRICT_HOSTNAME_VERIFIER; - } else { - hostnameVerifier = SSLSocketFactory.BROWSER_COMPATIBLE_HOSTNAME_VERIFIER; - } - - if (log.isDebugEnabled()) { - log.debug(String.format("Proceeding with %s : %s", HOST_NAME_VERIFIER, - hostnameVerifierOption)); - } - return hostnameVerifier; - - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/JWTUtils.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/JWTUtils.java deleted file mode 100644 index 172033df..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/JWTUtils.java +++ /dev/null @@ -1,254 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util; - -import com.nimbusds.jose.JOSEException; -import com.nimbusds.jose.JWSAlgorithm; -import com.nimbusds.jose.JWSObject; -import com.nimbusds.jose.JWSVerifier; -import com.nimbusds.jose.crypto.RSASSAVerifier; -import com.nimbusds.jose.jwk.source.RemoteJWKSet; -import com.nimbusds.jose.proc.BadJOSEException; -import com.nimbusds.jose.proc.JWSKeySelector; -import com.nimbusds.jose.proc.JWSVerificationKeySelector; -import com.nimbusds.jose.proc.SecurityContext; -import com.nimbusds.jose.proc.SimpleSecurityContext; -import com.nimbusds.jose.util.DefaultResourceRetriever; -import com.nimbusds.jwt.JWT; -import com.nimbusds.jwt.JWTParser; -import com.nimbusds.jwt.SignedJWT; -import com.nimbusds.jwt.proc.ConfigurableJWTProcessor; -import com.nimbusds.jwt.proc.DefaultJWTProcessor; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import net.minidev.json.JSONObject; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.net.MalformedURLException; -import java.net.URL; -import java.security.KeyFactory; -import java.security.NoSuchAlgorithmException; -import java.security.interfaces.RSAPublicKey; -import java.security.spec.InvalidKeySpecException; -import java.security.spec.X509EncodedKeySpec; -import java.text.ParseException; -import java.util.Base64; -import java.util.Date; -import java.util.Map; -import java.util.concurrent.ConcurrentHashMap; - -/** - * Util class for jwt related functions. - */ -public class JWTUtils { - - private static final Log log = LogFactory.getLog(JWTUtils.class); - private static final String RS = "RS"; - private static final String ALGORITHM_RSA = "RSA"; - - /** - * Decode request JWT. - * - * @param jwtToken jwt sent by the tpp - * @param jwtPart expected jwt part (header, body) - * @return json object containing requested jwt part - * @throws ParseException if an error occurs while parsing the jwt - */ - public static JSONObject decodeRequestJWT(String jwtToken, String jwtPart) throws ParseException { - - JSONObject jsonObject = new JSONObject(); - - JWSObject plainObject = JWSObject.parse(jwtToken); - - if ("head".equals(jwtPart)) { - jsonObject = plainObject.getHeader().toJSONObject(); - } else if ("body".equals(jwtPart)) { - jsonObject = plainObject.getPayload().toJSONObject(); - } - - return jsonObject; - - } - - /** - * Validate the signed JWT by querying a jwks. - * - * @param jwtString signed json web token - * @param jwksUri endpoint displaying the key set for the signing certificates - * @param algorithm the signing algorithm for jwt - * @return true if signature is valid - * @throws ParseException if an error occurs while parsing the jwt - * @throws BadJOSEException if the jwt is invalid - * @throws JOSEException if an error occurs while processing the jwt - * @throws MalformedURLException if an error occurs while creating the URL object - */ - @Generated(message = "Excluding from code coverage since can not call this method due to external https call") - public static boolean validateJWTSignature(String jwtString, String jwksUri, String algorithm) - throws ParseException, BadJOSEException, JOSEException, MalformedURLException { - - int defaultConnectionTimeout = 3000; - int defaultReadTimeout = 3000; - ConfigurableJWTProcessor jwtProcessor = new DefaultJWTProcessor<>(); - JWT jwt = JWTParser.parse(jwtString); - // set the Key Selector for the jwks_uri. - Map> jwkSourceMap = new ConcurrentHashMap<>(); - RemoteJWKSet jwkSet = jwkSourceMap.get(jwksUri); - if (jwkSet == null) { - int connectionTimeout = Integer.parseInt(OpenBankingConfigParser.getInstance().getJWKSConnectionTimeOut()); - int readTimeout = Integer.parseInt(OpenBankingConfigParser.getInstance().getJWKSReadTimeOut()); - int sizeLimit = RemoteJWKSet.DEFAULT_HTTP_SIZE_LIMIT; - if (connectionTimeout == 0 && readTimeout == 0) { - connectionTimeout = defaultConnectionTimeout; - readTimeout = defaultReadTimeout; - } - DefaultResourceRetriever resourceRetriever = new DefaultResourceRetriever( - connectionTimeout, - readTimeout, - sizeLimit); - jwkSet = new RemoteJWKSet<>(new URL(jwksUri), resourceRetriever); - jwkSourceMap.put(jwksUri, jwkSet); - } - // The expected JWS algorithm of the access tokens (agreed out-of-band). - JWSAlgorithm expectedJWSAlg = JWSAlgorithm.parse(algorithm); - //Configure the JWT processor with a key selector to feed matching public RSA keys sourced from the JWK set URL. - JWSKeySelector keySelector = new JWSVerificationKeySelector<>(expectedJWSAlg, jwkSet); - jwtProcessor.setJWSKeySelector(keySelector); - // Process the token, set optional context parameters. - SimpleSecurityContext securityContext = new SimpleSecurityContext(); - jwtProcessor.process((SignedJWT) jwt, securityContext); - return true; - } - - /** - * Validates the signature of a given JWT against a given public key. - * - * @param signedJWT the signed JWT to be validated - * @param publicKey the public key that is used for validation - * @return true if signature is valid else false - * @throws NoSuchAlgorithmException if the given algorithm doesn't exist - * @throws InvalidKeySpecException if the provided key is invalid - * @throws JOSEException if an error occurs during the signature validation process - */ - @Generated(message = "Excluding from code coverage as KeyFactory does not initialize in testsuite") - public static boolean isValidSignature(SignedJWT signedJWT, String publicKey) - throws NoSuchAlgorithmException, InvalidKeySpecException, JOSEException, OpenBankingException { - - byte[] publicKeyData = Base64.getDecoder().decode(publicKey); - X509EncodedKeySpec spec = new X509EncodedKeySpec(publicKeyData); - // Example : RS256 - String algorithm = signedJWT.getHeader().getAlgorithm().getName(); - KeyFactory kf = getKeyFactory(algorithm); - RSAPublicKey rsapublicKey = (RSAPublicKey) kf.generatePublic(spec); - JWSVerifier verifier = new RSASSAVerifier(rsapublicKey); - return signedJWT.verify(verifier); - } - - /** - * Validate legitimacy of a JWS. - * - * @param jwsString JWT string - * @return true if a given jwsString adheres a valid JWS Format - */ - public static boolean isValidJWSFormat(String jwsString) { - - return StringUtils.isBlank(jwsString) ? false : - StringUtils.countMatches(jwsString, OpenBankingConstants.DOT_SEPARATOR) == 2; - } - - /** - * Parses the provided JWT string into a SignedJWT object. - * - * @param jwtString the JWT string to parse - * @return the parsed SignedJWT object - * @throws IllegalArgumentException if the provided token identifier is not a parsable JWT - * - */ - public static SignedJWT getSignedJWT(String jwtString) throws ParseException { - - if (isValidJWSFormat(jwtString)) { - return SignedJWT.parse(jwtString); - } else { - if (log.isDebugEnabled()) { - log.debug(String.format("Provided token identifier is not a parsable JWT: %s", jwtString)); - } - throw new IllegalArgumentException("Provided token identifier is not a parsable JWT."); - } - } - - /** - * Checks if the given expiration time is valid based on the current system time and a default time skew. - * - * @param defaultTimeSkew defaultTimeSkew to adjust latency issues. - * @param expirationTime the exp of the jwt that should be validated. - * @return True if the expiration time is valid considering the default time skew; false otherwise. - */ - public static boolean isValidExpiryTime(Date expirationTime, long defaultTimeSkew) { - - if (expirationTime != null) { - long timeStampSkewMillis = defaultTimeSkew * 1000; - long expirationTimeInMillis = expirationTime.getTime(); - long currentTimeInMillis = System.currentTimeMillis(); - return (currentTimeInMillis + timeStampSkewMillis) <= expirationTimeInMillis; - } else { - return false; - } - } - - /** - * Checks if the given "not before" time is valid based on the current system time and a default time skew. - * - * @param defaultTimeSkew defaultTimeSkew to adjust latency issues. - * @param notBeforeTime nbf of the jwt that should be validated - * @return True if the "not before" time is valid considering the default time skew; false otherwise. - */ - public static boolean isValidNotValidBeforeTime(Date notBeforeTime, long defaultTimeSkew) { - - if (notBeforeTime != null) { - long timeStampSkewMillis = defaultTimeSkew * 1000; - long notBeforeTimeMillis = notBeforeTime.getTime(); - long currentTimeInMillis = System.currentTimeMillis(); - return currentTimeInMillis + timeStampSkewMillis >= notBeforeTimeMillis; - } else { - return false; - } - } - - /** - * Returns a KeyFactory instance for the specified algorithm. - * - * @param algorithm the algorithm name, such as "RS256". - * @return the KeyFactory instance. - * @throws OpenBankingException if the provided algorithm is not supported. - * @throws NoSuchAlgorithmException if the specified algorithm is invalid. - */ - @Generated(message = "Excluding from code coverage as KeyFactory does not initialize in testsuite") - private static KeyFactory getKeyFactory(String algorithm) throws OpenBankingException, NoSuchAlgorithmException { - - // In here if the algorithm is directly passes (like RS256) it will generate exceptions - // hence Base algorithm should be passed (Example: RSA) - if (algorithm.indexOf(RS) == 0) { - return KeyFactory.getInstance(ALGORITHM_RSA); - } else { - throw new OpenBankingException("Algorithm " + algorithm + " not yet supported."); - } - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/OpenBankingUtils.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/OpenBankingUtils.java deleted file mode 100644 index 98593449..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/OpenBankingUtils.java +++ /dev/null @@ -1,117 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingRuntimeException; -import com.wso2.openbanking.accelerator.common.identity.IdentityConstants; -import net.minidev.json.JSONObject; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.lang.reflect.InvocationTargetException; -import java.text.ParseException; - -/** - * Open Banking common utility class. - */ -public class OpenBankingUtils { - - private static final Log log = LogFactory.getLog(OpenBankingUtils.class); - - /** - * Method to obtain the Object when the full class path is given. - * - * @param classpath full class path - * @return new object instance - */ - @Generated(message = "Ignoring since method contains no logics") - public static Object getClassInstanceFromFQN(String classpath) { - - try { - return Class.forName(classpath).getDeclaredConstructor().newInstance(); - } catch (ClassNotFoundException e) { - log.error("Class not found: " + classpath.replaceAll("[\r\n]", "")); - throw new OpenBankingRuntimeException("Cannot find the defined class", e); - } catch (InstantiationException | InvocationTargetException | - NoSuchMethodException | IllegalAccessException e) { - //Throwing a runtime exception since we cannot proceed with invalid objects - throw new OpenBankingRuntimeException("Defined class" + classpath + "cannot be instantiated.", e); - } - } - - /** - * Extract software_environment (SANDBOX or PRODUCTION) from SSA. - * - * @param softwareStatement software statement (jwt) extracted from request payload - * @return software_environment - * @throws ParseException if an error occurs while parsing the software statement - */ - public static String getSoftwareEnvironmentFromSSA(String softwareStatement) throws ParseException { - - if (StringUtils.isEmpty(softwareStatement)) { - return IdentityConstants.PRODUCTION; - } - - final JSONObject softwareStatementBody = JWTUtils.decodeRequestJWT(softwareStatement, - OpenBankingConstants.JWT_BODY); - // Retrieve the SSA property name used for software environment identification - final String sandboxEnvIdentificationPropertyName = OpenBankingConfigParser.getInstance() - .getSoftwareEnvIdentificationSSAPropertyName(); - // Retrieve the expected value for the sandbox environment - final String sandboxEnvIdentificationValue = OpenBankingConfigParser.getInstance() - .getSoftwareEnvIdentificationSSAPropertyValueForSandbox(); - return sandboxEnvIdentificationValue.equalsIgnoreCase(softwareStatementBody - .getAsString(sandboxEnvIdentificationPropertyName)) - ? IdentityConstants.SANDBOX - : IdentityConstants.PRODUCTION; - } - - /** - * Method to obtain boolean value for check if the Dispute Resolution Data is publishable. - * - * @param statusCode for dispute data - * @return boolean - */ - public static boolean isPublishableDisputeData(int statusCode) { - - if (statusCode < 400 && - OpenBankingConfigParser.getInstance().isNonErrorDisputeDataPublishingEnabled()) { - return true; - } - - return statusCode >= 400; - } - - /** - * Method to reduce string length. - * - * @param input Input for dispute data - * @param maxLength Max length for dispute data - * @return String with reduced length - */ - public static String reduceStringLength(String input, int maxLength) { - if (StringUtils.isEmpty(input) || input.length() <= maxLength) { - return input; - } else { - return input.substring(0, maxLength); - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/SPQueryExecutorUtil.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/SPQueryExecutorUtil.java deleted file mode 100644 index 72c83d1c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/SPQueryExecutorUtil.java +++ /dev/null @@ -1,94 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util; - -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.HttpEntity; -import org.apache.http.HttpHeaders; -import org.apache.http.HttpResponse; -import org.apache.http.HttpStatus; -import org.apache.http.client.methods.HttpPost; -import org.apache.http.entity.StringEntity; -import org.apache.http.impl.client.CloseableHttpClient; -import org.apache.http.util.EntityUtils; - -import java.io.IOException; -import java.nio.charset.StandardCharsets; -import java.util.Base64; - -/** - * Util class to handle communications with stream processor. - */ -public class SPQueryExecutorUtil { - - private static Log log = LogFactory.getLog(SPQueryExecutorUtil.class); - - /** - * Executes the given query in SP. - * - * @param appName Name of the siddhi app. - * @param query Name of the query - * @param spUserName Username for SP - * @param spPassword Password for SP - * @param spApiHost Hostname of the SP - * @return JSON object with result - * @throws IOException IO Exception. - * @throws ParseException Parse Exception. - * @throws OpenBankingException OpenBanking Exception. - */ - public static JSONObject executeQueryOnStreamProcessor(String appName, String query, String spUserName, - String spPassword, String spApiHost) - throws IOException, ParseException, OpenBankingException { - byte[] encodedAuth = Base64.getEncoder() - .encode((spUserName + ":" + spPassword).getBytes(StandardCharsets.ISO_8859_1)); - String authHeader = "Basic " + new String(encodedAuth, StandardCharsets.UTF_8.toString()); - - CloseableHttpClient httpClient = HTTPClientUtils.getHttpsClient();; - HttpPost httpPost = new HttpPost(spApiHost + OpenBankingConstants.SP_API_PATH); - httpPost.setHeader(HttpHeaders.AUTHORIZATION, authHeader); - JSONObject jsonObject = new JSONObject(); - jsonObject.put(OpenBankingConstants.APP_NAME_CC, appName); - jsonObject.put(OpenBankingConstants.QUERY, query); - StringEntity requestEntity = new StringEntity(jsonObject.toJSONString()); - httpPost.setHeader(OpenBankingConstants.CONTENT_TYPE_TAG, OpenBankingConstants.JSON_CONTENT_TYPE); - httpPost.setEntity(requestEntity); - HttpResponse response; - - if (log.isDebugEnabled()) { - log.debug(String.format("Executing query %s on SP", query)); - } - response = httpClient.execute(httpPost); - HttpEntity entity = response.getEntity(); - if (response.getStatusLine().getStatusCode() != HttpStatus.SC_OK) { - String error = String.format("Error while invoking SP rest api : %s %s", - response.getStatusLine().getStatusCode(), response.getStatusLine().getReasonPhrase()); - log.error(error); - return null; - } - String responseStr = EntityUtils.toString(entity); - JSONParser parser = new JSONParser(JSONParser.MODE_PERMISSIVE); - return (JSONObject) parser.parse(responseStr); - } - } diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/SecurityUtils.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/SecurityUtils.java deleted file mode 100644 index a800f2ea..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/SecurityUtils.java +++ /dev/null @@ -1,80 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util; - -import org.apache.commons.lang3.StringUtils; - -import java.util.List; -import java.util.Set; -import java.util.stream.Collectors; - -/** - * Common Security Utils class. - */ -public class SecurityUtils { - - private static final String specialChars = "!@#$%&*()'+,-./:;<=>?[]^_`{|}"; - - /** - * Method to remove new line characters to avoid potential CRLF injection for logs. - * Bug kind and pattern: SECCRLFLOG - CRLF_INJECTION_LOGS - * - * @param string string - * @return string without new line characters - */ - public static String sanitizeString(String string) { - return string.replaceAll("[\r\n]", ""); - } - - /** - * Method to remove new line characters from a list of strings to avoid potential CRLF injection for logs. - * Bug kind and pattern: SECCRLFLOG - CRLF_INJECTION_LOGS - * - * @param stringList String List - * @return string without new line characters - */ - public static List sanitize(List stringList) { - return stringList.stream() - .map(SecurityUtils::sanitizeString) - .collect(Collectors.toList()); - } - - /** - * Method to remove new line characters from a setmvn of strings to avoid potential CRLF injection for logs. - * Bug kind and pattern: SECCRLFLOG - CRLF_INJECTION_LOGS - * - * @param stringSet String Set - * @return string without new line characters - */ - public static Set sanitize(Set stringSet) { - return stringSet.stream() - .map(SecurityUtils::sanitizeString) - .collect(Collectors.toSet()); - } - - /** - * Method to validate a string does not contain special characters. - * - * @param string String - * @return whether the string does not contain any special characters - */ - public static boolean containSpecialChars(String string) { - return StringUtils.containsAny(string, specialChars); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/ServiceProviderUtils.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/ServiceProviderUtils.java deleted file mode 100644 index fae34feb..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/ServiceProviderUtils.java +++ /dev/null @@ -1,47 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import org.wso2.carbon.identity.oauth.common.exception.InvalidOAuthClientException; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.util.OAuth2Util; - -/** - * Utility Class for Service Provider related functions. - */ -public class ServiceProviderUtils { - - /** - * Get Tenant Domain String for the client id. - * @param clientId the client id of the application - * @return tenant domain of the client - * @throws OpenBankingException if an error occurs while retrieving the tenant domain - */ - @Generated(message = "Ignoring because OAuth2Util cannot be mocked with no constructors") - public static String getSpTenantDomain(String clientId) throws OpenBankingException { - - try { - return OAuth2Util.getTenantDomainOfOauthApp(clientId); - } catch (InvalidOAuthClientException | IdentityOAuth2Exception e) { - throw new OpenBankingException("Error retrieving service provider tenant domain for client_id: " - + clientId, e); - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/CertificateContent.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/CertificateContent.java deleted file mode 100644 index 4dbd6efe..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/CertificateContent.java +++ /dev/null @@ -1,106 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor; - -import java.util.Date; -import java.util.List; - -/** - * Class That Contains Extracted PSD2 Attributes from the certificate. - */ -public class CertificateContent { - private String pspAuthorisationNumber; - private List pspRoles; - private String name; - private String ncaName; - private String ncaId; - private Date notAfter = null; - private Date notBefore = null; - - - public String getPspAuthorisationNumber() { - - return pspAuthorisationNumber; - } - - public void setPspAuthorisationNumber(String pspAuthorisationNumber) { - - this.pspAuthorisationNumber = pspAuthorisationNumber; - } - - public List getPspRoles() { - - return pspRoles; - } - - public void setPspRoles(List pspRoles) { - - this.pspRoles = pspRoles; - } - - public String getName() { - - return name; - } - - public void setName(String name) { - - this.name = name; - } - - public String getNcaName() { - - return ncaName; - } - - public void setNcaName(String ncaName) { - - this.ncaName = ncaName; - } - - public String getNcaId() { - - return ncaId; - } - - public void setNcaId(String ncaId) { - - this.ncaId = ncaId; - } - - public Date getNotAfter() { - - return new Date(notAfter.getTime()); - } - - public void setNotAfter(Date notAfter) { - - this.notAfter = new Date(notAfter.getTime()); - } - - public Date getNotBefore() { - - return new Date(notBefore.getTime()); - } - - public void setNotBefore(Date notBefore) { - - this.notBefore = new Date(notBefore.getTime()); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/CertificateContentExtractor.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/CertificateContentExtractor.java deleted file mode 100644 index 2f6b5093..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/CertificateContentExtractor.java +++ /dev/null @@ -1,107 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor; - -import com.wso2.openbanking.accelerator.common.exception.CertificateValidationException; -import com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor.common.PSD2QCStatement; -import com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor.common.PSD2QCType; -import com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor.common.PSPRole; -import com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor.common.PSPRoles; -import com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor.error.CertValidationErrors; -import org.apache.commons.lang3.ArrayUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.bouncycastle.asn1.ASN1ObjectIdentifier; -import org.bouncycastle.asn1.x500.X500Name; -import org.bouncycastle.asn1.x500.style.BCStyle; -import org.bouncycastle.asn1.x500.style.IETFUtils; -import org.bouncycastle.cert.jcajce.JcaX509CertificateHolder; - -import java.security.cert.CertificateEncodingException; -import java.security.cert.X509Certificate; -import java.util.ArrayList; -import java.util.List; - -/** - * Class that extracts the PSD2 attributes from v3 extensions of X509 certificates. - */ -public class CertificateContentExtractor { - - private static final Log log = LogFactory.getLog(CertificateContentExtractor.class); - - private CertificateContentExtractor() { - } - - public static CertificateContent extract(X509Certificate cert) - throws CertificateValidationException { - - if (cert == null) { - log.error("Error reading certificate "); - throw new CertificateValidationException( - CertValidationErrors.CERTIFICATE_INVALID.toString()); - } - - CertificateContent tppCertData = new CertificateContent(); - - tppCertData.setNotAfter(cert.getNotAfter()); - tppCertData.setNotBefore(cert.getNotBefore()); - - PSD2QCType psd2QcType = PSD2QCStatement.getPsd2QCType(cert); - PSPRoles pspRoles = psd2QcType.getPspRoles(); - List rolesArray = pspRoles.getRoles(); - - List roles = new ArrayList<>(); - - for (PSPRole pspRole : rolesArray) { - roles.add(pspRole.getPsd2RoleName()); - } - tppCertData.setPspRoles(roles); - - tppCertData.setNcaName(psd2QcType.getnCAName().getString()); - tppCertData.setNcaId(psd2QcType.getnCAId().getString()); - - try { - X500Name x500name = new JcaX509CertificateHolder(cert).getSubject(); - - tppCertData.setPspAuthorisationNumber(getNameValueFromX500Name(x500name, BCStyle.ORGANIZATION_IDENTIFIER)); - tppCertData.setName(getNameValueFromX500Name(x500name, BCStyle.CN)); - - if (log.isDebugEnabled()) { - log.debug("Extracted TPP eIDAS certificate data: " + "[ " + tppCertData.toString() + " ]"); - } - - } catch (CertificateEncodingException e) { - log.error("Certificate read error. caused by, ", e); - throw new CertificateValidationException(CertValidationErrors.CERTIFICATE_INVALID.toString(), e); - - } - return tppCertData; - - } - - private static String getNameValueFromX500Name(X500Name x500Name, ASN1ObjectIdentifier asn1ObjectIdentifier) { - - if (ArrayUtils.contains(x500Name.getAttributeTypes(), asn1ObjectIdentifier)) { - return IETFUtils.valueToString(x500Name.getRDNs(asn1ObjectIdentifier)[0].getFirst().getValue()); - } else { - return ""; - } - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/NcaId.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/NcaId.java deleted file mode 100644 index c820e28f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/NcaId.java +++ /dev/null @@ -1,40 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor.common; - -import org.bouncycastle.asn1.DERUTF8String; - -/** - * NcaId class. - */ -public class NcaId extends DERUTF8String { - - public NcaId(String string) { - - super(string); - } - - public static NcaId getInstance(Object obj) { - - if (obj instanceof NcaId) { - return (NcaId) obj; - } - return new NcaId(DERUTF8String.getInstance(obj).getString()); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/NcaName.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/NcaName.java deleted file mode 100644 index 02436c1d..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/NcaName.java +++ /dev/null @@ -1,41 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor.common; - -import org.bouncycastle.asn1.DERUTF8String; - -/** - * NcaName class. - */ -public class NcaName extends DERUTF8String { - - public NcaName(String string) { - - super(string); - } - - public static NcaName getInstance(Object obj) { - - if (obj instanceof NcaName) { - return (NcaName) obj; - } - return new NcaName(DERUTF8String.getInstance(obj).getString()); - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSD2Constants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSD2Constants.java deleted file mode 100644 index adfee23c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSD2Constants.java +++ /dev/null @@ -1,43 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor.common; - -/** - * PSD2Constants class. - */ -public class PSD2Constants { - - //Role Names on the certificate - public static final String PSP_AS = "PSP_AS"; - public static final String PSP_PI = "PSP_PI"; - public static final String PSP_AI = "PSP_AI"; - public static final String PSP_IC = "PSP_IC"; - - //PSD2 Role OIDs in the certificate - public static final String PSP_AS_OID = "0.4.0.19495.1.1"; - public static final String PSP_PI_OID = "0.4.0.19495.1.2"; - public static final String PSP_AI_OID = "0.4.0.19495.1.3"; - public static final String PSP_IC_OID = "0.4.0.19495.1.4"; - - //PSD2 Role Names - public static final String ASPSP = "ASPSP"; - public static final String PISP = "PISP"; - public static final String AISP = "AISP"; - public static final String CBPII = "CBPII"; -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSD2QCStatement.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSD2QCStatement.java deleted file mode 100644 index 469048ef..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSD2QCStatement.java +++ /dev/null @@ -1,119 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor.common; - -import com.wso2.openbanking.accelerator.common.exception.CertificateValidationException; -import com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor.error.CertValidationErrors; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.bouncycastle.asn1.ASN1Encodable; -import org.bouncycastle.asn1.ASN1InputStream; -import org.bouncycastle.asn1.ASN1ObjectIdentifier; -import org.bouncycastle.asn1.ASN1Sequence; -import org.bouncycastle.asn1.DEROctetString; -import org.bouncycastle.asn1.x509.Extension; -import org.bouncycastle.asn1.x509.qualified.QCStatement; - -import java.io.ByteArrayInputStream; -import java.io.IOException; -import java.security.cert.X509Certificate; -import java.util.Iterator; -import java.util.Optional; - -/** - * PSD2QCStatement class. - */ -public class PSD2QCStatement { - - private static final ASN1ObjectIdentifier psd2QcStatementOid = new ASN1ObjectIdentifier("0.4.0.19495.2"); - private static final Log log = LogFactory.getLog(PSD2QCStatement.class); - - public static PSD2QCType getPsd2QCType(X509Certificate cert) throws CertificateValidationException { - - byte[] extensionValue = cert.getExtensionValue(Extension.qCStatements.getId()); - if (extensionValue == null) { - log.debug("Extension that contains the QCStatement not found in the certificate"); - throw new CertificateValidationException(CertValidationErrors.EXTENSION_NOT_FOUND.toString()); - } - - QCStatement qcStatement = extractQCStatement(extensionValue); - - ASN1Encodable statementInfo = qcStatement.getStatementInfo(); - - return PSD2QCType.getInstance(statementInfo); - - } - - private static QCStatement extractQCStatement(byte[] extensionValue) throws CertificateValidationException { - - ASN1Sequence qcStatements; - try { - try (ASN1InputStream derAsn1InputStream = new ASN1InputStream(new ByteArrayInputStream(extensionValue))) { - DEROctetString oct = (DEROctetString) (derAsn1InputStream.readObject()); - try (ASN1InputStream asn1InputStream = new ASN1InputStream(oct.getOctets())) { - qcStatements = (ASN1Sequence) asn1InputStream.readObject(); - } - } - } catch (IOException e) { - log.error("Error reading QCStatement ", e); - throw new CertificateValidationException(CertValidationErrors.QCSTATEMENT_INVALID.toString()); - } - - if (qcStatements.size() <= 0) { - log.error("QCStatements not found in the certificate"); - throw new CertificateValidationException(CertValidationErrors.QCSTATEMENTS_NOT_FOUND.toString()); - } - - ASN1Encodable object = qcStatements.getObjectAt(0); - if (object.toASN1Primitive() instanceof ASN1ObjectIdentifier) { - return getSingleQcStatement(qcStatements); - } - - return extractPsd2QcStatement(qcStatements) - .orElseThrow(() -> new CertificateValidationException( - CertValidationErrors.PSD2_QCSTATEMENT_NOT_FOUND.toString())); - } - - private static QCStatement getSingleQcStatement(ASN1Sequence qcStatements) throws CertificateValidationException { - - QCStatement qcStatement = QCStatement.getInstance(qcStatements); - if (!psd2QcStatementOid.getId().equals(qcStatement.getStatementId().getId())) { - log.error("Invalid QC statement type in psd2 certificate. expected [" + - psd2QcStatementOid.getId().replaceAll("[\r\n]", "") + "] but found [" + - qcStatement.getStatementId().getId().replaceAll("[\r\n]", "") + "]"); - throw new CertificateValidationException(CertValidationErrors.PSD2_QCSTATEMENT_NOT_FOUND.toString()); - } - - return qcStatement; - } - - private static Optional extractPsd2QcStatement(ASN1Sequence qcStatements) { - - Iterator iterator = qcStatements.iterator(); - - while (iterator.hasNext()) { - QCStatement qcStatement = QCStatement.getInstance(iterator.next()); - if (qcStatement != null && qcStatement.getStatementId().getId().equals(psd2QcStatementOid.getId())) { - return Optional.of(qcStatement); - } - } - - return Optional.empty(); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSD2QCType.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSD2QCType.java deleted file mode 100644 index ea9b0c24..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSD2QCType.java +++ /dev/null @@ -1,64 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor.common; - -import org.bouncycastle.asn1.ASN1Encodable; -import org.bouncycastle.asn1.ASN1Sequence; - -/** - * PSD2QCType class. - */ -public class PSD2QCType { - - private final PSPRoles pspRoles; - private final NcaName nCAName; - private final NcaId nCAId; - - public PSD2QCType(PSPRoles pspRoles, NcaName nCAName, NcaId nCAId) { - - this.pspRoles = pspRoles; - this.nCAName = nCAName; - this.nCAId = nCAId; - } - - public static PSD2QCType getInstance(ASN1Encodable asn1Encodable) { - - ASN1Sequence sequence = ASN1Sequence.getInstance(asn1Encodable); - PSPRoles pspRoles = PSPRoles.getInstance(sequence.getObjectAt(0)); - NcaName nCAName = NcaName.getInstance(sequence.getObjectAt(1)); - NcaId nCAId = NcaId.getInstance(sequence.getObjectAt(2)); - return new PSD2QCType(pspRoles, nCAName, nCAId); - } - - - public PSPRoles getPspRoles() { - - return pspRoles; - } - - public NcaName getnCAName() { - - return nCAName; - } - - public NcaId getnCAId() { - - return nCAId; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSPRole.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSPRole.java deleted file mode 100644 index 6ec64070..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSPRole.java +++ /dev/null @@ -1,88 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor.common; - -import org.bouncycastle.asn1.ASN1Encodable; -import org.bouncycastle.asn1.ASN1ObjectIdentifier; -import org.bouncycastle.asn1.ASN1Sequence; -import org.bouncycastle.asn1.ASN1UTF8String; -import org.bouncycastle.asn1.DERUTF8String; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Iterator; -import java.util.List; - -/** - * PSPRole enum. - */ -public enum PSPRole { - PSP_AS(PSD2Constants.PSP_AS_OID, PSD2Constants.PSP_AS, PSD2Constants.ASPSP), - PSP_PI(PSD2Constants.PSP_PI_OID, PSD2Constants.PSP_PI, PSD2Constants.PISP), - PSP_AI(PSD2Constants.PSP_AI_OID, PSD2Constants.PSP_AI, PSD2Constants.AISP), - PSP_IC(PSD2Constants.PSP_IC_OID, PSD2Constants.PSP_IC, PSD2Constants.CBPII); - - private String pspRoleOid; //Object Identifier in the Certificate - private String pspRoleName; //Role Name stated on the certificate - private String psd2RoleName; //PSD2 Actor Name related to the role in the certificate - - PSPRole(String pspRoleOid, String pspRoleName, String psd2RoleName) { - - this.pspRoleOid = pspRoleOid; - this.pspRoleName = pspRoleName; - this.psd2RoleName = psd2RoleName; - } - - public static List getInstance(ASN1Encodable asn1Encodable) { - - List pspRoleList = new ArrayList<>(); - ASN1Sequence sequence = ASN1Sequence.getInstance(asn1Encodable); - - Iterator it = sequence.iterator(); - while (it.hasNext()) { - ASN1ObjectIdentifier objectIdentifier = ASN1ObjectIdentifier.getInstance(it.next()); - ASN1UTF8String instance = DERUTF8String.getInstance(it.next()); - - pspRoleList.add(Arrays.stream(PSPRole.values()) - .filter(role -> role.getPspRoleOid().equals(objectIdentifier.getId()) - && role.getPspRoleName().equals(instance.getString())) - .findFirst().orElseThrow(() -> new IllegalArgumentException( - "unknown object in getInstance: " + asn1Encodable.getClass().getName()))); - } - - return pspRoleList; - } - - - public String getPspRoleOid() { - - return pspRoleOid; - } - - public String getPspRoleName() { - - return pspRoleName; - } - - public String getPsd2RoleName() { - - return psd2RoleName; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSPRoles.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSPRoles.java deleted file mode 100644 index 3e6f4160..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/common/PSPRoles.java +++ /dev/null @@ -1,62 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor.common; - -import org.bouncycastle.asn1.ASN1Encodable; -import org.bouncycastle.asn1.DERSequence; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.List; - -/** - * Model to hold PSP roles. - */ -public class PSPRoles { - - private final List roles; - - public PSPRoles(List roles) { - - this.roles = roles; - } - - public static PSPRoles getInstance(Object obj) { - - if (obj instanceof PSPRoles) { - return (PSPRoles) obj; - } - - ASN1Encodable[] array = DERSequence.getInstance(obj).toArray(); - - List pspRoles = new ArrayList<>(); - List arrayList = Arrays.asList(array); - for (ASN1Encodable asn1Encodable : arrayList) { - pspRoles.addAll(PSPRole.getInstance(asn1Encodable)); - } - return new PSPRoles(pspRoles); - } - - - public List getRoles() { - - return roles; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/error/CertValidationErrors.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/error/CertValidationErrors.java deleted file mode 100644 index 8a1c9404..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/util/eidas/certificate/extractor/error/CertValidationErrors.java +++ /dev/null @@ -1,42 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor.error; - -/** - * CertValidationErrors class. - */ -public enum CertValidationErrors { - CERTIFICATE_INVALID("Content of the certificate is invalid."), - EXTENSION_NOT_FOUND("X509 V3 Extensions not found in the certificate."), - QCSTATEMENT_INVALID("Invalid QCStatement in the certificate."), - QCSTATEMENTS_NOT_FOUND("QCStatements not found in the certificate."), - PSD2_QCSTATEMENT_NOT_FOUND("No PSD2 QCStatement found in the certificate."); - - private String description; - - CertValidationErrors(String description) { - this.description = description; - } - - @Override - public String toString() { - return description; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/OpenBankingValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/OpenBankingValidator.java deleted file mode 100644 index 44c60e32..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/OpenBankingValidator.java +++ /dev/null @@ -1,75 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.validator; - -import java.util.Set; - -import javax.validation.ConstraintViolation; -import javax.validation.Validation; -import javax.validation.Validator; - - -/** - * Common Validator to validate objects based on annotation. - */ -public class OpenBankingValidator { - - public static final Validator FAIL_FAST_VALIDATOR = Validation - .byDefaultProvider().providerResolver(new OsgiServiceDiscoverer()) - .configure().addProperty("hibernate.validator.fail_fast", "true") - .buildValidatorFactory() - .getValidator(); - - private static volatile OpenBankingValidator instance; - - private OpenBankingValidator() { - } - - public static OpenBankingValidator getInstance() { - - if (instance == null) { - synchronized (OpenBankingValidator.class) { - if (instance == null) { - instance = new OpenBankingValidator(); - } - } - } - return instance; - } - - - /** - * Check for violations on request object. Stop at the first violation and return error. - * Validations are executed based on annotation in model of the class. - * - * @param object Object to be validated - * @return Error message if there is a violation, null otherwise - */ - public String getFirstViolation(Object object) { - - Set> violations = FAIL_FAST_VALIDATOR.validate(object); - return violations.stream().findFirst().map(ConstraintViolation::getMessage).orElse(null); - } - - public String getFirstViolation(Object object, Class validationGroup) { - - Set> violations = FAIL_FAST_VALIDATOR.validate(object, validationGroup); - return violations.stream().findFirst().map(ConstraintViolation::getMessage).orElse(null); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/OsgiServiceDiscoverer.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/OsgiServiceDiscoverer.java deleted file mode 100644 index 8ec01ecb..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/OsgiServiceDiscoverer.java +++ /dev/null @@ -1,38 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.validator; - -import org.hibernate.validator.HibernateValidator; - -import java.util.Collections; -import java.util.List; - -import javax.validation.ValidationProviderResolver; -import javax.validation.spi.ValidationProvider; - -/** - * To discover validation provider in OSGI environment. - */ -public class OsgiServiceDiscoverer implements ValidationProviderResolver { - - @Override - public List> getValidationProviders() { - return Collections.singletonList(new HibernateValidator()); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/annotation/RequiredParameter.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/annotation/RequiredParameter.java deleted file mode 100644 index bba01a25..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/annotation/RequiredParameter.java +++ /dev/null @@ -1,52 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.validator.annotation; - -import com.wso2.openbanking.accelerator.common.validator.impl.MandatoryParameterValidator; - -import java.lang.annotation.Documented; -import java.lang.annotation.ElementType; -import java.lang.annotation.Repeatable; -import java.lang.annotation.Retention; -import java.lang.annotation.Target; - -import javax.validation.Constraint; -import javax.validation.Payload; - -import static java.lang.annotation.RetentionPolicy.RUNTIME; - -/** - * Annotation to check required fields. - */ -@Target(ElementType.TYPE) -@Repeatable(RequiredParameters.class) -@Retention(RUNTIME) -@Documented -@Constraint(validatedBy = {MandatoryParameterValidator.class}) -public @interface RequiredParameter { - - String message() default "Mandatory parameter missing"; - - Class[] groups() default {}; - - Class[] payload() default {}; - - String param(); - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/annotation/RequiredParameters.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/annotation/RequiredParameters.java deleted file mode 100644 index c53c64d0..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/annotation/RequiredParameters.java +++ /dev/null @@ -1,36 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.validator.annotation; - -import java.lang.annotation.Documented; -import java.lang.annotation.ElementType; -import java.lang.annotation.Retention; -import java.lang.annotation.Target; - -import static java.lang.annotation.RetentionPolicy.RUNTIME; - -/** - * To enable repeated use of @RequiredParameter. - */ -@Target(ElementType.TYPE) -@Retention(RUNTIME) -@Documented -public @interface RequiredParameters { - RequiredParameter[] value(); -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/annotation/ValidAudience.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/annotation/ValidAudience.java deleted file mode 100644 index 49901df9..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/annotation/ValidAudience.java +++ /dev/null @@ -1,52 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.validator.annotation; - - -import com.wso2.openbanking.accelerator.common.validator.impl.AudienceValidator; - -import java.lang.annotation.Documented; -import java.lang.annotation.ElementType; -import java.lang.annotation.Retention; -import java.lang.annotation.Target; - -import javax.validation.Constraint; -import javax.validation.Payload; - -import static java.lang.annotation.RetentionPolicy.RUNTIME; - -/** - * An annotation to execute audience validation. - */ -@Target(ElementType.TYPE) -@Retention(RUNTIME) -@Documented -@Constraint(validatedBy = {AudienceValidator.class}) -public @interface ValidAudience { - - String message() default "Invalid audience given"; - - Class[] groups() default {}; - - Class[] payload() default {}; - - String audience() default "aud"; - - String clientId() default "clientId"; -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/annotation/ValidScopeFormat.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/annotation/ValidScopeFormat.java deleted file mode 100644 index 0f1f1820..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/annotation/ValidScopeFormat.java +++ /dev/null @@ -1,50 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.validator.annotation; - -import com.wso2.openbanking.accelerator.common.validator.impl.ScopeValidator; - -import java.lang.annotation.Documented; -import java.lang.annotation.ElementType; -import java.lang.annotation.Retention; -import java.lang.annotation.Target; - -import javax.validation.Constraint; -import javax.validation.Payload; - -import static java.lang.annotation.RetentionPolicy.RUNTIME; - -/** - * Annotation to check scope validation. - */ -@Target(ElementType.TYPE) -@Retention(RUNTIME) -@Documented -@Constraint(validatedBy = {ScopeValidator.class}) -public @interface ValidScopeFormat { - - String message() default "Invalid scope given in the request"; - - Class[] groups() default {}; - - Class[] payload() default {}; - - String scope() default "scope"; - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/impl/AudienceValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/impl/AudienceValidator.java deleted file mode 100644 index 97b4048d..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/impl/AudienceValidator.java +++ /dev/null @@ -1,102 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.validator.impl; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.ServiceProviderUtils; -import com.wso2.openbanking.accelerator.common.validator.annotation.ValidAudience; -import org.apache.commons.beanutils.BeanUtils; -import org.apache.commons.beanutils.NestedNullException; -import org.apache.commons.beanutils.PropertyUtilsBean; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.util.OAuth2Util; - -import java.lang.reflect.InvocationTargetException; -import java.util.List; - -import javax.validation.ConstraintValidator; -import javax.validation.ConstraintValidatorContext; - - -/** - * To validate if the audience is the same as the token issuer of the SP. - */ -public class AudienceValidator implements ConstraintValidator { - - private String audienceXpath; - private String clientIdXPath; - private static Log log = LogFactory.getLog(AudienceValidator.class); - - @Override - public void initialize(ValidAudience constraintAnnotation) { - - this.audienceXpath = constraintAnnotation.audience(); - this.clientIdXPath = constraintAnnotation.clientId(); - } - - @Override - public boolean isValid(Object object, ConstraintValidatorContext constraintValidatorContext) { - - try { - final Object audiences = new PropertyUtilsBean().getProperty(object, audienceXpath); - final String clientId = BeanUtils.getProperty(object, clientIdXPath); - - return audienceValidate(audiences, clientId); - - } catch (IllegalAccessException | InvocationTargetException | NoSuchMethodException | NestedNullException e) { - log.error("Error while resolving validation fields", e); - return false; - } - } - - public boolean audienceValidate(Object aud, String clientId) { - - String issuer; - - try { - issuer = OAuth2Util.getIdTokenIssuer(ServiceProviderUtils.getSpTenantDomain(clientId)); - } catch (IdentityOAuth2Exception | OpenBankingException e) { - log.error("Unable to retrieve the ID token issuer per tenant ", e); - return false; - } - - return validateAudience(issuer, aud); - } - - private boolean validateAudience(String currentAudience, Object audiencesObj) { - - if (audiencesObj instanceof List) { - List audiences = (List) audiencesObj; - for (Object a : audiences) { - if (a instanceof String) { - String aud = (String) a; - if (StringUtils.equals(currentAudience, aud)) { - return true; - } - } - } - } - - return false; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/impl/MandatoryParameterValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/impl/MandatoryParameterValidator.java deleted file mode 100644 index 9181e02c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/impl/MandatoryParameterValidator.java +++ /dev/null @@ -1,68 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.validator.impl; - -import com.wso2.openbanking.accelerator.common.validator.annotation.RequiredParameter; -import org.apache.commons.beanutils.NestedNullException; -import org.apache.commons.beanutils.PropertyUtilsBean; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.lang.reflect.InvocationTargetException; - -import javax.validation.ConstraintValidator; -import javax.validation.ConstraintValidatorContext; - -/** - * To validate if a mandatory parameter is in the object. - */ -public class MandatoryParameterValidator implements ConstraintValidator { - - private String paramXPath; - private static Log log = LogFactory.getLog(MandatoryParameterValidator.class); - - @Override - public void initialize(RequiredParameter constraintAnnotation) { - this.paramXPath = constraintAnnotation.param(); - } - - @Override - public boolean isValid(Object object, ConstraintValidatorContext constraintValidatorContext) { - - try { - - final Object parameterValue = new PropertyUtilsBean().getProperty(object, paramXPath); - - if (parameterValue instanceof Integer) { - return (Integer) parameterValue != 0; - - } else if (parameterValue instanceof Boolean) { - return (Boolean) parameterValue; - - } else { - return parameterValue != null; - } - - } catch (IllegalAccessException | InvocationTargetException | NoSuchMethodException | NestedNullException e) { - log.error("Mandatory parameter missing", e); - return false; - } - - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/impl/ScopeValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/impl/ScopeValidator.java deleted file mode 100644 index f3faf4b7..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/java/com/wso2/openbanking/accelerator/common/validator/impl/ScopeValidator.java +++ /dev/null @@ -1,72 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.validator.impl; - -import com.wso2.openbanking.accelerator.common.validator.annotation.ValidScopeFormat; -import org.apache.commons.beanutils.NestedNullException; -import org.apache.commons.beanutils.PropertyUtilsBean; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.lang.reflect.InvocationTargetException; -import java.util.ArrayList; -import java.util.Arrays; - -import javax.validation.ConstraintValidator; -import javax.validation.ConstraintValidatorContext; - -/** - * To validate if the scope is in the correct format. - */ -public class ScopeValidator implements ConstraintValidator { - - private String scopeXPath; - private static Log log = LogFactory.getLog(ScopeValidator.class); - - @Override - public void initialize(ValidScopeFormat constraintAnnotation) { - this.scopeXPath = constraintAnnotation.scope(); - } - - @Override - public boolean isValid(Object object, ConstraintValidatorContext constraintValidatorContext) { - - try { - final Object scope = new PropertyUtilsBean().getProperty(object, scopeXPath); - - return scopeValidate(scope); - - } catch (IllegalAccessException | InvocationTargetException | NoSuchMethodException | NestedNullException e) { - log.error("Error while resolving validation fields", e); - return false; - } - } - - boolean scopeValidate(Object scope) { - - if (scope instanceof String) { - ArrayList scopes = new ArrayList<>(Arrays.asList( - ((String) scope).replaceAll("\\s+", " ").trim().split(" "))); - - return scopes.contains("openid"); - } - - return false; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/resources/findbugs-exclude.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/resources/findbugs-exclude.xml deleted file mode 100644 index 3a2ff1ab..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/resources/findbugs-exclude.xml +++ /dev/null @@ -1,41 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/resources/findbugs-include.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/resources/findbugs-include.xml deleted file mode 100644 index 6773a2ce..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,35 +0,0 @@ - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/OBConfigParserTests.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/OBConfigParserTests.java deleted file mode 100644 index 44d6d038..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/OBConfigParserTests.java +++ /dev/null @@ -1,470 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.test; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingRuntimeException; -import com.wso2.openbanking.accelerator.common.test.util.CommonTestUtil; -import com.wso2.openbanking.accelerator.common.util.CarbonUtils; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -import java.io.File; -import java.util.Arrays; -import java.util.List; -import java.util.Map; - -/** - * Test class for Config Parser functionality. - */ -public class OBConfigParserTests { - - String absolutePathForTestResources; - - @BeforeClass - public void beforeClass() throws ReflectiveOperationException { - - //to execute util class initialization - new CarbonUtils(); - System.setProperty("some.property", "property.value"); - System.setProperty("carbon.home", "."); - CommonTestUtil.injectEnvironmentVariable("CARBON_HOME", "."); - String path = "src/test/resources"; - File file = new File(path); - absolutePathForTestResources = file.getAbsolutePath(); - } - - //Runtime exception is thrown here because carbon home is not defined properly for an actual carbon product - @Test(expectedExceptions = OpenBankingRuntimeException.class, priority = 1) - public void testConfigParserInitiationWithoutPath() { - - OpenBankingConfigParser openBankingConfigParser = OpenBankingConfigParser.getInstance(); - - } - - @Test(expectedExceptions = OpenBankingRuntimeException.class, priority = 2) - public void testRuntimeExceptionInvalidConfigFile() { - - String path = absolutePathForTestResources + "/open-banking-empty.xml"; - OpenBankingConfigParser openBankingConfigParser = OpenBankingConfigParser.getInstance(path); - - } - - @Test(expectedExceptions = OpenBankingRuntimeException.class, priority = 3) - public void testRuntimeExceptionNonExistentFile() { - - String path = absolutePathForTestResources + "/open-banking.xml" + "/value"; - OpenBankingConfigParser openBankingConfigParser = OpenBankingConfigParser.getInstance(path); - - } - - @Test(priority = 4) - public void testConfigParserInit() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - OpenBankingConfigParser openBankingConfigParser = OpenBankingConfigParser.getInstance(dummyConfigFile); - Assert.assertEquals(openBankingConfigParser.getConfiguration().get("Sample.OBHandler"), "DummyValue"); - Assert.assertEquals(openBankingConfigParser.getConfiguration().get("Sample.OBHandler2"), "property.value"); - Assert.assertNotNull(openBankingConfigParser.getConfiguration().get("Sample.OBHandler4")); - Map> openBankingExecutors = - OpenBankingConfigParser.getInstance().getOpenBankingExecutors(); - - openBankingExecutors.get("CustomType1").get(1). - equals("com.wso2.openbanking.accelerator.common.test.CustomHandler2"); - openBankingExecutors.get("CustomType2").get(1). - equals("com.wso2.openbanking.accelerator.common.test.CustomHandler"); - - Map> dcrRegistrationConfigs = OpenBankingConfigParser.getInstance() - .getOpenBankingDCRRegistrationParams(); - - dcrRegistrationConfigs.get("ParameterType").get("Required").equals("true"); - Assert.assertTrue(((List) dcrRegistrationConfigs.get("ParameterType").get("AllowedValues")).contains("Sample")); - - Map> stepsConfig = - OpenBankingConfigParser.getInstance().getConsentAuthorizeSteps(); - - stepsConfig.get("Persist").get(1).equals("com.wso2.openbanking.accelerator.common.test.CustomStep2"); - stepsConfig.get("Retrieve").get(1).equals("com.wso2.openbanking.accelerator.common.test.CustomStep1"); - - Map> apiMap = openBankingConfigParser.getAllowedAPIs(); - List roles = apiMap.get("DynamicClientRegistration"); - Assert.assertNotNull(apiMap); - Assert.assertNotNull(apiMap.get("DynamicClientRegistration")); - Assert.assertTrue(apiMap.get("AccountandTransactionAPI") instanceof List); - Assert.assertTrue(roles.contains("AISP")); - Assert.assertFalse(openBankingConfigParser.getServiceActivatorSubscribers().isEmpty()); - - Map openBankingEventExecutors = - OpenBankingConfigParser.getInstance().getOpenBankingEventExecutors(); - - openBankingEventExecutors.get(1). - equals("com.wso2.openbanking.accelerator.common.test.CustomEventExecutor1"); - openBankingEventExecutors.get(2). - equals("com.wso2.openbanking.accelerator.common.test.CustomEventExecutor2"); - - } - - @Test(priority = 5) - public void testSingleton() { - - OpenBankingConfigParser instance1 = OpenBankingConfigParser.getInstance(); - OpenBankingConfigParser instance2 = OpenBankingConfigParser.getInstance(); - Assert.assertEquals(instance2, instance1); - } - - @Test(priority = 6) - public void testCarbonPath() { - - String carbonConfigDirPath = CarbonUtils.getCarbonConfigDirPath(); - System.setProperty("carbon.config.dir.path", carbonConfigDirPath); - Assert.assertEquals(CarbonUtils.getCarbonConfigDirPath(), carbonConfigDirPath); - } - - @Test(priority = 7) - public void testGetDatasourceName() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - String config = OpenBankingConfigParser.getInstance(dummyConfigFile).getDataSourceName(); - Assert.assertNotNull(config); - } - - @Test(priority = 8) - public void testGetConnectionPoolMaxConnections() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - int maxConnections = OpenBankingConfigParser.getInstance(dummyConfigFile).getConnectionPoolMaxConnections(); - int maxConnectionsPerRoute = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getConnectionPoolMaxConnectionsPerRoute(); - - Assert.assertEquals(maxConnections, 1000); - Assert.assertEquals(maxConnectionsPerRoute, 500); - } - - @Test(priority = 8) - public void testConsentPeriodicalExpirationConfigs() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - String expirationCronValue = OpenBankingConfigParser.getInstance(dummyConfigFile). - getConsentExpiryCronExpression(); - String wordingForExpiredConsents = OpenBankingConfigParser.getInstance(dummyConfigFile). - getStatusWordingForExpiredConsents(); - String eligibleStatusesForConsentExpiry = OpenBankingConfigParser.getInstance(dummyConfigFile). - getEligibleStatusesForConsentExpiry(); - boolean periodicalJobEnabled = OpenBankingConfigParser.getInstance(dummyConfigFile) - .isConsentExpirationPeriodicalJobEnabled(); - - Assert.assertEquals(expirationCronValue, "0 0 0 * * ?"); - Assert.assertEquals(wordingForExpiredConsents, "Expired"); - Assert.assertEquals(periodicalJobEnabled, false); - Assert.assertEquals(eligibleStatusesForConsentExpiry, "authorised"); - } - - @Test (priority = 9) - public void testGetTrustStoreDynamicLoadingInterval() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - long dynamicLoadingInterval = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getTruststoreDynamicLoadingInterval(); - - Assert.assertEquals(dynamicLoadingInterval, Long.parseLong("86400")); - } - - @Test (priority = 10) - public void testGetAuthServletExtension() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - String authServletExtension = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getAuthServletExtension(); - - Assert.assertEquals(authServletExtension, "sampleServletExtension"); - } - - @Test (priority = 11) - public void testGetEmptyCibaServletExtension() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - String authServletExtension = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getCibaServletExtension(); - - Assert.assertEquals(authServletExtension, "sampleCIBAServletExtension"); - } - - @Test (priority = 12) - public void testGetJWKSConnectionTimeout() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - String connectionTimeOut = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getJWKSConnectionTimeOut(); - - Assert.assertEquals(connectionTimeOut, "1000"); - } - - @Test (priority = 13) - public void testGetConnectionVerificationTimeout() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - int connectionTimeOut = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getConnectionVerificationTimeout(); - - Assert.assertEquals(connectionTimeOut, 1000); - } - - @Test (priority = 14) - public void testGetJWKSReadTimeout() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - String connectionTimeOut = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getJWKSReadTimeOut(); - - Assert.assertEquals(connectionTimeOut, "3000"); - } - - @Test (priority = 15) - public void testGetSPMetadataFilterExtension() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - String connectionTimeOut = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getSPMetadataFilterExtension(); - - Assert.assertEquals(connectionTimeOut, "sampleSPMetadataFilterExtension"); - } - - @Test (priority = 16) - public void testGetEventNotificationTokenIssuer() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - String tokenIssuer = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getEventNotificationTokenIssuer(); - - Assert.assertEquals(tokenIssuer, "www.wso2.com"); - } - - @Test (priority = 17) - public void testGetNumberOfSetsToReturn() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - int maxEvents = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getNumberOfSetsToReturn(); - - Assert.assertEquals(maxEvents, 5); - } - - @Test (priority = 18) - public void testGetCommonCacheModifiedExpiryTime() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - String connectionTimeOut = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getCommonCacheModifiedExpiryTime(); - - Assert.assertEquals(connectionTimeOut, "60"); - } - - @Test (priority = 19) - public void testGetCommonCacheAccessExpiryTime() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - String connectionTimeOut = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getCommonCacheAccessExpiryTime(); - - Assert.assertEquals(connectionTimeOut, "60"); - } - - @Test (priority = 20) - public void testGetJwsRequestSigningAlgorithms() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - List algorithmConstraints = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getJwsRequestSigningAlgorithms(); - - Assert.assertEquals(algorithmConstraints, Arrays.asList("PS256")); - } - - @Test (priority = 20) - public void testIsJwsSignatureValidationEnabled() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - boolean isEnabled = OpenBankingConfigParser.getInstance(dummyConfigFile) - .isJwsSignatureValidationEnabled(); - - Assert.assertFalse(isEnabled); - } - - @Test (priority = 21) - public void testGetOBIdnRetrieverSigningCertificateAlias() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - String certificateAlias = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getOBIdnRetrieverSigningCertificateAlias(); - - Assert.assertEquals(certificateAlias, "wso2carbon"); - } - - @Test (priority = 22) - public void testOBIdnRetrieverSandboxSigningCertificateAlias() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - String certificateAlias = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getOBIdnRetrieverSandboxSigningCertificateAlias(); - - Assert.assertEquals(certificateAlias, "wso2carbon-sandbox"); - } - - @Test (priority = 23) - public void testGetOBIdnRetrieverSigningCertificateKid() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - String certificateAlias = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getOBIdnRetrieverSigningCertificateKid(); - - Assert.assertEquals(certificateAlias, "1234"); - } - - @Test (priority = 24) - public void testGetJwsResponseSigningAlgorithm() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - String certificateAlias = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getJwsResponseSigningAlgorithm(); - - Assert.assertEquals(certificateAlias, "PS256"); - } - - @Test (priority = 25) - public void testIsJwsResponseSigningEnabled() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - boolean isEnabled = OpenBankingConfigParser.getInstance(dummyConfigFile) - .isJwsResponseSigningEnabled(); - - Assert.assertEquals(isEnabled, false); - } - - @Test (priority = 26) - public void testGetJwksRetrieverSizeLimit() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - String sizeLimit = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getJwksRetrieverSizeLimit(); - - Assert.assertEquals(sizeLimit, "51200"); - } - - @Test (priority = 27) - public void testGetJwksRetrieverConnectionTimeout() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - String timeout = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getJwksRetrieverConnectionTimeout(); - - Assert.assertEquals(timeout, "2000"); - } - - @Test (priority = 28) - public void testGetJwksRetrieverReadTimeout() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - String timeout = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getJwksRetrieverReadTimeout(); - - Assert.assertEquals(timeout, "2000"); - } - - @Test (priority = 29) - public void testIsToeClaimIncluded() { - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - Boolean isToeClaimIncluded = OpenBankingConfigParser.getInstance(dummyConfigFile).isToeClaimIncluded(); - - Assert.assertTrue(isToeClaimIncluded); - } - - @Test (priority = 30) - public void testWithRetentionConfigs() { - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - OpenBankingConfigParser openBankingConfigParser = OpenBankingConfigParser.getInstance(dummyConfigFile); - - Assert.assertTrue(openBankingConfigParser.isConsentDataRetentionEnabled()); - Assert.assertTrue(openBankingConfigParser.isRetentionDataDBSyncEnabled()); - Assert.assertNotNull(openBankingConfigParser.getRetentionDataDBSyncCronExpression()); - Assert.assertNotNull(openBankingConfigParser.getRetentionDataSourceName()); - Assert.assertEquals(openBankingConfigParser.getRetentionDataSourceConnectionVerificationTimeout(), 1); - } - - @Test (priority = 31) - public void testIsDisputeResolutionEnabled() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - boolean isEnabled = OpenBankingConfigParser.getInstance(dummyConfigFile) - .isDisputeResolutionEnabled(); - - Assert.assertTrue(isEnabled); - } - - @Test (priority = 32) - public void testIsNonErrorDisputeDataPublishingEnabled() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - boolean isEnabled = OpenBankingConfigParser.getInstance(dummyConfigFile) - .isNonErrorDisputeDataPublishingEnabled(); - - Assert.assertTrue(isEnabled); - } - - @Test (priority = 33) - public void testRealtimeEventNotificationConfigs() { - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - OpenBankingConfigParser openBankingConfigParser = OpenBankingConfigParser.getInstance(dummyConfigFile); - - Assert.assertTrue(openBankingConfigParser.isRealtimeEventNotificationEnabled()); - Assert.assertEquals(openBankingConfigParser.getRealtimeEventNotificationSchedulerCronExpression(), - "0 0/1 0 ? * * *"); - Assert.assertEquals(openBankingConfigParser.getRealtimeEventNotificationTimeoutInSeconds(), 60); - Assert.assertEquals(openBankingConfigParser.getRealtimeEventNotificationMaxRetries(), 5); - Assert.assertEquals(openBankingConfigParser.getRealtimeEventNotificationInitialBackoffTimeInSeconds(), - 60); - Assert.assertEquals(openBankingConfigParser.getRealtimeEventNotificationBackoffFunction(), "EX"); - Assert.assertEquals(openBankingConfigParser.getRealtimeEventNotificationCircuitBreakerOpenTimeoutInSeconds(), - 600); - Assert.assertEquals(openBankingConfigParser.getEventNotificationThreadpoolSize(), 20); - Assert.assertEquals(openBankingConfigParser.getRealtimeEventNotificationRequestGenerator(), - "com.wso2.openbanking.accelerator.event.notifications.service.realtime" + - ".service.DefaultRealtimeEventNotificationRequestGenerator"); - - } - - @Test (priority = 34) - public void testIsConsentAmendmentHistoryEnabled() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - boolean isEnabled = OpenBankingConfigParser.getInstance(dummyConfigFile) - .isConsentAmendmentHistoryEnabled(); - - Assert.assertTrue(isEnabled); - } - - @Test (priority = 35) - public void testGetOBKeyManagerExtensionImpl() { - - String dummyConfigFile = absolutePathForTestResources + "/open-banking.xml"; - String className = OpenBankingConfigParser.getInstance(dummyConfigFile) - .getOBKeyManagerExtensionImpl(); - - Assert.assertEquals(className, "com.wso2.openbanking.accelerator.keymanager.OBKeyManagerImpl"); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/config/TextFileReaderTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/config/TextFileReaderTest.java deleted file mode 100644 index 0884d150..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/config/TextFileReaderTest.java +++ /dev/null @@ -1,73 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.test.config; - -import com.wso2.openbanking.accelerator.common.config.TextFileReader; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.junit.Assert; -import org.testng.annotations.Test; - -import java.io.IOException; - -/** - * Text file reader test. - */ -public class TextFileReaderTest { - - private static final Log logger = LogFactory.getLog(TextFileReader.class); - - @Test - public void testReadFile() { - - try { - TextFileReader textFileReader = TextFileReader.getInstance(); - textFileReader.setDirectoryPath("src/test/resources"); - String file = textFileReader.readFile("testFile.js"); - Assert.assertNotNull(file); - } catch (IOException e) { - logger.error("Error while reading file", e); - } - } - - @Test - public void testRetrieveFileFromMap() { - - try { - TextFileReader textFileReader = TextFileReader.getInstance(); - String file = textFileReader.readFile("testFile.js"); - Assert.assertNotNull(file); - } catch (IOException e) { - logger.error("Error while reading file", e); - } - } - - @Test(description = "test whether empty string is returned when trying to retrieve non existing file") - public void testRetrieveWrongFile() { - - try { - TextFileReader textFileReader = TextFileReader.getInstance(); - textFileReader.setDirectoryPath("src/test/resources"); - String file = textFileReader.readFile("testFileOne.js"); - Assert.assertEquals(file, ""); - } catch (IOException e) { - logger.error("Error while reading file", e); - } - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/event/executor/OBEventExecutorTests.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/event/executor/OBEventExecutorTests.java deleted file mode 100644 index 9910e26a..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/event/executor/OBEventExecutorTests.java +++ /dev/null @@ -1,92 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.test.event.executor; - -import com.wso2.openbanking.accelerator.common.event.executor.DefaultOBEventExecutor; -import com.wso2.openbanking.accelerator.common.event.executor.OBEventQueue; -import com.wso2.openbanking.accelerator.common.event.executor.model.OBEvent; -import com.wso2.openbanking.accelerator.common.internal.OpenBankingCommonDataHolder; -import org.apache.logging.log4j.LogManager; -import org.apache.logging.log4j.Logger; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -import java.io.ByteArrayOutputStream; -import java.io.PrintStream; -import java.util.HashMap; -import java.util.Map; - -import static org.powermock.api.mockito.PowerMockito.mock; -import static org.powermock.api.mockito.PowerMockito.mockStatic; -import static org.powermock.api.mockito.PowerMockito.when; - -/** - * Test for Open Banking event executor. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({OpenBankingCommonDataHolder.class}) -public class OBEventExecutorTests extends PowerMockTestCase { - - private static ByteArrayOutputStream outContent; - private static Logger logger = null; - private static PrintStream printStream; - - @BeforeClass - public void beforeTests() { - - outContent = new ByteArrayOutputStream(); - printStream = new PrintStream(outContent); - System.setOut(printStream); - logger = LogManager.getLogger(OBEventExecutorTests.class); - } - - @Test - public void testAddingDataToQueue() { - - outContent.reset(); - Map configs = new HashMap<>(); - configs.put("Event.WorkerThreadCount", "3"); - configs.put("Event.QueueSize", "10"); - - OBEvent obEvent = new OBEvent("revoked", new HashMap<>()); - - OpenBankingCommonDataHolder openBankingCommonDataHolderMock = mock(OpenBankingCommonDataHolder.class); - mockStatic(OpenBankingCommonDataHolder.class); - when(OpenBankingCommonDataHolder.getInstance()).thenReturn(openBankingCommonDataHolderMock); - - when(openBankingCommonDataHolderMock.getOBEventQueue()).thenReturn(new OBEventQueue(Integer - .parseInt(configs.get("Event.QueueSize").toString()), Integer.parseInt(configs - .get("Event.WorkerThreadCount").toString()))); - - Map obEventExecutors = new HashMap<>(); - obEventExecutors.put(1, DefaultOBEventExecutor.class.getName()); - when(openBankingCommonDataHolderMock.getOBEventExecutors()).thenReturn(obEventExecutors); - - - OBEventQueue obEventQueue = openBankingCommonDataHolderMock.getOBEventQueue(); - - obEventQueue.put(obEvent); - // there should be an error log or a warning if the queue is full. - Assert.assertTrue(outContent.toString().isEmpty()); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/CertificateUtilsTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/CertificateUtilsTest.java deleted file mode 100644 index 3cfb2c49..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/CertificateUtilsTest.java +++ /dev/null @@ -1,86 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.test.util; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.CertificateUtils; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -import java.security.cert.X509Certificate; - -/** - * Certificate Util test class. - */ -public class CertificateUtilsTest { - - private X509Certificate expiredX509Cert; - - @BeforeClass - public void init() throws OpenBankingException { - this.expiredX509Cert = CommonTestUtil.getExpiredSelfCertificate(); - } - - @Test(description = "when valid transport cert, return x509 certificate") - public void testParseCertificate() throws OpenBankingException { - Assert.assertNotNull(CertificateUtils - .parseCertificate(CommonTestUtil.TEST_CLIENT_CERT)); - } - - @Test (expectedExceptions = OpenBankingException.class) - public void testParseCertificateWithInvalidCert() throws OpenBankingException { - Assert.assertNull(CertificateUtils - .parseCertificate("-----INVALID CERTIFICATE-----")); - } - - @Test - public void testParseCertificateWithInvalidBase64CharactersCert() throws OpenBankingException { - Assert.assertNotNull(CertificateUtils - .parseCertificate(CommonTestUtil.WRONGLY_FORMATTED_CERT)); - } - - @Test - public void testParseCertificateWithEmptyCert() throws OpenBankingException { - Assert.assertNull(CertificateUtils - .parseCertificate("")); - } - - @Test(description = "when certificate expired, return true") - public void testIsExpiredWithExpiredCert() throws OpenBankingException { - X509Certificate testCert = CertificateUtils - .parseCertificate(CommonTestUtil.EXPIRED_SELF_CERT); - Assert.assertNotNull(testCert); - Assert.assertTrue(CommonTestUtil.hasExpired(testCert)); - } - - @Test(description = "when valid certificate, return false") - public void testIsExpired() throws OpenBankingException { - X509Certificate testCert = CertificateUtils - .parseCertificate(CommonTestUtil.TEST_CLIENT_CERT); - Assert.assertNotNull(testCert); - Assert.assertFalse(CommonTestUtil.hasExpired(testCert)); - } - - @Test - public void testIsCertValidWithExpiredCert() { - Assert.assertTrue(CertificateUtils.isExpired(expiredX509Cert)); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/CommonTestUtil.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/CommonTestUtil.java deleted file mode 100644 index 0d35ddec..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/CommonTestUtil.java +++ /dev/null @@ -1,219 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.test.util; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.CertificateUtils; - -import java.io.ByteArrayInputStream; -import java.lang.reflect.Field; -import java.security.cert.CertificateException; -import java.security.cert.CertificateFactory; -import java.security.cert.X509Certificate; -import java.util.Base64; -import java.util.Map; -import java.util.Optional; - -/** - * Utility class for tests. - */ -public class CommonTestUtil { - - public static final String BEGIN_CERT = "-----BEGIN CERTIFICATE-----"; - public static final String END_CERT = "-----END CERTIFICATE-----"; - public static final String X509_CERT_INSTANCE_NAME = "X.509"; - private static X509Certificate expiredSelfCertificate = null; - public static final String EIDAS_CERT = "-----BEGIN CERTIFICATE-----" + - "MIIEjDCCA3SgAwIBAgILAKTSmx6PZuerUKkwDQYJKoZIhvcNAQELBQAwSDELMAkG" + - "A1UEBhMCREUxDDAKBgNVBAoMA0JEUjERMA8GA1UECwwISVQgLSBEZXYxGDAWBgNV" + - "BAMMD1BTRDIgVGVzdCBTdWJDQTAeFw0xODEyMTIxNDIzMjBaFw0xOTA2MTAxNDIz" + - "MjBaMIHFMQswCQYDVQQGEwJERTEkMCIGA1UECgwbSGFuc2VhdGljIEJhbmsgR21i" + - "SCAmIENvIEtHMRAwDgYDVQQHDAdIYW1idXJnMRAwDgYDVQQIDAdoYW1idXJnMSAw" + - "HgYDVQQJDBdCcmFtZmVsZGVyIENoYXVzc2VlIDEwMTEdMBsGA1UEAwwUd3d3Lmhh" + - "bnNlYXRpY2JhbmsuZGUxGzAZBgNVBGETElBTRERFLUJBRklOLTEyMzQ1NjEOMAwG" + - "A1UEERMFMjIxNzcwggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQCq+FfA" + - "Yg8kcypd0HWhZqW8vtm/1KV+CVbertirwc3nbeufIha82kmJr/0ybxPhdJuPSXPA" + - "9YPnGyg4aHBjwGWJhI5sMxynVB6+JrENu1wp4MSUr6BUrNvpiYo7uU2mEe9jEheQ" + - "vqQ45vPw1f2B1YSZgQ5OaSAeLnOqjwDoHseT+mNSJbRznJguwb7hLl78VCuJeYrB" + - "8E1AfJrrKWAVov6TldInq8xP47kspJCheIrEMZskehvuvn11ir24CnTrFe6G4B2v" + - "e5VDR40YbYGD/yD/m8Y2/Y5BGZGw7ty5RqS0ubB99lRkc13KpkAEI45QWQyXVTIF" + - "cORodKZHdoSwcORZAgMBAAGjgfgwgfUwgYgGCCsGAQUFBwEDBHwwejB4BgYEAIGY" + - "JwIwbjA5MBEGBwQAgZgnAQQMBlBTUF9JQzARBgcEAIGYJwECDAZQU1BfUEkwEQYH" + - "BACBmCcBAwwGUFNQX0FJDCdGZWRlcmFsIEZpbmFuY2lhbCBTdXBlcnZpc29yeSBB" + - "dXRob3JpdHkMCERFLUJBRklOMAsGA1UdDwQEAwIFoDAdBgNVHSUEFjAUBggrBgEF" + - "BQcDAQYIKwYBBQUHAwIwGwYDVR0RBBQwEoIQaGFuc2VhdGljYmFuay5kZTAfBgNV" + - "HSMEGDAWgBRczgCARJu0XDNe5JrFOPI4CIgFojANBgkqhkiG9w0BAQsFAAOCAQEA" + - "l8IGrflLcPpmpKIxlmASRtPk96Dh5E3Is2dDCO/yiv2TKoBjyGRYLSPKD7mS1YBr" + - "g2/l2yPK6+5V5n/pIZ3V8SezfFlzs65i1Jc9XwB/236BjgRMXuAMJmB0Rjo31Nd0" + - "o/FAOEvIpNh0+GVz6SnY07qry2BWCYAqyHSih20Wjj6yOHHvQtXRaQijQSwo5WGS" + - "grHH0Thh+MBlyc8iNajrrNxKRYWyXrpGpukMOR4CYWp2CeC22+zLQIEI9gcnl0hr" + - "/yBbG/db3ZujZvjW34KreOkxjzc+/bhQlubv7KrSruj1OoDzq8e+ELCoNII2JU3h" + - "R783WZESc2tbq1LYOH5wNg==" + - "-----END CERTIFICATE-----"; - public static final String TEST_CLIENT_CERT = "-----BEGIN CERTIFICATE-----" + - "MIIFLTCCBBWgAwIBAgIEWcbiiDANBgkqhkiG9w0BAQsFADBTMQswCQYDVQQGEwJH" + - "QjEUMBIGA1UEChMLT3BlbkJhbmtpbmcxLjAsBgNVBAMTJU9wZW5CYW5raW5nIFBy" + - "ZS1Qcm9kdWN0aW9uIElzc3VpbmcgQ0EwHhcNMjMxMTE1MDUxMDA4WhcNMjQxMjE1" + - "MDU0MDA4WjBhMQswCQYDVQQGEwJHQjEUMBIGA1UEChMLT3BlbkJhbmtpbmcxGzAZ" + - "BgNVBAsTEjAwMTU4MDAwMDFIUVFyWkFBWDEfMB0GA1UEAxMWakZRdVE0ZVFiTkNN" + - "U3FkQ29nMjFuRjCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEBAKi36HD0" + - "prx1N3pfafJc6pSMg0i0jOiQrt+WZ6GphKUhA2JrbBxWTvabX1Q7hifl5UbkBOkn" + - "n3SCyqBplPSqksYLdPSckb2J7UVLj76O7RkELoFD+vSrQ8FWEn8lXv4UiS8ylvlr" + - "IPa8+pVtAwOuI3/YFmqB1lcHgBI8EELDUWHXSp3OhkcKjrm76t1sQz0tsmi5aHZR" + - "3FXb0dACQO+dGakIerqzUmAdccLtWuSJMT+b7c7IDDm/IR/MV7iol04yeBkOlWre" + - "IKOqHU16PI3N/PSMEt96JAUKZ/n0VC0x4YHaBYDX27sf4Gypubgmkcd4IbG/8XEu" + - "Hq3Ik84oEpP5QEUCAwEAAaOCAfkwggH1MA4GA1UdDwEB/wQEAwIGwDAVBgNVHSUE" + - "DjAMBgorBgEEAYI3CgMMMIHgBgNVHSAEgdgwgdUwgdIGCysGAQQBqHWBBgFkMIHC" + - "MCoGCCsGAQUFBwIBFh5odHRwOi8vb2IudHJ1c3Rpcy5jb20vcG9saWNpZXMwgZMG" + - "CCsGAQUFBwICMIGGDIGDVXNlIG9mIHRoaXMgQ2VydGlmaWNhdGUgY29uc3RpdHV0" + - "ZXMgYWNjZXB0YW5jZSBvZiB0aGUgT3BlbkJhbmtpbmcgUm9vdCBDQSBDZXJ0aWZp" + - "Y2F0aW9uIFBvbGljaWVzIGFuZCBDZXJ0aWZpY2F0ZSBQcmFjdGljZSBTdGF0ZW1l" + - "bnQwbQYIKwYBBQUHAQEEYTBfMCYGCCsGAQUFBzABhhpodHRwOi8vb2IudHJ1c3Rp" + - "cy5jb20vb2NzcDA1BggrBgEFBQcwAoYpaHR0cDovL29iLnRydXN0aXMuY29tL29i" + - "X3BwX2lzc3VpbmdjYS5jcnQwOgYDVR0fBDMwMTAvoC2gK4YpaHR0cDovL29iLnRy" + - "dXN0aXMuY29tL29iX3BwX2lzc3VpbmdjYS5jcmwwHwYDVR0jBBgwFoAUUHORxiFy" + - "03f0/gASBoFceXluP1AwHQYDVR0OBBYEFKjCef/JxD+ND9eSb7hQlmEhSxUqMA0G" + - "CSqGSIb3DQEBCwUAA4IBAQCnKH9FdLmJMruX2qfbrpT0qaV8bP7xa9UDRYSMsAWC" + - "2kqCxs8CJmARt5+xsxBW6P65+mkLS2vXgQl7J8RTMiQVnHJvvNaldYnV6odsYOqv" + - "v+vGib8Qe0gKWSjih+Gd1Ct4UQFtn6P3ph+6OBB0OieZb7DYXqPJrX5UlG7K2fQ4" + - "0MdFgBdeQZ3iNkXi43UIrQ5cF4cjYavmEFRmYeHya8AKfNCiWly15mNazW/X6SWf" + - "7pz+yk/l+gBv0wm3QT7ANXGf8izgoh6T5fmixPXSbdn8RUIV0kXp2TRRZ+CYUWBP" + - "Jc3PvRXiiEEo2eHLXfEHG2jzrt1iKnjk6hzuC1hUzK0t" + - "-----END CERTIFICATE-----"; - public static final String EXPIRED_SELF_CERT = "-----BEGIN CERTIFICATE-----" + - "MIIDiTCCAnGgAwIBAgIENx3SZjANBgkqhkiG9w0BAQsFADB1MQswCQYDVQQGEwJs" + - "azEQMA4GA1UECBMHd2VzdGVybjEQMA4GA1UEBxMHY29sb21ibzENMAsGA1UEChME" + - "d3NvMjEUMBIGA1UECxMLb3BlbmJhbmtpbmcxHTAbBgNVBAMTFG9wZW5iYW5raW5n" + - "LndzbzIuY29tMB4XDTIwMDMwMTEyMjE1MVoXDTIwMDUzMDEyMjE1MVowdTELMAkG" + - "A1UEBhMCbGsxEDAOBgNVBAgTB3dlc3Rlcm4xEDAOBgNVBAcTB2NvbG9tYm8xDTAL" + - "BgNVBAoTBHdzbzIxFDASBgNVBAsTC29wZW5iYW5raW5nMR0wGwYDVQQDExRvcGVu" + - "YmFua2luZy53c28yLmNvbTCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEB" + - "AKWMb1mhSthxi5vmQcvEnt0rauYv8uFWjGyiuCkk5wQbArybGXyC8rrZf5qNNY4s" + - "RG2+Yimxph2Z8MWWPFBebTIABPuRcVDquX7fL4+8FZJTH3JLwfT+slunAA4473mZ" + - "9s2fAVu6CmQf1V09+fEbMGI9WWh53g19wg5WdlToOX4g5lh4QtGRpbWpEWaYrKzS" + - "B5EWOUI7lroFtv6s9OpEO59VAkXWKUbT98T8TCYqiDH+nMy3k+GbVawxXeHYHQr+" + - "XlbcChPaCwhMXspqKG49xaJmrOuRMoAWCBGUW8r2RDhQ+FP5V/sTRMqKmBv9gTe6" + - "RJwoKPlDt+0aX9vaFjKpjPcCAwEAAaMhMB8wHQYDVR0OBBYEFGH0gyeHIz1+ONGI" + - "PuGnAhrS3apoMA0GCSqGSIb3DQEBCwUAA4IBAQCVEakh1SLnZOz2IK0ISbAV5UBb" + - "nerLNDl+X+YSYsCQM1SBcXDjlkSAeP3ErJEO3RW3wdRQjLRRHomwSCSRE84SUfSL" + - "VPIbeR7jm4sS9x5rnlGF6iqhYh2MlZD/hFxdrGoYv8g/JN4FFFMXRmmaQ8ouYJwc" + - "4ZoxRdCXszeI5Zp2+b14cs/nf4geYliHtcDr/w7fkvQ0hn+c1lTihbW0/eE32aUK" + - "SULAmjx0sCDfDAQItP79CC7jCW0TFN0CMORw/+fzp/dnVboSZ2MgcuRIH1Ez+6/1" + - "1QJD2SrkkaRSEaXI6fe9jgHVhnqK9V3y3WAuzEKjaKw6jV8BjkXAA4dQj1Re" + - "-----END CERTIFICATE-----"; - - public static final String WRONGLY_FORMATTED_CERT = "-----BEGIN CERTIFICATE-----\n" + - "MIIFljCCA36gAwIBAgIJAN5zDsVzPq0aMA0GCSqGSIb3DQEBBQUAMIGsMQswCQYD " + - "VQQGEwJMSzELMAkGA1UECAwCV1AxDDAKBgNVBAcMA0NPTDEaMBgGA1UECgwRV1NP" + - "MiAoVUspIExJTUlURUQxFDASBgNVBAsMC09wZW5CYW5raW5nMS4wLAYDVQQDDCVP" + - "cGVuQmFua2luZyBQcmUtUHJvZHVjdGlvbiBJc3N1aW5nIENBMSAwHgYJKoZIhvcN" + - "AQkBFhFtYWxzaGFuaUB3c28yLmNvbTAeFw0yMjAxMTgwNzI3NDJaFw0yNDAxMTgw" + - "NzI3NDJaMHMxCzAJBgNVBAYTAkdCMRowGAYDVQQKDBFXU08yIChVSykgTElNSVRF" + - "RDErMCkGA1UEYQwiUFNER0ItT0ItVW5rbm93bjAwMTU4MDAwMDFIUVFyWkFBWDEb" + - "MBkGA1UEAwwSMDAxNTgwMDAwMUhRUXJaQUFYMIIBIjANBgkqhkiG9w0BAQEFAAOC" + - "AQ8AMIIBCgKCAQEA59+TouW8sLFWk7MUht40v+DDglinjL2qmQ+wP3YNtvza/7Ue" + - "KZ+gWw92jd0v99xZz7c5KOgtTgctAmIU1qjGLwzHzn/fl/ZrO4spGLIbU7RwGHA7" + - "BSpB4k0vGdpCBigaaILHhBrAczDJ1BLYMS4lg69+6fYTeY2s0Khv92NWl8TXorAH" + - "W0D8KrbZ3chWIynZamNu8KN6s+GL5jyu6pzJpXVNOXiUdRr4U9fLctw7qPw4RbBM" + - "edXohmVFwMTQ7lMKax+wHOjfQDQW7KuZxRRYiUqB3hjyhrKlIpjjWtnxLclymTAI" + - "TRMqFlH8KFq/rVBGQ8F3SnDp90E25RbSWdfNRwIDAQABo4HyMIHvMA4GA1UdDwEB" + - "/wQEAwIHgDAdBgNVHSUEFjAUBggrBgEFBQcDAgYIKwYBBQUHAwEwHQYDVR0OBBYE" + - "FNxNxhzaeU3VdIMlXkNiYbnjheOnMIGeBggrBgEFBQcBAwSBkTCBjjATBgYEAI5G" + - "AQYwCQYHBACORgEGAzB3BgYEAIGYJwIwbTBGMEQGBwQAgZgnAQEMBlBTUF9BUwYH" + - "BACBmCcBAgwGUFNQX1BJBgcEAIGYJwEDDAZQU1BfQUkGBwQAgZgnAQQMBlBTUF9J" + - "QwwbRmluYW5jaWFsIENvbmR1Y3QgQXV0aG9yaXR5DAZHQi1GQ0EwDQYJKoZIhvcN" + - "AQEFBQADggIBABBM63bCwANVRR44wFCZysbppYAT4mms3dUqoP3XCUXaO3+7zNWa" + - "siZ90cje3fuiTD5SAyykm/I/mlgVx92ZbYFW0VG7IVkuC7Fid5iPywHX7Bm1xmEY" + - "bL1AtAm4sBzE1Kw5dnB1L30do7sp9fuJCdom5/fhrh2GyLBd0iA62qQ+F9uALrC0" + - "bub0KnGaEf9g1UltgxuqguoYoHb46ICJ03kMGZMC5BcjDDEbDQQ3kT+g9evaBUBm" + - "3A3cNJURF7/07iLEfHNYrMxDLIw6aC4svbcx+IquO81xpTCefhTU4UFSLN1/DXWW" + - "qrjCqkvHE53mb33QCXmnsooTP8pABG2q2+w5EC9yeX6Fln6M8VwZL5P2stELWXZE" + - "876kCo0LkmoP3s6Z62bF4u9hJvM9mQRvmDVqN2Y7eLMty4qmGEmAYYiHOG+FXNKo" + - "io9MXbB3B7tdeM4g2HlQGfRIrTrfAOu2cH1l1ZwHZgx7oCXN1nuZgE3r07kJx4Bn" + - "DXCRpXoZq4pB3AlzcWEPh51/SS8Wsz52CNSDGoMB7HPkNnoDrYoibb1LFrOwJ3IM" + - "VUKCSnt1QdnrKtMVMTd0iI4uk7kCKt7QFeiizN+oW6BI/MNm6mHEWd9CKWmrZT56" + - "wU3ZM7vgwugq9tAs+oi8Lf3ZODuXAsiSpgcd6dceatoqeyB4E+6kp0Ge" + - "-----END CERTIFICATE-----"; - - public static void injectEnvironmentVariable(String key, String value) - throws ReflectiveOperationException { - - Class processEnvironment = Class.forName("java.lang.ProcessEnvironment"); - - Field unmodifiableMapField = getAccessibleField(processEnvironment, "theUnmodifiableEnvironment"); - Object unmodifiableMap = unmodifiableMapField.get(null); - injectIntoUnmodifiableMap(key, value, unmodifiableMap); - - Field mapField = getAccessibleField(processEnvironment, "theEnvironment"); - Map map = (Map) mapField.get(null); - map.put(key, value); - } - - private static Field getAccessibleField(Class clazz, String fieldName) - throws NoSuchFieldException { - - Field field = clazz.getDeclaredField(fieldName); - field.setAccessible(true); - return field; - } - - public static synchronized X509Certificate getExpiredSelfCertificate() - throws OpenBankingException { - if (expiredSelfCertificate == null) { - expiredSelfCertificate = CertificateUtils.parseCertificate(EXPIRED_SELF_CERT); - } - return expiredSelfCertificate; - } - - private static void injectIntoUnmodifiableMap(String key, String value, Object map) - throws ReflectiveOperationException { - - Class unmodifiableMap = Class.forName("java.util.Collections$UnmodifiableMap"); - Field field = getAccessibleField(unmodifiableMap, "m"); - Object obj = field.get(map); - ((Map) obj).put(key, value); - } - - public static Optional parseTransportCert(String strTransportCert) throws CertificateException { - - // decoding pem formatted transport cert - byte[] decodedTransportCert = Base64.getDecoder().decode(strTransportCert - .replace(BEGIN_CERT, "").replace(END_CERT, "")); - - X509Certificate transportCert = (X509Certificate) CertificateFactory.getInstance(X509_CERT_INSTANCE_NAME) - .generateCertificate(new ByteArrayInputStream(decodedTransportCert)); - - return Optional.ofNullable(transportCert); - } - - /** - * Test util method to check cert expiry. - * - * @param peerCertificate - * @return - */ - public static boolean hasExpired(X509Certificate peerCertificate) { - try { - peerCertificate.checkValidity(); - } catch (CertificateException e) { - return true; - } - return false; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/HTTPClientUtilsTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/HTTPClientUtilsTest.java deleted file mode 100644 index c58d9484..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/HTTPClientUtilsTest.java +++ /dev/null @@ -1,71 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.test.util; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.HTTPClientUtils; -import org.apache.http.conn.ssl.SSLSocketFactory; -import org.testng.Assert; -import org.testng.annotations.Test; - -import java.io.File; - -/** - * Http Client util test. - */ -public class HTTPClientUtilsTest { - - String path = "src/test/resources"; - File file = new File(path); - String absolutePathForTestResources = file.getAbsolutePath(); - - @Test - public void testLoadKeystore() throws OpenBankingException { - - Assert.assertNotNull(HTTPClientUtils.loadKeyStore(absolutePathForTestResources + "/wso2carbon.jks", - "wso2carbon")); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testLoadInvalidKeystore() throws OpenBankingException { - - HTTPClientUtils.loadKeyStore(absolutePathForTestResources + "/wso2carbon2.jks", - "wso2carbon"); - } - - @Test - public void testHostNameVerifier() throws OpenBankingException { - - Assert.assertEquals(HTTPClientUtils.getX509HostnameVerifier(), - SSLSocketFactory.BROWSER_COMPATIBLE_HOSTNAME_VERIFIER); - - System.setProperty(HTTPClientUtils.HOST_NAME_VERIFIER, - String.valueOf(SSLSocketFactory.ALLOW_ALL_HOSTNAME_VERIFIER)); - - Assert.assertEquals(HTTPClientUtils.getX509HostnameVerifier(), - SSLSocketFactory.BROWSER_COMPATIBLE_HOSTNAME_VERIFIER); - - System.setProperty(HTTPClientUtils.HOST_NAME_VERIFIER, - String.valueOf(SSLSocketFactory.STRICT_HOSTNAME_VERIFIER)); - - Assert.assertEquals(HTTPClientUtils.getX509HostnameVerifier(), - SSLSocketFactory.STRICT_HOSTNAME_VERIFIER); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/JWTUtilsTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/JWTUtilsTest.java deleted file mode 100644 index 0658d00c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/JWTUtilsTest.java +++ /dev/null @@ -1,72 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.test.util; - -import com.wso2.openbanking.accelerator.common.test.util.testutils.JWTUtilsTestDataProvider; -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import org.testng.Assert; -import org.testng.annotations.Test; - -import java.text.ParseException; -import java.util.Date; - -/** - * Test class for Unit Testing JWTUtils. - */ -public class JWTUtilsTest { - - @Test(dataProviderClass = JWTUtilsTestDataProvider.class, dataProvider = "jwtStrings") - public void testIsJWT(String jwtString, boolean expected) { - - Assert.assertEquals(JWTUtils.isValidJWSFormat(jwtString), expected); - } - - @Test(dataProviderClass = JWTUtilsTestDataProvider.class, dataProvider = "validParsableJwtStrings") - public void testGetSignedJWT(String jwtString) throws ParseException { - - Assert.assertNotNull(JWTUtils.getSignedJWT(jwtString)); - } - - @Test(expectedExceptions = ParseException.class, - dataProviderClass = JWTUtilsTestDataProvider.class, dataProvider = "validNotParsableJwtStrings") - public void testGetSignedJWTWIthNotParsableJWT(String jwtString) throws ParseException { - - JWTUtils.getSignedJWT(jwtString); - } - - @Test(expectedExceptions = IllegalArgumentException.class, - dataProviderClass = JWTUtilsTestDataProvider.class, dataProvider = "notValidJwtStrings") - public void testGetSignedJWTWIthNotValidJWT(String jwtString) throws ParseException { - - JWTUtils.getSignedJWT(jwtString); - } - - @Test(dataProviderClass = JWTUtilsTestDataProvider.class, dataProvider = "expiryTimeProvider") - public void testValidExpirationTime(Date time, long timeSkew, boolean expected) { - - Assert.assertEquals(JWTUtils.isValidExpiryTime(time, timeSkew), expected); - } - - @Test(dataProviderClass = JWTUtilsTestDataProvider.class, dataProvider = "nbfProvider") - public void testValidNotValidBefore(Date time, long timeSkew, boolean expected) { - - Assert.assertEquals(JWTUtils.isValidNotValidBeforeTime(time, timeSkew), expected); - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/OpenBankingUtilsTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/OpenBankingUtilsTest.java deleted file mode 100644 index 5a408663..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/OpenBankingUtilsTest.java +++ /dev/null @@ -1,165 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.test.util; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.identity.IdentityConstants; -import com.wso2.openbanking.accelerator.common.util.OpenBankingUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.mockito.Mock; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -import java.text.ParseException; - -import static org.mockito.Mockito.when; - - -/** - * Test for Open Banking Utils. - */ -@PrepareForTest({OpenBankingConfigParser.class}) -@PowerMockIgnore({"jdk.internal.reflect.*", "javax.management.*"}) -public class OpenBankingUtilsTest extends PowerMockTestCase { - - private static final Log log = LogFactory.getLog(OpenBankingUtilsTest.class); - @Mock - OpenBankingConfigParser openBankingConfigParser; - - @BeforeMethod() - public void before() { - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - openBankingConfigParser = PowerMockito.mock(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()) - .thenReturn(openBankingConfigParser); - - } - - @Test(priority = 1) - public void getSoftwareEnvironmentFromSSA() throws ParseException { - String sandboxSsa = "eyJ0eXAiOiJKV1QiLCJraWQiOiJoM1pDRjBWcnpnWGduSENxYkhiS1h6emZqVGciLCJhbGciOiJQUzI1NiJ9." + - "eyJpYXQiOjE2OTg2ODQ4MjUsIm5iZiI6MTY5ODY4NDgyMSwiZXhwIjoxNjk4Njg4NDI2LCJqdGkiOiIyNDdlNjdmNjBmODA0YT" + - "k5MTY5ODY4NDgyNSIsImlzcyI6Ik9wZW5CYW5raW5nIEx0ZCIsInNvZnR3YXJlX2Vudmlyb25tZW50Ijoic2FuZGJveCIsInNv" + - "ZnR3YXJlX21vZGUiOiJUZXN0Iiwic29mdHdhcmVfaWQiOiIxMlp6RkZCeFNMR0VqUFpvZ1JBYnZGZHMxMTY5ODY4NDgyNSIsIn" + - "NvZnR3YXJlX2NsaWVudF9pZCI6IjEwWnpGRkJ4U0xHRWpQWm9nUkFidkZkczEiLCJzb2Z0d2FyZV9jbGllbnRfbmFtZSI6IldT" + - "TzIgT3BlbiBCYW5raW5nIFRQUCAoU2FuZGJveCkiLCJzb2Z0d2FyZV9jbGllbnRfZGVzY3JpcHRpb24iOiJXU08yIE9wZW4gQm" + - "Fua2luZyBUUFAgZm9yIHRlc3RpbmciLCJzb2Z0d2FyZV92ZXJzaW9uIjoxLjUsInNvZnR3YXJlX2NsaWVudF91cmkiOiJodHRw" + - "czovL3d3dy5nb29nbGUuY29tIiwic29mdHdhcmVfcmVkaXJlY3RfdXJpcyI6WyJodHRwczovL3d3dy5nb29nbGUuY29tL3JlZG" + - "lyZWN0cy9yZWRpcmVjdDEiXSwic29mdHdhcmVfcm9sZXMiOlsiUElTUCIsIkFJU1AiLCJDQlBJSSJdLCJvcmdhbmlzYXRpb25f" + - "Y29tcGV0ZW50X2F1dGhvcml0eV9jbGFpbXMiOnsiYXV0aG9yaXR5X2lkIjoiT0JHQlIiLCJyZWdpc3RyYXRpb25faWQiOiJVbm" + - "tub3duMDAxNTgwMDAwMUhRUXJaQUFYIiwic3RhdHVzIjoiQWN0aXZlIiwiYXV0aG9yaXNhdGlvbnMiOlt7Im1lbWJlcl9zdGF0" + - "ZSI6IkdCIiwicm9sZXMiOlsiUElTUCIsIkFJU1AiLCJDQlBJSSJdfSx7Im1lbWJlcl9zdGF0ZSI6IklFIiwicm9sZXMiOlsiUE" + - "lTUCIsIkNCUElJIiwiQUlTUCJdfSx7Im1lbWJlcl9zdGF0ZSI6Ik5MIiwicm9sZXMiOlsiUElTUCIsIkFJU1AiLCJDQlBJSSJd" + - "fV19LCJzb2Z0d2FyZV9sb2dvX3VyaSI6Imh0dHBzOi8vd3d3Lmdvb2dsZS5jb20iLCJvcmdfc3RhdHVzIjoiQWN0aXZlIiwib3" + - "JnX2lkIjoiMDAxNTgwMDAwMUhRUXJaQUFYIiwib3JnX25hbWUiOiJXU08yIChVSykgTElNSVRFRCIsIm9yZ19jb250YWN0cyI6" + - "W3sibmFtZSI6IlRlY2huaWNhbCIsImVtYWlsIjoic2FjaGluaXNAd3NvMi5jb20iLCJwaG9uZSI6Iis5NDc3NDI3NDM3NCIsIn" + - "R5cGUiOiJUZWNobmljYWwifSx7Im5hbWUiOiJCdXNpbmVzcyIsImVtYWlsIjoic2FjaGluaXNAd3NvMi5jb20iLCJwaG9uZSI6" + - "Iis5NDc3NDI3NDM3NCIsInR5cGUiOiJCdXNpbmVzcyJ9XSwib3JnX2p3a3NfZW5kcG9pbnQiOiJodHRwczovL2tleXN0b3JlLm" + - "9wZW5iYW5raW5ndGVzdC5vcmcudWsvMDAxNTgwMDAwMUhRUXJaQUFYLzAwMTU4MDAwMDFIUVFyWkFBWC5qd2tzIiwib3JnX2p3" + - "a3NfcmV2b2tlZF9lbmRwb2ludCI6Imh0dHBzOi8va2V5c3RvcmUub3BlbmJhbmtpbmd0ZXN0Lm9yZy51ay8wMDE1ODAwMDAxSF" + - "FRclpBQVgvcmV2b2tlZC8wMDE1ODAwMDAxSFFRclpBQVguandrcyIsInNvZnR3YXJlX2p3a3NfZW5kcG9pbnQiOiJodHRwczov" + - "L2tleXN0b3JlLm9wZW5iYW5raW5ndGVzdC5vcmcudWsvMDAxNTgwMDAwMUhRUXJaQUFYLzAwMTU4MDAwMDFIUVFyWkFBWC5qd2" + - "tzIiwic29mdHdhcmVfandrc19yZXZva2VkX2VuZHBvaW50IjoiaHR0cHM6Ly9rZXlzdG9yZS5vcGVuYmFua2luZ3Rlc3Qub3Jn" + - "LnVrLzAwMTU4MDAwMDFIUVFyWkFBWC9yZXZva2VkLzlaekZGQnhTTEdFalBab2dSQWJ2RmQuandrcyIsInNvZnR3YXJlX3BvbG" + - "ljeV91cmkiOiJodHRwczovL3d3dy5nb29nbGUuY29tIiwic29mdHdhcmVfdG9zX3VyaSI6Imh0dHBzOi8vd3d3Lmdvb2dsZS5j" + - "b20iLCJzb2Z0d2FyZV9vbl9iZWhhbGZfb2Zfb3JnIjpudWxsfQ.SUZaSo0sEfBU2ffN73IqNG8KAoYEO8vUIrZHBOxA-gF5dKN" + - "IZR6pQ9cnuc3NzhmfHr9TAhiC_KVV9ULiwg0Kh0V79z57Ykjz6NuZ8m0tZPQbjOMQBrRXdnLkqqot_pO_2vwLCRFDfhWM2wqR4" + - "lTXkM0KsdNSWgG3vl25JTkwqo1tTsYlZUcQFltlLQ-lCXT2nWnu_dPZWUqzVb9g4s2DcQ78xkJwqHJKgGLsloXzAMDx36MZQ01" + - "fHP2eIFu82D0PgsxqvHbNeyXVlg5XsX5TLRwrRy8W4wP_SLMoP7jDic0yEufBRULROX2ckpoZuk31a_QyaJFKtIiPj9zlltM9Zg"; - PowerMockito.when(OpenBankingConfigParser.getInstance() - .getSoftwareEnvIdentificationSSAPropertyValueForSandbox()).thenReturn("sandbox"); - PowerMockito.when(OpenBankingConfigParser.getInstance() - .getSoftwareEnvIdentificationSSAPropertyName()).thenReturn("software_environment"); - String softwareEnvironmentFromSSA = OpenBankingUtils.getSoftwareEnvironmentFromSSA(sandboxSsa); - Assert.assertEquals(softwareEnvironmentFromSSA, IdentityConstants.SANDBOX); - } - - @Test() - public void getSoftwareEnvironmentFromSSAForProd() throws ParseException { - String prodSsa = "eyJ0eXAiOiJKV1QiLCJraWQiOiJoM1pDRjBWcnpnWGduSENxYkhiS1h6emZqVGciLCJhbGciOiJQUzI1NiJ9." + - "eyJpYXQiOjE2OTg2ODQ4MjUsIm5iZiI6MTY5ODY4NDgyMSwiZXhwIjoxNjk4Njg4NDI2LCJqdGkiOiIyNDdlNjdmNjBmODA0YT" + - "k5MTY5ODY4NDgyNSIsImlzcyI6Ik9wZW5CYW5raW5nIEx0ZCIsInNvZnR3YXJlX2Vudmlyb25tZW50IjoicHJvZCIsInNvZnR3" + - "YXJlX21vZGUiOiJUZXN0Iiwic29mdHdhcmVfaWQiOiIxMlp6RkZCeFNMR0VqUFpvZ1JBYnZGZHMxMTY5ODY4NDgyNSIsInNvZn" + - "R3YXJlX2NsaWVudF9pZCI6IjEwWnpGRkJ4U0xHRWpQWm9nUkFidkZkczEiLCJzb2Z0d2FyZV9jbGllbnRfbmFtZSI6IldTTzIg" + - "T3BlbiBCYW5raW5nIFRQUCAoU2FuZGJveCkiLCJzb2Z0d2FyZV9jbGllbnRfZGVzY3JpcHRpb24iOiJXU08yIE9wZW4gQmFua2" + - "luZyBUUFAgZm9yIHRlc3RpbmciLCJzb2Z0d2FyZV92ZXJzaW9uIjoxLjUsInNvZnR3YXJlX2NsaWVudF91cmkiOiJodHRwczov" + - "L3d3dy5nb29nbGUuY29tIiwic29mdHdhcmVfcmVkaXJlY3RfdXJpcyI6WyJodHRwczovL3d3dy5nb29nbGUuY29tL3JlZGlyZW" + - "N0cy9yZWRpcmVjdDEiXSwic29mdHdhcmVfcm9sZXMiOlsiUElTUCIsIkFJU1AiLCJDQlBJSSJdLCJvcmdhbmlzYXRpb25fY29t" + - "cGV0ZW50X2F1dGhvcml0eV9jbGFpbXMiOnsiYXV0aG9yaXR5X2lkIjoiT0JHQlIiLCJyZWdpc3RyYXRpb25faWQiOiJVbmtub3" + - "duMDAxNTgwMDAwMUhRUXJaQUFYIiwic3RhdHVzIjoiQWN0aXZlIiwiYXV0aG9yaXNhdGlvbnMiOlt7Im1lbWJlcl9zdGF0ZSI6" + - "IkdCIiwicm9sZXMiOlsiUElTUCIsIkFJU1AiLCJDQlBJSSJdfSx7Im1lbWJlcl9zdGF0ZSI6IklFIiwicm9sZXMiOlsiUElTUC" + - "IsIkNCUElJIiwiQUlTUCJdfSx7Im1lbWJlcl9zdGF0ZSI6Ik5MIiwicm9sZXMiOlsiUElTUCIsIkFJU1AiLCJDQlBJSSJdfV19" + - "LCJzb2Z0d2FyZV9sb2dvX3VyaSI6Imh0dHBzOi8vd3d3Lmdvb2dsZS5jb20iLCJvcmdfc3RhdHVzIjoiQWN0aXZlIiwib3JnX2" + - "lkIjoiMDAxNTgwMDAwMUhRUXJaQUFYIiwib3JnX25hbWUiOiJXU08yIChVSykgTElNSVRFRCIsIm9yZ19jb250YWN0cyI6W3si" + - "bmFtZSI6IlRlY2huaWNhbCIsImVtYWlsIjoic2FjaGluaXNAd3NvMi5jb20iLCJwaG9uZSI6Iis5NDc3NDI3NDM3NCIsInR5cG" + - "UiOiJUZWNobmljYWwifSx7Im5hbWUiOiJCdXNpbmVzcyIsImVtYWlsIjoic2FjaGluaXNAd3NvMi5jb20iLCJwaG9uZSI6Iis5" + - "NDc3NDI3NDM3NCIsInR5cGUiOiJCdXNpbmVzcyJ9XSwib3JnX2p3a3NfZW5kcG9pbnQiOiJodHRwczovL2tleXN0b3JlLm9wZW" + - "5iYW5raW5ndGVzdC5vcmcudWsvMDAxNTgwMDAwMUhRUXJaQUFYLzAwMTU4MDAwMDFIUVFyWkFBWC5qd2tzIiwib3JnX2p3a3Nf" + - "cmV2b2tlZF9lbmRwb2ludCI6Imh0dHBzOi8va2V5c3RvcmUub3BlbmJhbmtpbmd0ZXN0Lm9yZy51ay8wMDE1ODAwMDAxSFFRcl" + - "pBQVgvcmV2b2tlZC8wMDE1ODAwMDAxSFFRclpBQVguandrcyIsInNvZnR3YXJlX2p3a3NfZW5kcG9pbnQiOiJodHRwczovL2tl" + - "eXN0b3JlLm9wZW5iYW5raW5ndGVzdC5vcmcudWsvMDAxNTgwMDAwMUhRUXJaQUFYLzAwMTU4MDAwMDFIUVFyWkFBWC5qd2tzIi" + - "wic29mdHdhcmVfandrc19yZXZva2VkX2VuZHBvaW50IjoiaHR0cHM6Ly9rZXlzdG9yZS5vcGVuYmFua2luZ3Rlc3Qub3JnLnVr" + - "LzAwMTU4MDAwMDFIUVFyWkFBWC9yZXZva2VkLzlaekZGQnhTTEdFalBab2dSQWJ2RmQuandrcyIsInNvZnR3YXJlX3BvbGljeV" + - "91cmkiOiJodHRwczovL3d3dy5nb29nbGUuY29tIiwic29mdHdhcmVfdG9zX3VyaSI6Imh0dHBzOi8vd3d3Lmdvb2dsZS5jb20i" + - "LCJzb2Z0d2FyZV9vbl9iZWhhbGZfb2Zfb3JnIjpudWxsfQ.NLglx-H9D-i2f9GmSrxq00wTlKGHW_6zmKxGg_UhX0P0dzqJmNW" + - "UCDBdz-HhjlPSGeLqumyM_hJZELGv96p6CllmHdNA12gIGem3oBqnaPq9wfcr5Esn7sfRODPComjr6lKxNSXraLT7qpRHCJoxq" + - "yi72RH7T6HyF5lobTHWcZRkCNtc9cWJMKbftGCDSGRlO0XSYvvdGMDBCQT5-KiuKiWcKcBcFX2TLpTDDYaf-GNtATQ0O_vl266" + - "fDPyzG9XF6NLheG0ITrTBGuVN2JzSDC50_vCqR754LtFKNLXKQ2WTnrY3TgEBbyaKj3N0_YdDIuT442zkadg8lvoNpXyk4A"; - - PowerMockito.when(OpenBankingConfigParser.getInstance() - .getSoftwareEnvIdentificationSSAPropertyValueForSandbox()).thenReturn("sandbox"); - PowerMockito.when(OpenBankingConfigParser.getInstance() - .getSoftwareEnvIdentificationSSAPropertyName()).thenReturn("software_environment"); - String softwareEnvironmentFromSSA = OpenBankingUtils.getSoftwareEnvironmentFromSSA(prodSsa); - Assert.assertEquals(softwareEnvironmentFromSSA, IdentityConstants.PRODUCTION); - } - - @Test - public void testDisputeDataWhenNonErrorPublishingEnabled() throws Exception { - - when(openBankingConfigParser.isNonErrorDisputeDataPublishingEnabled()).thenReturn(true); - - Assert.assertTrue(OpenBankingUtils.isPublishableDisputeData(400)); - Assert.assertTrue(OpenBankingUtils.isPublishableDisputeData(200)); - } - - @Test - public void testDisputeDataWhenNonErrorPublishingDisabled() throws Exception { - - when(openBankingConfigParser.isNonErrorDisputeDataPublishingEnabled()).thenReturn(false); - - Assert.assertTrue(OpenBankingUtils.isPublishableDisputeData(400)); - Assert.assertFalse(OpenBankingUtils.isPublishableDisputeData(200)); - } - - @Test - public void testReducingStringLength() throws Exception { - - String body = "String Body"; - Assert.assertEquals(OpenBankingUtils.reduceStringLength(body, 25), body); - Assert.assertEquals(OpenBankingUtils.reduceStringLength(body, 6), "String"); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/SPQueryExecutorUtilTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/SPQueryExecutorUtilTest.java deleted file mode 100644 index 3fba38f3..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/SPQueryExecutorUtilTest.java +++ /dev/null @@ -1,100 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.test.util; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.HTTPClientUtils; -import com.wso2.openbanking.accelerator.common.util.SPQueryExecutorUtil; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.ParseException; -import org.apache.commons.io.FileUtils; -import org.apache.http.HttpEntity; -import org.apache.http.HttpStatus; -import org.apache.http.StatusLine; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpGet; -import org.apache.http.impl.client.CloseableHttpClient; -import org.mockito.Mockito; -import org.mockito.MockitoAnnotations; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; - -import java.io.ByteArrayInputStream; -import java.io.File; -import java.io.IOException; -import java.io.InputStream; -import java.nio.charset.StandardCharsets; - -/** - * Test class for SPQueryExecutorUtil. - */ -@PrepareForTest({HTTPClientUtils.class}) -@PowerMockIgnore({"jdk.internal.reflect.*"}) -public class SPQueryExecutorUtilTest { - private static final String spApiHost = "https://localhost:7444"; - private static final String spUsername = "admin@wso2.com@carbon.super"; - private static final String spPassword = "wso2123"; - private static final String appName = "dummyAppName"; - private static final String query = "dummyQuery"; - private static final String records = "records"; - - @BeforeClass - public void initTest() throws ReflectiveOperationException { - - MockitoAnnotations.initMocks(this); - - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - - @Test - public void testSPQueryExecutor() throws OpenBankingException, IOException, ParseException { - File file = new File("src/test/resources/test-data.json"); - byte[] crlBytes = FileUtils.readFileToString(file, String.valueOf(StandardCharsets.UTF_8)) - .getBytes(StandardCharsets.UTF_8); - InputStream inStream = new ByteArrayInputStream(crlBytes); - - StatusLine statusLineMock = Mockito.mock(StatusLine.class); - Mockito.doReturn(HttpStatus.SC_OK).when(statusLineMock).getStatusCode(); - HttpEntity httpEntityMock = Mockito.mock(HttpEntity.class); - Mockito.doReturn(inStream).when(httpEntityMock).getContent(); - - CloseableHttpResponse httpResponseMock = Mockito.mock(CloseableHttpResponse.class); - Mockito.doReturn(statusLineMock).when(httpResponseMock).getStatusLine(); - Mockito.doReturn(httpEntityMock).when(httpResponseMock).getEntity(); - CloseableHttpClient closeableHttpClientMock = Mockito.mock(CloseableHttpClient.class); - Mockito.doReturn(httpResponseMock).when(closeableHttpClientMock).execute(Mockito.any(HttpGet.class)); - PowerMockito.mockStatic(HTTPClientUtils.class); - Mockito.when(HTTPClientUtils.getHttpsClient()).thenReturn(closeableHttpClientMock); - - JSONObject result = SPQueryExecutorUtil.executeQueryOnStreamProcessor(appName, query, spUsername, - spPassword, spApiHost); - Assert.assertNotNull(result); - Assert.assertTrue(result.containsKey(records)); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/SecurityUtilsTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/SecurityUtilsTest.java deleted file mode 100644 index 2e9437f0..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/SecurityUtilsTest.java +++ /dev/null @@ -1,65 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.test.util; - -import com.wso2.openbanking.accelerator.common.util.SecurityUtils; -import org.testng.Assert; -import org.testng.annotations.Test; - -import java.util.ArrayList; -import java.util.HashSet; -import java.util.List; -import java.util.Set; - -/** - * Tests Common Security Utils. - */ -public class SecurityUtilsTest { - - @Test - public void testSanitizeString() { - String sanitizedString = SecurityUtils.sanitizeString("tests\nsanitizing"); - Assert.assertFalse(sanitizedString.contains("\n")); - } - - @Test - public void testSanitizeStringList() { - List sanitizedList = new ArrayList<>(); - sanitizedList.add("tests\nsanitizing"); - sanitizedList.add("tests\nsan\nitizing"); - sanitizedList.add("tests\nsanitizing\n"); - - Assert.assertFalse(SecurityUtils.sanitize(sanitizedList).stream().anyMatch(s -> s.contains("\n"))); - } - - @Test - public void testSanitizeStringSet() { - Set sanitizedList = new HashSet<>(); - sanitizedList.add("tests\nsanitizing"); - sanitizedList.add("tests\nsanitizingtext"); - sanitizedList.add("tests\nsanitizingwords"); - - Assert.assertFalse(SecurityUtils.sanitize(sanitizedList).stream().anyMatch(s -> s.contains("\n"))); - } - - @Test - public void testContainSpecialChars() { - Assert.assertTrue(SecurityUtils.containSpecialChars("tests&sanitizing")); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/eidas/certificate/extractor/CertificateContentExtractorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/eidas/certificate/extractor/CertificateContentExtractorTest.java deleted file mode 100644 index 03913254..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/eidas/certificate/extractor/CertificateContentExtractorTest.java +++ /dev/null @@ -1,65 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.test.util.eidas.certificate.extractor; - -import com.wso2.openbanking.accelerator.common.test.util.CommonTestUtil; -import com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor.CertificateContent; -import com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor.CertificateContentExtractor; -import org.testng.Assert; -import org.testng.annotations.Test; - -import java.security.cert.CertificateException; -import java.security.cert.X509Certificate; - -/** - * Certificate content extractor test. - */ -public class CertificateContentExtractorTest { - - - @Test - public void testExtractValidCertificate() throws Exception { - - X509Certificate cert = - CommonTestUtil.parseTransportCert(CommonTestUtil.EIDAS_CERT).orElse(null); - - CertificateContent extract = CertificateContentExtractor.extract(cert); - - Assert.assertTrue(extract.getPspRoles().size() == 3); - Assert.assertTrue(extract.getPspRoles().contains("AISP")); - Assert.assertTrue(extract.getPspRoles().contains("PISP")); - Assert.assertTrue(extract.getPspRoles().contains("CBPII")); - Assert.assertTrue(extract.getPspAuthorisationNumber().equals("PSDDE-BAFIN-123456")); - Assert.assertTrue(extract.getName().equals("www.hanseaticbank.de")); - Assert.assertTrue(extract.getNcaName().equals("Federal Financial Supervisory Authority")); - Assert.assertTrue(extract.getNcaId().equals("DE-BAFIN")); - } - - @Test - public void testExtractInvalidCertificate() throws CertificateException { - - X509Certificate cert = CommonTestUtil - .parseTransportCert(CommonTestUtil.TEST_CLIENT_CERT).orElse(null); - try { - CertificateContentExtractor.extract(cert); - } catch (Exception ex) { - Assert.assertEquals("X509 V3 Extensions not found in the certificate.", ex.getMessage()); - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/testutils/JWTUtilsTestDataProvider.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/testutils/JWTUtilsTestDataProvider.java deleted file mode 100644 index c1ef208f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/util/testutils/JWTUtilsTestDataProvider.java +++ /dev/null @@ -1,132 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.test.util.testutils; - -import org.testng.annotations.DataProvider; - -import java.util.Date; - -/** - * Data Provider for JWTUtilsTest. - */ -public class JWTUtilsTestDataProvider { - - @DataProvider(name = "jwtStrings") - - public Object[][] getJwtStrings() { - - return new Object[][] { - - // Valid JWT String - {"eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6Ikpv" + - "aG4gRG9lIiwiaWF0IjoxNTE2MjM5MDIyfQSflK.xwRJSMeKKF2QT4fwpMeJf36POk6yJV_adQssw5c", - true}, - // Empty String - {"", false}, - // Null String - {null, false}, - // Invalid JWT String with less than 2 dots - {"eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9", false}, - // Invalid JWT String with more than 2 dots - {"eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4" + - "gRG9lIiwiaWF0IjoxNTE2MjM5MDIyfQ.SflKxwRJSMeKKF2QT4fwpMeJf36POk6yJV_adQssw5c.extra", - false}, - // JWT String with whitespace - {" eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG" + - "4gRG9lIiwiaWF0IjoxNT.E2MjM5MDIyfQSflKxwRJSMeKKF2QT4fwpMeJf36POk6yJV_adQssw5c ", - true}, - // JWT String with only dots - {"...", false}, - // JWT String with valid segments but invalid encoding - {"eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.InvalidBase64.SflKxwRJSMeKKF2QT4fwpMeJf36POk6yJV_adQssw5c", - true}, - // JWT String with valid segments and valid encoding but invalid JSON - {"eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG" + - "9lIiwiaWF0IjoxNTE2MjM5MDIyfQSflKxwRJSMeKKF2QT4fwpMeJf36POk6yJV_adQssw5c.invalidJSON", - true} - }; - } - - @DataProvider(name = "validParsableJwtStrings") - public Object[][] getValidParsablejwtStrings() { - - String parsableJwtString = - "eyJhbGciOiJSUzI1NiIsInR5cCI6IkpXVCJ9.eyJkaWQiOiI1NTBmNDQ1My05NTQ3LT" + - "RlNGYtYmUwNi04ZGIyZWVkNTYzYjMiLCJsb2dpbkhpbnQiOiJhZG1pbkB3c28yLmNvb" + - "SIsImlhdCI6MTcxNDkyOTk2MCwianRpIjoiNmU0MWM4N2UtYWJmNi00ZjU1LTliNjQt" + - "NjYwMWFlODg2NjZjIiwiZXhwIjoxNzE0OTMxNzYwLCJuYmYiOjE3MTQ5Mjk5NjB9.WB" + - "7qvq3w6htUop600H5C4HwL-r0wb8GekJE6X4-zrFn2IofEcwV0yisSE5fH8uyrzdmVm" + - "OiBgFXY9Y9cUVlS6t9HMbhlzs2qY0bVzDYVNG7GjgnYIcyh3lx9obqL9O3DJKNre5GS" + - "3b-ATPN6VvYC9F2KnwwuoNky-3Wlcw3G9-E"; - - return new String[][] { - {parsableJwtString} - }; - } - - @DataProvider(name = "validNotParsableJwtStrings") - public Object[][] getValidNotParsableJwtStrings() { - - String notParsableJwtString = - "eyJhbGciOiJSUzI1NiIsInR5cCI6IkpXNOTVCJ9.eyJkaWQiOiI1NTBmNDQ1My05NTQ" + - "RlNGYtYmUwNi04ZGIyZWVkNTYzYjMiLCJsb2dpbkhpbnQiOiJhZG1pbkB3c28yLmNvb" + - "SIsImlhdCI6MTcxNDkyOTk2MCwianRpIjoiNmU0MWM4N2UtYWJmNi00ZjU1LTliNjQt" + - "NjYwMWFlODg2NjZjIiwiZXhwIjoxNzE0OTMxNzYwLCJuYmYiOjE3MTQ5Mjk5NjB9.WB" + - "7qvq3w6htUop600H5C4HwL-r0wb8GekJE6X4-zrFn2IofEcwV0yisSE5fH8uyrzdmVm" + - "OiBgFXY9Y9cUVlS6t9HMbhlzs2qY0bVzDYVNG7GjgnYIcyh3lx9obqL9O3DJKNre5GS" + - "3b-ATPN6VvYC9F2KnwwuoNky-3Wlcw3G9-E3LT"; - - return new String[][] { - {notParsableJwtString} - }; - } - - @DataProvider(name = "notValidJwtStrings") - public Object[][] getNotValidJwtStrings() { - - String notValidJwtString = - "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4" + - "gRG9lIiwiaWF0IjoxNTE2MjM5MDIyfQ.SflKxwRJSMeKKF2QT4fwpMeJf36POk6yJV_adQssw5c.extra"; - - return new String[][] { - {notValidJwtString} - }; - } - - @DataProvider(name = "expiryTimeProvider") - public Object[][] getExpiryTime() { - - return new Object[][]{ - {null, 0, false}, - {new Date(System.currentTimeMillis() - 10 * 1000), 0, false}, - {new Date(System.currentTimeMillis() + 10 * 100), 0, true} - }; - } - - @DataProvider(name = "nbfProvider") - public Object[][] getNbf() { - - return new Object[][]{ - {null, 0, false}, - {new Date(System.currentTimeMillis() + 10 * 1000), 0, false}, - {new Date(System.currentTimeMillis() - 3 * 1000), 0, true} - }; - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/LogicValidatorsTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/LogicValidatorsTest.java deleted file mode 100644 index 53f631c9..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/LogicValidatorsTest.java +++ /dev/null @@ -1,88 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.test.validator; - -import com.nimbusds.jwt.JWTClaimsSet; -import com.wso2.openbanking.accelerator.common.test.validator.resources.SampleChildRequestObject; -import com.wso2.openbanking.accelerator.common.test.validator.resources.SampleRequestObject; -import com.wso2.openbanking.accelerator.common.test.validator.resources.ValidatorTestDataProvider; -import com.wso2.openbanking.accelerator.common.validator.OpenBankingValidator; -import org.hibernate.validator.HibernateValidator; -import org.testng.annotations.Test; - -import java.text.ParseException; -import java.util.Set; - -import javax.validation.ConstraintViolation; -import javax.validation.Validation; -import javax.validation.Validator; - - -import static org.testng.AssertJUnit.assertNotNull; -import static org.testng.AssertJUnit.assertNull; - -/** - * Logic validators test. - */ -public class LogicValidatorsTest { - - private SampleRequestObject sampleRequestObject = new SampleRequestObject(); - private static Validator uut = Validation.byProvider(HibernateValidator.class).configure().addProperty( - "hibernate.uut.fail_fast", "true").buildValidatorFactory().getValidator(); - - @Test(dataProvider = "dp-checkValidScopeFormat", dataProviderClass = ValidatorTestDataProvider.class) - public void checkValidScopeFormat(String claimsString) throws ParseException { - - //Assign - sampleRequestObject.setClaimSet(JWTClaimsSet.parse(claimsString)); - - //Act - Set> violations = uut.validate(sampleRequestObject); - - //Assert - String violation = violations.stream().findFirst().map(ConstraintViolation::getMessage).orElse(null); - assertNull("Valid scope formats should pass", violation); - - } - - @Test(dataProvider = "dp-checkValidationsInherited", dataProviderClass = ValidatorTestDataProvider.class) - public void checkValidationsInherited(String claimsString) throws ParseException { - - SampleChildRequestObject sampleChildRequestObject = new SampleChildRequestObject(); - - sampleChildRequestObject.setClaimSet(JWTClaimsSet.parse(claimsString)); - Set> violations = uut.validate(sampleChildRequestObject); - String violation = violations.stream().findFirst().map(ConstraintViolation::getMessage).orElse(null); - assertNotNull("Inherited validations should work", violation); - } - - @Test(dataProvider = "dp-checkValidScopeFormat", dataProviderClass = ValidatorTestDataProvider.class) - public void checkOpenBankingValidator(String claimsString) throws ParseException { - - //Assign - sampleRequestObject.setClaimSet(JWTClaimsSet.parse(claimsString)); - - //Act - String violation = OpenBankingValidator.getInstance().getFirstViolation(sampleRequestObject); - - //Assert - assertNull("Valid scope formats should pass", violation); - - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/ModelValidatorsTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/ModelValidatorsTest.java deleted file mode 100644 index c8a44ae7..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/ModelValidatorsTest.java +++ /dev/null @@ -1,131 +0,0 @@ -package com.wso2.openbanking.accelerator.common.test.validator; -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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. - */ - -import com.nimbusds.jwt.JWTClaimsSet; -import com.wso2.openbanking.accelerator.common.test.validator.resources.SampleDifferentClass; -import com.wso2.openbanking.accelerator.common.test.validator.resources.SampleRequestObject; -import com.wso2.openbanking.accelerator.common.test.validator.resources.ValidatorTestDataProvider; -import com.wso2.openbanking.accelerator.common.validator.OpenBankingValidator; -import org.hibernate.validator.HibernateValidator; -import org.testng.annotations.Test; - -import java.text.ParseException; -import java.util.Set; - -import javax.validation.ConstraintViolation; -import javax.validation.Validation; -import javax.validation.Validator; - -import static org.testng.AssertJUnit.assertEquals; -import static org.testng.AssertJUnit.assertNull; - -/** - * Model validators test. - */ -public class ModelValidatorsTest { - - private SampleRequestObject sampleRequestObject = new SampleRequestObject(); - private static Validator validator = Validation.byProvider(HibernateValidator.class).configure().addProperty( - "hibernate.validator.fail_fast", "true").buildValidatorFactory().getValidator(); - - private static Validator validator2 = Validation.byProvider(HibernateValidator.class).configure().addProperty( - "hibernate.validator.fail_fast", "true").buildValidatorFactory().getValidator(); - - private static Validator validator3 = Validation.byProvider(HibernateValidator.class).configure().addProperty( - "hibernate.validator.fail_fast", "true").buildValidatorFactory().getValidator(); - - - @Test(dataProvider = "dp-checkValidScopeFormat", dataProviderClass = ValidatorTestDataProvider.class) - public void checkValidScopeFormat(String claimsString) throws ParseException { - - sampleRequestObject.setClaimSet(JWTClaimsSet.parse(claimsString)); - Set> violations = validator.validate(sampleRequestObject); - String violation = violations.stream().findFirst().map(ConstraintViolation::getMessage).orElse(null); - assertNull("Valid scope formats should pass", violation); - } - - @Test(dataProvider = "dp-checkValidSingleScopes", dataProviderClass = ValidatorTestDataProvider.class) - public void checkValidSingleScopes(String claimsString) throws ParseException { - - sampleRequestObject.setClaimSet(JWTClaimsSet.parse(claimsString)); - Set> violations = validator.validate(sampleRequestObject); - String violation = violations.stream().findFirst().map(ConstraintViolation::getMessage).orElse(null); - assertNull("Valid single scope should pass", violation); - } - - @Test - public void checkMandatoryParamsValidationFailing() { - - SampleDifferentClass sampleRequestObject = new SampleDifferentClass(); - sampleRequestObject.setName("name"); - sampleRequestObject.setMale(true); - - Set> violations = validator.validate(sampleRequestObject); - String violation = violations.stream().findFirst().map(ConstraintViolation::getMessage).orElse(null); - assertEquals("age failed", violation); - - // - sampleRequestObject = new SampleDifferentClass(); - sampleRequestObject.setName("name"); - sampleRequestObject.setAge(70); - - violations = validator2.validate(sampleRequestObject); - violation = violations.stream().findFirst().map(ConstraintViolation::getMessage).orElse(null); - assertEquals("male failed", violation); - - // - sampleRequestObject = new SampleDifferentClass(); - sampleRequestObject.setMale(true); - sampleRequestObject.setAge(70); - - violations = validator3.validate(sampleRequestObject); - violation = violations.stream().findFirst().map(ConstraintViolation::getMessage).orElse(null); - assertEquals("name failed", violation); - - } - - @Test - public void checkOpenBankingValidator() { - - SampleDifferentClass sampleRequestObject = new SampleDifferentClass(); - sampleRequestObject.setName("name"); - sampleRequestObject.setMale(true); - - String violation = OpenBankingValidator.getInstance().getFirstViolation(sampleRequestObject); - assertEquals("age failed", violation); - - // - sampleRequestObject = new SampleDifferentClass(); - sampleRequestObject.setName("name"); - sampleRequestObject.setAge(70); - - violation = OpenBankingValidator.getInstance().getFirstViolation(sampleRequestObject); - assertEquals("male failed", violation); - - // - sampleRequestObject = new SampleDifferentClass(); - sampleRequestObject.setMale(true); - sampleRequestObject.setAge(70); - - violation = OpenBankingValidator.getInstance().getFirstViolation(sampleRequestObject); - assertEquals("name failed", violation); - - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/resources/SampleChildRequestObject.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/resources/SampleChildRequestObject.java deleted file mode 100644 index 74849b20..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/resources/SampleChildRequestObject.java +++ /dev/null @@ -1,38 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.test.validator.resources; - -import com.wso2.openbanking.accelerator.common.validator.annotation.RequiredParameter; - -/** - * Sample child request object resource. - */ -@RequiredParameter(param = "name", message = "name failed") -public class SampleChildRequestObject extends SampleRequestObject { - private String name; - - public String getName() { - return name; - } - - public void setName(String name) { - this.name = name; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/resources/SampleDifferentClass.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/resources/SampleDifferentClass.java deleted file mode 100644 index 80e0a429..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/resources/SampleDifferentClass.java +++ /dev/null @@ -1,58 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.test.validator.resources; - -import com.wso2.openbanking.accelerator.common.validator.annotation.RequiredParameter; - -/** - * Sample Different class resource. - */ -@RequiredParameter(param = "name", message = "name failed") -@RequiredParameter(param = "age", message = "age failed") -@RequiredParameter(param = "male", message = "male failed") -public class SampleDifferentClass { - - private String name; - private int age; - private boolean male; - - public String getName() { - return name; - } - - public void setName(String name) { - this.name = name; - } - - public int getAge() { - return age; - } - - public void setAge(int age) { - this.age = age; - } - - public boolean isMale() { - return male; - } - - public void setMale(boolean male) { - this.male = male; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/resources/SampleRequestObject.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/resources/SampleRequestObject.java deleted file mode 100644 index dc5a6c67..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/resources/SampleRequestObject.java +++ /dev/null @@ -1,45 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.common.test.validator.resources; - -import com.nimbusds.jwt.JWTClaimsSet; -import com.wso2.openbanking.accelerator.common.validator.annotation.ValidScopeFormat; - -/** - * Sample request object resource. - */ -@ValidScopeFormat(scope = "claimsSet.claims.scope", message = "Non Confirming Scope") -public class SampleRequestObject { - - private JWTClaimsSet claimsSet; - - public SampleRequestObject() { - } - - public JWTClaimsSet getClaimsSet() { - return claimsSet; - } - - public void setClaimSet(JWTClaimsSet claimsSet) { - this.claimsSet = claimsSet; - } - - public void setClaimsSet(JWTClaimsSet claimsSet) { - this.claimsSet = claimsSet; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/resources/ValidatorTestDataProvider.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/resources/ValidatorTestDataProvider.java deleted file mode 100644 index a5e91864..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/java/com/wso2/openbanking/accelerator/common/test/validator/resources/ValidatorTestDataProvider.java +++ /dev/null @@ -1,155 +0,0 @@ -package com.wso2.openbanking.accelerator.common.test.validator.resources; -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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. - */ -import org.testng.annotations.DataProvider; - -/** - * validator test data provider resource. - */ -public class ValidatorTestDataProvider { - - private String claimsTemplate = "{\n" + - " \"aud\": \"https://localhost:8243/token\",\n" + - " \"response_type\": \"code id_token\",\n" + - " \"client_id\": \"iTKOfuqz46Y1HVY2BF0Z7JM18Awa\",\n" + - " \"redirect_uri\": \"https://localhost/test/a/app1/callback\",\n" + - " \"scope\": \"${SCOPE}\",\n" + - " \"state\": \"af0ifjsldkj\",\n" + - " \"nonce\": \"n-0S6_WzA2Mj\",\n" + - " \"claims\": {\n" + - " \"sharing_duration\": \"7200\",\n" + - " \"id_token\": {\n" + - " \"acr\": {\n" + - " \"essential\": true,\n" + - " \"values\": [\n" + - " \"urn:cds.au:cdr:3\"\n" + - " ]\n" + - " }\n" + - " },\n" + - " \"userinfo\": {\n" + - " \"given_name\": null,\n" + - " \"family_name\": null\n" + - " }\n" + - " }\n" + - "}"; - - - @DataProvider(name = "dp-checkValidScopeFormat") - public Object[][] dpCheckValidScopeFormat() { - - return new Object[][]{ - {claimsTemplate.replace("${SCOPE}", - "openid bank:accounts.basic:read bank:accounts.detail:read bank:transactions:read")}, - {claimsTemplate.replace("${SCOPE}", - "openid bank:accounts.basic:read")}, - {claimsTemplate.replace("${SCOPE}", - " openid bank:accounts.basic:read bank:accounts.detail:read bank:transactions:read ")}, - {claimsTemplate.replace("${SCOPE}", - "openid profile bank:accounts.basic:read bank:accounts.detail:read bank:transactions:read")} - }; - } - - @DataProvider(name = "dp-checkValidSingleScopes") - public Object[][] dpCheckValidSingleScopes() { - - return new Object[][]{ - {claimsTemplate.replace("${SCOPE}", - "openid bank:accounts.basic:read")}, - {claimsTemplate.replace("${SCOPE}", - "openid bank:accounts.detail:read")}, - {claimsTemplate.replace("${SCOPE}", - "openid bank:transactions:read")}, - {claimsTemplate.replace("${SCOPE}", - "openid bank:payees:read")}, - {claimsTemplate.replace("${SCOPE}", - "openid bank:regular_payments:read")}, - {claimsTemplate.replace("${SCOPE}", - "openid common:customer.basic:read")}, - {claimsTemplate.replace("${SCOPE}", - "openid common:customer.detail:read")}, - - // - {claimsTemplate.replace("${SCOPE}", - "accounts openid")}, - {claimsTemplate.replace("${SCOPE}", - "openid payments")}, - {claimsTemplate.replace("${SCOPE}", - "openid fundsconfirmations")}, - - // - {claimsTemplate.replace("${SCOPE}", - "ais openid")}, - {claimsTemplate.replace("${SCOPE}", - "pis openid")} - }; - } - - @DataProvider(name = "dp-checkInValidScopeFormat") - public Object[][] dpCheckInValidScopeFormat() { - - return new Object[][]{ - {claimsTemplate.replace("${SCOPE}", - "bank:accounts.basic:read")}, - {claimsTemplate.replace("${SCOPE}", - "openid")}, - {claimsTemplate.replace("${SCOPE}", - "openid bank:accounts.basic")}, - {claimsTemplate.replace("${SCOPE}", - "openid xyz")}, - {claimsTemplate.replace("${SCOPE}", - "xyz")}, - {claimsTemplate.replace("${SCOPE}", - "openid bank:accounts.basic")}, - {claimsTemplate.replace("${SCOPE}", - "openid common:customer.detail:read")}, - - // - {claimsTemplate.replace("${SCOPE}", - "Accounts openid")}, - {claimsTemplate.replace("${SCOPE}", - "openid Payments")}, - {claimsTemplate.replace("${SCOPE}", - "openid FundsConfirmations")}, - - // - {claimsTemplate.replace("${SCOPE}", - "AIS openid")}, - {claimsTemplate.replace("${SCOPE}", - "PIS openid")} - }; - } - - @DataProvider(name = "dp-checkValidationsInherited") - public Object[][] dpCheckValidationsInherited() { - - return new Object[][]{ - {claimsTemplate.replace("${SCOPE}", - "sactions:read")} - }; - } - - @DataProvider(name = "dp-checkMandatoryParamsValidationFailing") - public Object[][] dpCheckMandatoryParamsValidationFailing() { - - return new Object[][]{ - {claimsTemplate.replace("${SCOPE}", - "openid bank:accounts.basic:read")} - }; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/open-banking-empty.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/open-banking-empty.xml deleted file mode 100644 index 0882a6d1..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/open-banking-empty.xml +++ /dev/null @@ -1,17 +0,0 @@ - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/open-banking.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/open-banking.xml deleted file mode 100644 index a3dbb816..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/open-banking.xml +++ /dev/null @@ -1,284 +0,0 @@ - - - - - - - - sampleDataSourceName - - 1000 - - - - jdbc/WSO2OB_RET_DB - - 1 - - - - - sampleRequestObjectValidator - - - - - sampleServletExtension - - - - - sampleCIBAServletExtension - - - - - sampleSPMetadataFilterExtension - - - - - - DummyValue - ${some.property} - - ${carbon.home} - - - - Nothing - Everything - Anything - - - - - - - - - - - - - - - - - - - - - - - - - false - - - - - true - - - - - - - - - - 3600 - 3600 - 86400 - - true - 3 - - - true - PROXY_HOSTNAME - 8080 - - - - - - - - - CN=Test Pre-Production Issuing CA, O=Test, C=GB - - - true - - - - - - true - true - 0 0 0 * * ? - - - false - 0 0 0 * * ? - Expired - authorised - - - - - - - - - - - - - - - - - true - - - - - - - - - - - - - - - - - - true - true - - - - - - - - - - true - - Sample - - - - - 1000 - 3000 - - - - 1000 - 500 - - - - - full.qualified.name.class1 - full.qualified.name.class2 - - - - - com.wso2.openbanking.accelerator.keymanager.OBKeyManagerImpl - - - - - - - www.wso2.com - 5 - com.wso2.openbanking.accelerator.event.notifications.service.handler.DefaultEventCreationServiceHandler - com.wso2.openbanking.accelerator.event.notifications.service.handler.DefaultEventPollingServiceHandler - com.wso2.openbanking.accelerator.event.notifications.service.service.DefaultEventNotificationGenerator - com.wso2.openbanking.accelerator.event.notifications.service.handler.DefaultEventSubscriptionServiceHandler - - true - true - true - - - - - - false - - PS256 - ES256 - - - - false - - PS256 - ES256 - - - - - - - wso2carbon - wso2carbon-sandbox - 1234 - 5678 - - - 51200 - 2000 - 2000 - - - - - - - - - - - - - - true - 0 0/1 0 ? * * * - 60 - 5 - 60 - EX - 600 - 20 - com.wso2.openbanking.accelerator.event.notifications.service.realtime.service.DefaultRealtimeEventNotificationRequestGenerator - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/test-data.json b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/test-data.json deleted file mode 100644 index dae4490a..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/test-data.json +++ /dev/null @@ -1,11 +0,0 @@ -{ - "records": [ - [ - 1689313470, - "d40e6fa2-fd1a-41b0-89e8-09d5b0791904", - "10.00", - "GBP", - "Sweepco" - ] - ] -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/testFile.js b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/testFile.js deleted file mode 100644 index 2701713b..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/testFile.js +++ /dev/null @@ -1,78 +0,0 @@ - -var psuChannel = 'Online Banking'; - -function onLoginRequest(context) { - reportingData(context, "AuthenticationAttempted", false, psuChannel); - - executeStep(1, { - onSuccess: function (context) { - var supportedAcrValues = ['urn:openbanking:psd2:sca', 'urn:openbanking:psd2:ca']; - var selectedAcr = selectAcrFrom(context, supportedAcrValues); - reportingData(context, "AuthenticationSuccessful", false, psuChannel); - - if (isACREnabled()) { - - context.selectedAcr = selectedAcr; - if (isTRAEnabled()) { - if (selectedAcr === 'urn:openbanking:psd2:ca') { - executeTRAFunction(context); - } else { - executeStep(2, { - onSuccess: function (context) { - reportingData(context, "AuthenticationSuccessful", true, psuChannel); - }, - onFail: function (context) { - reportingData(context, "AuthenticationFailed", false, psuChannel); - } - }); - } - } else { - if (selectedAcr == 'urn:openbanking:psd2:sca') { - executeStep(2, { - onSuccess: function (context) { - reportingData(context, "AuthenticationSuccessful", true, psuChannel); - }, - onFail: function (context) { - reportingData(context, "AuthenticationFailed", false, psuChannel); - } - }); - } - } - - } else { - if (isTRAEnabled()) { - executeTRAFunction(context); - } else { - executeStep(2, { - onSuccess: function (context) { - reportingData(context, "AuthenticationSuccessful", true, psuChannel); - }, - onFail: function (context) { - reportingData(context, "AuthenticationFailed", false, psuChannel); - } - }); - } - } - }, - onFail: function (context) { //basic auth fail - reportingData(context, "AuthenticationFailed", false, psuChannel); - //retry - } - }); -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/testng.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/testng.xml deleted file mode 100644 index 9b956e1e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/testng.xml +++ /dev/null @@ -1,68 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/wso2carbon.jks b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.common/src/test/resources/wso2carbon.jks deleted file mode 100644 index c8775783fb3555d6e88255c1236576e7a4a7df1f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 98874 zcmdqJ1z1(>+Ac~rNOw#?>6nuaX%LW*Zj_pIcc^p;NC`+uHxkktE;W!*v<&7=!taImUC}&;7*vUT$4(K|w*mK>ldl{7YbDW@2Utws+Kb zaB#8*8<`o|+t@f7>O(<6TTa&#bV0#uuy6u-p`c*vfIhHQKp&WtR9F~j7#K{n)wyMY zu}7ET%X;S|yFgwbIs!67fKzBG;zp` zVvrdD($IXk!Js52U(21&c%S`!_^xZxSFa4p4&JCO}LC zNKSwb}z}p`!0(5rB==bnHbsIj=RCgZ$G&~fn5A;1Kpbs<(j1M$4)O!rPo+U9a zozlRAD!m_e*jYtWG;f0P4||RPymT6RXeVA1d3Z5w`VT8g>HJ>3no2=I^CfT(k0^Y# z>ZYs%@?-dZ?=2gbbN9^j!`$>8^E1kQCH>hHz$nE+>G&m~(EzsgvmY9Sob{4l)3|^3S}Q3V0C^b(BslsS~umB*iExkIGgnYxoeim7lX)^*L842EXqrEp`Lx8KKJ)7+7d1=nO-k zJ`e-)QpxT?LtnwcK@2GXc<*|Qb3KL!!b1K5l3gFbfWwEwmDorxK8)(}0Wm9vcQKl!Z@skOIDb}plDjK_S6An7A^?@gaAtw207|8=G%kZo!edyAG44O1dyX5x^ zpMHFbSFB=0MCxH#QTUK^iXy?9zPo<|byNtbY>aH|9rS;}Te0Fdyy2&SKJa4@h=2SE zZ`u4Gos)((a}>w&W7@C*$#<029` zf*m0L1!3O2bvQ&EL}h(TV;g%T5HXPO`f?N;EU_nYPsE=)W+V-(d`=l5U_(lyno=I2QuVgW9Q`Mi&e7g+aCI#d>qSWo5_#Mp(j9qNQ2yO&o z0jZ@F;29~T+D!SpecwmiejDGqy`{uVQ2l-{#xbt3Mu)Nv^{@FOd~l-Xart zN|&6}-BhMHkt4b6r+iD0qCnw0{8v}}5=P)Od+(v&>-XC{Z#T!;*oOYXp9ufRpLh5V z0#w**<3iaRe&knGq*!P9K~4l0Dq1$FYx$!}Z|Z~BFh4Fz($S9qZy(|?7pkR~jJ(8( zT4Skw$C6-Q+CU_3sn407P>zpnrWHCrDws+9jTQE=NS_RogdsD;{qrS!_WZ|@j>~sN z;f%Ec@h4*r$QqBVvy%=-X$NB_xP9$O2OZ|pCF5avbcdzgpy)~4HxmJ*i1*FZT|e-i z^ZBJ#Bs~vnVu?1Ot>T>^AS`Nlh-`Z0m-uja1Qm8s{T=Rc|HhM!hCu47N@+1Sf1TQm zDAcsJZ>G7j_@%`0DBqNsqUzF+37ycuxgK5ARm3fH{+~#$`WOv%GChzZ-Fu+eN0|TEPSf~DQ%1RXe zru>06`Ekuh7s{}!!2Ox}&>e!=DrwaIeQO{pF^0VJ7S{REn%ajKP3=}F(VJtQ17^&9rL28ij~fj%I0f_QO z7C>AE;m;3Jxc!!Z>q{ZNNKEAB!+r@=WN3V%pU;8$_CW~WKAdaedF@hoxquKa^4q1b zfgl6WZWcfF>`QmFf*~X zwS=e@YbPrcwqH1V4E~KX_|Fi<)eON}-mT(7lK%i@f(EPm|q`VbP#D5%C_&NWzxMBuKvi%H}fNY#V9(FDu@H$xH0{t=kcTV+p zdZ%+X9cE$TZpuWPpKdzzBes=CCN^ zf_gP8se+1Z3xEkH^q3oVZR`c}q%E{4pJpKoZ(3kRjL8_56EXQtByk54FLEHFSyvd%EzH zLQ}vC8~$8>5V9v~`#12~qALcn2{44i=S+95JK5XHU>k`Xj!8ao^D^(rvkNoV9b}5M z@}|>+od|MS_z?1S_el3oqswQ&dP@>ajkm|d&-&=uU~wMF>6`n@Dmk(uP2VLT)RQS5MUwW z7}_w&htb@tg&&^vcHcSp0}eJns*6iu-SR{F$5Hm3`oxS+Wfy!MT~0Zr-2LUk5|mqK zT|46LkoEi> zp*5i)KR^~B69N(=0wiC72m_4|Fm-gaF2iw{>m^s?myRrO+^0i}N{R=X0q!p5X z_7OhXxj1pN;oJ|^M@Z+kq@r}Oud6;ci#W+6+qo@U#dgYM1?lwx9jiN$rpkfugc0#A zz%ee+MVVEcsNNHHczKQlJRTaIKUmK`f}O*(p=(~u_y`zPH{v>Gpr{No#Sjv39w}M~ zg{Q~r^SWBdEN7>lD9eJy!2&&JdC8oj;-#dx4&J>C z(3}aNVdiivzgaIi^+lo2eohm?ZNR#X@GdH()S~=7SW7+`Ipi!1`!+QTN>O?02e_t1 z&M42(5N(ZqB(Q#m=cQ>w3n%gvdun~CjmJ8vue)NxDundrhhF(x(1hcjU$mXvvok%7 z{DMWijI!{m0A_s&8!G^LQ_Qv)a}P%6`3Jax2W}`I=eM0W zm{~&f#vg>wPVxugL&zS!0Ydm$KMDT{$q=hb9jnY=%nB&^Un2ZJB+nm&uY5E6Kc#y% z4z9a&e?9#7P4stU-?|s*Ip_*H=J|5+DPF9;{kWccRo<>vCOpf5Ovn0d5v+O<54RXA zfdEBEk11xeYY`DFBjzcSoA38;HKsMH`*7kpyg$z2Ldf^h7fr{h#^=nM8) z_OG0sKHVZKf0xRsofg%#D;;&|j{+~aGLE|Nq;4vP`#f@ujgzXMf=`~-Ke+_&4cml1 zx}=<*Xi=2pH(xCUjkQ-HHM(h2)A5ffxJ>G#mV7I@H>CKA@0o-XAbx}E+%X3Q4t=cr z@y5=>#HhM5^0BA~m<4E#b`-H4v$FryXLCI5@#gmrGRqSFD^{-2a4bnaR$O_lAR~$Z zUsQ%eP88QKs=Xp5l54<}_(uI)1iu3z({o_bC zt0&IVdpAbmDG_Nl{<6;1z8T>jLr=KBt$tzZSJY}=dNn^b`G7isR)+CW3j6I##WX6B zQacz!VxJO&f~V@(6ti8^PN^St-6BFs<`hE@ zO&F5-@lz2;>rYrO5ew<{B87N5)iKc5W)n7hAu9#$U0@sA+^G~3Tqs(5jU*GL9!Z7`5GfaOHk;;(!!^sCf? za$D+f)VBm&%X)oBNJR#qay=6yFlGS}0P$`mK8Com%C7_9dUhz-OG z;dw~F3K@bpAem53E#Uw01bX5F5FCr z8#4Ee-C3GpQHWAHx=`-?X~3*@8^fytmImARpHYb)L^#@i_AoEt7y$-fEt}$asGxR?6m4oHMUe{_m4&v= zHhmGX-Nd&arrG3C6BSj0eM4Fk!tX$a18El=N`K z*K7$hk%*y^UVU*{ATebZM)#KAu9J8FP|f&rxdY)&cCR(3G}D90?sRfqCc^2e6;IDU z81pZtaZ|j=AZAJ=fYQk$-=;I>=4PXRp>67G0 zO!o3E@pKoCOc_9YDRQm*{6*HJ`SRt;#hmvMn-b1rz z3vAmJuT!?S5<8-iN>DPX(kQqzakg`vL{C@4#o@gV@=wo&f#{3*X$8c37~}}34I@Ln zMEv@oFW4&sFCs&J2(X2L%h#TlsynZLuB78_!UL0be0Pa8M+tL59g9STdyP0rUx~7v z?8C{4c>_9h2e2Uo4L7j+6-<+N{(-rA2cjnPOFnMaQozE8j#$!@&XIPY6*pcCAG zeqUR^Yz_-cgp|yBx4Q0RCda!rTdFOEtfBFBo`_)en=N|cq1OYjWUxC{=q9A9@2>Z8r&5RUEjpAP7pt?1(j@0O)LWidrSX=i~Wh4W=6ClNAWJp{G2kpa+3j>Eh4hJO$B?3gf4g>G~ece%%trGtf z?&4Bc#{JX9v8*temRK>6D8q>Vp94pW&jRy(P;X>%55+l~u$JSyxb@`myau0=y>>-*2AD9rPLbbbEqQ$~MQ2U|?1A2?yII2T2dFg5b?W^Wg0|stReo zv6P3JU)fDnm(Ir5Kc`e-T+&FdQ>QO_mzP0Ab1$9+sbuHEtT&z*IczZ}h4Bj)hsXUg zSJ@bXOavP8N+Tt%XL=^%NuS{zQCCf<2awK#)`~(r=YfaZFE85ICYaxe1A&&?h%*Bm z=L1*5-(?PsA0en@@3Wy8(f}%AYv*f*4f@hk4^7}*=?#f(;6_f`WHS;14>>F||~i)^EXVgKlG}U&a65od_mBeJjH0Z=Azi`_)H~s`}TTI7dy>IwhM;t&kQk z{(3*vaubMxfCTA6=n;T}h5e7z{v~hRFNpm*UkqZq&KO^ZS=R&3pTqyuZ8$@W4e{o? z!$2dzGkU{8QNbEO!4a+27m9R2V2dGk!_2JVhem zhT7exjFKFt3p1XlIRm5hJ%#}zjOXzy>9+bdSRTY@suM4ftrt9>_DKNlX*y$$q~70U zp5fZJr031}FSxycJ8rM5uOV;#Excly9Cx~}TTd7n!;;_Kw+3qh1vG*IKFIWzfU&@q zadLdpLA_|Z?_@voJb1S^(q8}FQ_Lt$wHn`c%vGK>4H~W&!%G|iyfDD$T*|w^WdlRL z_=U<0{zQ&b@(Ii+yQh&jIn$ZeQQd4KN0i3xtn0Cc=3t4a80hkUWp!+~ijKJVx7EQW zK)g#V#PVMLvbtzez7bxNbrb5QPsIs`e~REy|3kemx9Qyf_YIE?g2sO=ZQt#E8DFJi zRTAQ|mkx^NI5;sU2dQ|ayjKyakerRp9V+%>u0iQ_x(LpCasPCY3XQQoh5(hliAYWa zAp)uaO%C%Cv0|0fno1P}j%yes7$7$|m|VA12o>@}i|wuIf}QKMO2d^U37+19Dp7h~ z9BEm~gVVo;jr0+!gp?6$!o`*=v8cx_aLUJa1=-P7)vP$fG^J^}(yeSi*_K}Qu;402 z|KN+T4OxF!)lM z1NI_(7#DG^Rd1AsIpQpC)k1LfI;ny4B*luaziu*}o?Ke(@;%QFSu%+Mh5eV+;a!;Y ze0p3Huex77sn?YEvO%pPjw@&6uJ3@D*^Te$%u)?q>=3awqs?)`+IbkjWiyc6=E_56 z7MmVr#=>Kn_yTSyII9%I{pt&4cY)>C5YmK^CaD27g1&TJvp5<}b=Q#|}V{Kkn}VK&J(ZjNN-FZ%iE zO2vri)<6 zc6X=?_gTZ{e?@y893OAF`w&>nGiy_y(MS)VEpWWTp>myxQ&J`Otxy5LZWn!gqBg_b z{mC;Qj<#;lrP^SU;CT`q6^?^tFU`>_`qf($JAy6E^c`%h!S?#M{cFw;r*hrFrg8Zj z#qbXy`8OU2k^#3U#)67z#7{kDd9>%j>tlZ!@-_DE^S!RTaD#Ax*f+Wm4OQiO2IZR> zKomE01_z6Vf9L4$fXo?m6OeI3Duz7QE`amKBV3mde+~clKltwwZ7q!(w7ZlU2WB*? zFOxs3;z#OS>`*o-d$}_YM1~rIb~o?ue##oKe~xK&$X&I(^4x7bw8AH(NUE|BY6|SM z$sqTvIq(PoX7tXL6bz=00pdq^ep5%3S|uuKk}xUdCP84hocU4s!uRZH?NWH0Nu$i` zFe5~D9#u0AE&&{=ohJ)}#^jm3#8pqZ*oVpy8VI2;T>KQEB#uILzfsq}yyTqUQt-$F zh7k=Szcw||ZF(3-s9Z;H@VWD;eRF&4&?IweIfWVQ=K6xd9@@j*!UJXIUDBg1OI}_F zFZ>Rr66&vWf>YBEB~$A-UoE^f4QX6%vfmy!&Dz4=I)6);N8>z2CxZzf9rVO^VcxRw zI+m_ehOg5#Z6E4z!WRnG3OOc_9FpA|2zC>c6L_1eKf^~paPky(uK5dT=wO)KU<_9o z_cKGoM0VU=W*Q6S=fRc?Cn1YwAoU zrj%=(HMx?-mH~4DYW>_dck8B8`zP*a)YONQ~`Q&U$?xm&8au5!J%8tSlV$p^-PCfDozTsg11qU~Ej>aV&1Nr-hXh>JPgJ`Bp(b7~x;;SLc+?y1Pph>Qzp8Y0qG2u14FB9~Q ziHdWVsI&PAV@Ye zCAZl8Bo|PP17y{Ggf1yIOp>pOQKJ)vAoboc@LBsgNJRu^p`8-r-5fdk{6`Nl|A^18 zG}^!@uIl@3)c2sr0Wvf9&Z7LC7<3%MtUMk>c5X`)A;>^=50+{wOz;Kq(~CWN8po*8 zqihrQcs2I4$qqv_2u)Ds$NRl?4cZ${FV>J=2e31_ zxgeWm1)rDMMg%&C4^q{M_B1%40%#ce5(R9@?^}Kibj~A)t?EUOI(to+_g1?9-KCzm zJ!K8iR?87+k5?)v;=Yp#{`;Q86Ej>R-m{GAT3g|ETGFOgHRV%7aNhwb2L6#nfQB65 z&=UIVtYq!TDd+nNLTwCZ<5fu&BP)I5--tr+?4;gTt&eF24MjBJmyNF#YU-sl+x{W%iHzHJjh!9kXWq}guba1aBK?wW;RaVUQu11Q|w z0HnXe5K^f7&Am6&dz1F!0&#P)v$I``@#`$uui?LYy1ygjN~$;KnNr2yBrQ{;v&#L&O7-F?1*{gn_B&2q6A0`ix*ZdSZE{}=jX_`gX9NQ=x#nv!-FNg9 zK6G_vtIcs27jWS4KKRCKy!6&RANy^Rrg4Da`@ND*L47Wty$w0^a=3!m=s_AIitaPe$!0N$af|z|VGqI2nvtx4)NxciF?9 zzZEfibWe=b>@=~WPv87&r_8u)#G?W{%`{0BdKO>0XQVCZ1@;6`2(KkJLl)QtJST|6 zmkjJb%~G}Ps|`&*i?QFbdUh-Yt9|jAA@#chwI3!4^y(kl@2`o#e<0{TirkBKbMMA- zM8_xXA}!uTZ+yz=$IbW)Y;HJN@^l)d1S^u~y`1jxGDe&Vk&jfMN@%`O{Fv5d5cFq_ ztoTxD73wN;GJyIrw1Ei0Ao8SBm$gcS(&eqeWG#yGJ+jf>pYEq>cMuLE&B(ax}^@beVzz$$bR%* z`edh1uN(?f7lW%w?eN-WK%UrB^Hzz)tKU+bTxAO$@h9{&!4vAr;XTApAX0!!_>f=& z4pDmcPy>$c!AE0#(94=RryAm1?-6@~vd^#}PApf~^tm-)sHfMg#_Ypd+Smmz4y(nt zgGRK4QZMOTblM*1KREG61E03Cv1Yx6UphfG!N+g&sb(Yk@uS9VisI`=aRJ6O$)v=R z`zp*rjPAMFuCkeajFTJcQ`Hq7(l>dU?C(149Njz>gDX4CtAQQnJfE!J&2!_g!yCwI zV@q!3l$U9!wM!p(Vw)m^>~^hAB4XO? zh1ms^T2@tI<@>5mLFk>N8%`$BH#Ro2x4OxUT7w;3Z0s!{?Q_4FnF;ncGs7xCSXmYl z#d6#-Gt%=XaJ-~L@yvHk>st1T|Cy2h(7+-$Gu?9X9~$`1QGopK8@SfM*NxebWGDxa zhmGT!h_74dKtG57rdj?j_AK}7W+z~=cw=dyuavrx(d+g`cKMB0e3z<`I5^zIXaQkS zhgMEFySju3YKw%bAjd#7@@F;gm`@9o8K>%;1*2t=kV47RsY!>--&57bL$g?cyBhBs ztKOFp7B3|kop15~ID;u9uvLA&w@0l!)dY^S8UG<{7xtndW~iqz^(ig&%tdw|4F(kb z#=}Ru0(wnC3IkI#B8u;DxkceJY}Io~0=Jk?J+?zp7W0gQiMm97Op9~r_b#|R>Ui<} zFv^?jVv=Le{^bE0enUClAN=zNy&zHytR_vgSf>k@m$)lnDmb=wH2kAp{3~DGX)C_& zw)253s|Z&AfbX8$axpTVn8?wRRL+Jgmyr zk;k)T3H;PdW;NASK|92x)~}?@CqCfN(Mkdj)-s4n+7<}|O|EjRYf=tFyN?=G8}WYq zvQ@k#11pu28V~I?1D=uS?;f_h6dUD6&we-a_0ijx7UkH|f>QA(#pjAb(>cw4 z-!yrHKh&SaK(O;K%&AWQ#vEJ>&<8FOf;+cc%)v=mDEwmsZONYjhY!TP2|97mC2s4C zOxH`LSRhLrxUY|+;sCGLO8uOc1%i22F$HCSnFGMg+R)~Fu(cuh zSHy~ngHLBj4}b)vT!24+HYBZmE3xkXJpNNs{p(d1*Q>7BxY>Yg>>SrBi|yx-^LhyM z{=e@l{T-83##D-lp0-&r*htO3k>yo~-~Qx=C1T38wRxEU9iGNRcYcid$e2ZroVb)& zzODabOi;;}D^LhK>|~X%hakZ3>DC7|x}1}_8eu!`&s~xjaSWy(sbUQ9CMCy@`WUZ# zWWKx?oROW3@KO*$e&q)R$z%%F}!FI0+^$yW?p!T}L|lb1`GEG#o0uIE*;>vVVb zjF^5w<~8nJr%BU-b-5C~aO|}NKI7oVp!c$sE9MDcdFQEK{Enb#gS3l|@>wP6*lIc) z)sMl6v94Pt@h$;+zyJG&y=!A4!-Rt@;q$t8ZEbf<#T#NP9Av2x36dH&3y6<{1qcfR7x5SGc0*jzL%@iu(*J+1ha$`~eBhx|efzu)Vk zfsRzLJbZ?`5bgL8=N*twVX<8AGwCjsc$l4JVF&S-wZsG{J)A4sm(Q@|q)E!~lsvwu zOf)6O6k8;Wy<6K`5(`_v66b3wZtxX#PzuSf%c1W*G9pEh>#xr*OkwD|^7Nc|v0l0b ze9-JS!7-ri>Z`Vy#Px}00qeVoG+9(Z=-I}V`?mk%90jl49G%B+Sf0VhKU5N#`&w?; z3oTZg*|c$`WE?zk@VpfD$JN($RfOawvaX_3I!k&YgodLITp*0LgUU z{QN4f@B-N(Qi5CSe_+OcQAv@Mz=%D2q1&{^+fR9tf?7H3K_0~?rAfML0Al{=PIpP1 zqf=Jd9k(hmq;cBsVkMUTF%ZmwLg7BW@sK_3bFy5fjYtfS=NEFnjsADiQGss)(fIry z8>b{|M$5(NBLqCYsigeX?fx&q1c4QLsl;`}PD=5|G&9yS1HOGJns?s?tSMv_E({22 z*$n%O^<*2U7FlFI?&RzuVc`Y#m#ybUgI(|sLe$1nSD)(2qxvRkJJY8k(@>flzyR$A zD#CTv`R84c{d1k7_&aniq;}99sdAAPtUpWlw4>GM=$q{w^UpJM{a`#Wj^O^+_QY{( zPk4jB?Fo7rVoURoKY>5(iGe4RG!_MJM)U(+7V0;Sw=j3U1I&N%?Q;Aw9hAGK^XuUM zggyOtX4Hg~qkmWCZp)Lf(1@_2P;k|4ZiC^L4=_?#c!@}1Gm8?Ic@dHEI}J-~0#073 zH>>xEo;su*gGH~%o1s2fJ78S3Q!W!`=8xM6?yCPy&V$YAPx%6(_qFwq&K1GA`m2SDhJlbNk_-b!V=nbA9?u9Sndr z*PoU%QLQawUvi+zngj-lAm1b#FLLJe&87$HNs8Z%Ng!RXEO)Z{C5)u*tTMuch$1wI z6&&7L0c3eF;4e%K6ucvg6l8y`Fh#+^R<$;B2RqsTlxY#K=Q_r1Xx4n0RwyBlYR`A%v3Yn2VJO5zh^$K2!lZnX6bvjsVuaB(H5(Q zWqrJ6oJTX}8Kb;~;vZD|gg#3&^l9a}KL9G<6g7?5A3CY^k)cmeB~|J6W&R`0m%W@! z!<7@d$5QOLDwP}KHa%a1MhZwKV)@B5ZxsWC=>IXe_$R?cby&~NDsFA>c8PeDWjU{4 zx}Wv?YD$-ZAs3&(6@ZBM_QDBPm0>>tcO#MWeNs4L*$shw6C38Cua{tu?Pro~7?fH) z4=at91nEmdHdKp6zK?|k{3YK8krhKRecL$ZJ~Pe{SL)%?TC%x>)_<^n+xxX!sHWq- zDDH^*`3j9#9@(;2L=UVAo#3(1;ECLpISOAiP9i@pHI8ADP=6q;+p?_UBJ_KIU1I*y zVF%32!Tow`qa!V~h}?DzS;u=UcKpX3l80tCc>7pSeaVxaD-~V9Okn3~nh;Lb+fFBz znr(UY1Pe=wh+Oi)npu#h0)3L?5tZdzIxV6Y%NEI`5ez-{aTeM(FOvK4a4cZ?{$a_Fz8kBRc5 z@kXPOkWP$x;n&Ev9qHPrea_a*&@-g_X83_xEU4LeqX!^krTbcMxY1wysgOey;WOTn zi=@)eNoSv&PwEk2>1SphNbUH=_#?I-&Af`lCwaigL=zH&G$|XACLU8Du&ceE>C?0K z9OsHu5lH{QAYhE=iTqhof%DbXID2@r+n6(x?N=7D_Ny9oVu4pJn@xVY;|~4qtjs?i z`LLzwKN$(-($KhXDV`6~AI(>7Jy`;-9j;-qIbW8DzV*H=Ox<13U+~9ELmc;bP^h|F*J<@SPQqbeAuPy)49_$Epa0_=*QW1*Z= z7%5*Fb$u~=o_xUBG-?v={|d%{IGu)%Oxwv%c`~3dz{^)@{PgmB1Z&{C>H}IRIm~-a z;25mpQDN#D)6Sd`y9vgc4~pdb8CDb@)y#1;lQOb>H_yNJZ#yUM5*?1q!ZRY1uZ>m( z(V%$ldZrGx^7_NR>A>2rNtzy=Q^U4c({;P}a5}=VH(hD9s`W5pB)eS&dwvoz!=6Xo zb-R)`MJBlgJ7$jOzH=H)cRo3BhJ2cFHA3sBcSl61pB|;G%ocu$t5nK;w8m#!Nehi0 zc`q;hNL*c9-#b6SeNZ=l-RJ(RD1A212cz$Ogtq!I%0sc6Rm=KzO*A><`~%Ov>|J~( zlpPbxo#nSX_)_uMwyG-Q+qSS=F}sMFTt&&%2AH5%(pnE?=aILF^c3{0BfWA~Xge?P6cj`pB z6&De)s)qTE{20S;zc=xNX0Zb61N5mRRVAIt$r^O^g{vEEZ%EW9r7~mw-Iviay)e#R zQjJHgHLuJ>^Lvv729px?D^K*gnz3VT<2u&G2lsdMCbx@2oh^p1*1!&5FGR6f;37Th*kMX z$Z=3SY{kA$vCiAv^JtAZYkQq_pY*V4swHL03nScV*AjIMThaulns6_xZs9qn_`MgW zllN}u@%Lsl?;+|T4-%I|{!FXrmO%eCLk={$BOf3woLfXC^Q00H&Z-V=n zK;+cs<7@VZ=V#KzSR;51O~)hbaIAVkqC}XFf}N=9eAjK|6Qs6v-gv*Hz5k{y*l0<= znxT227O|ZVVRR%mp&e=fc@{38D-}Et8lPi=O?d`BEu(Dlq$VVHa{ODNQO}Eh0Z-EFQ$~K@VcjGG!U6> zds<_UT?V)Zr}ZUH{~h(`A5ToCzgN87MqwhsU(oA%eGltA(Tj-lN=gJFT-c4w2udjL zLu;DDBq^*ZW$}~s=|*X0{#z+>z5UE}YK;sF4)UUX?!3i6lB2?FOyu`4&^BE#zvJm{ zs5umzqr=`NARfS7Tg}hqg0RrP{|9rYuM}?gu895Bqw@suqEOh|b?ex!4cWUWUUXza zDbpv7^=WZA~5@Mt3}umyGf!cZ2%)VQ2~I@C5ye3FMQ1 zM}}~u5S>E^K{nJasv&E3{;U)K)Jdr<@w+<*2UWz#=%y?I5E1>gsQ@Yt?(d^2Dhj_! z6L+=>_veP2+e+O84v@YbNWB8c&chD6#`N9cf9f{=4%D%+LzL2b3l`r* z5Yc8@=caI|zczEWpN+_4p_n{9LeRFzLgLq^$LCuO{J??mxgI_X_PI|NJ!a8-kaxq# zyTMClwPw7YMVe$h7R)n*a3I`FPPt+ERcTVu!(x|8wa*m=>1$?u0vPOWY(lm^QStg# zVL+sCJE3eTb)u68Q!>UnS&rdvi#mRMM4(zNbePRzIa8V{#RCIQIgQO`W|{V^>w%L@ z_pv88I5|xtx}ZwpC;=5(DuFhKB&}CH_Fp+_rVus7K)~y%`V;pK=0l=DZ-f2 z^4&L;4eb;CSjd!(&Bf9em>zc!3jI%ZOz{rCCIxCZ??9lppBL^w`-C*JOq2i6dxun3 z{$lqxIZGxhBNB?Q=+iaOr?G|{Y*Lc(MR8jf_W0Qjl3Irw(`!c~XA<4?!Q#42k9Y;X zZP&}G1dx+iE)vj+%?GP7=GoZGOS3Q9KWCs$s29AVW7x{LBBibsQ%5$DlQBSGKDn|? z)*~Mym3dJCVjb^$u6r_L;Y;Yst)%4O%eVJdKr&}F~zd(?W1_<$Xu7YBrFogQtA1xAlzPH4==Ree2}ViB)2 znZz^Nme=y8(hR+{D4$aHDy!_)+5}JL=EwAgEZTLGlK{k#KDt*5U$ic9&XeJj_SG}K z9OSR#pyoN;5y}E4+WMuRyk6fNXmO(7)Zb${kXkadH3y zY^*Kat~&&;6QQs;^#7I0%OCqVK&nBIK0ZZ(YDRL#0Nf$-7T#W zpF`u^5iA(6Qu0u&rTA?ohp>o5kHZn0j+CDz2_6zKt(X0~_|FGVG1v{Y<+*%DVrJe6hJb zje8&Y&WlxkGsPjxu#^aF-+ z{vaNzWXtG;R)HoT60lJAJ>C5CXrNTB#J+bZ3%3^NsC6duSG@;*jak5`CqJS%{|s zJ$4RX8l2|o=$jxtIwkM%O3Efg;`=Sr*g4ra>l>Ll+)~ZYr6@v(znKOh4d{cA3}GAp zTc&}5Z@c?2uNx^O?$C^=+^@YZQE)I7oov*uKl}HVcRzQ}gT=xAed=0?Tq}~><(=0R zG!76iOG#vwA5JN8zd&m>$F1)G|iYNZMpQ|s_8KrhivvI$_52E)s51+Q?46R^3 z*UIfFuhN0R_$0n<3oVPXu&Uvnq?X1k8CLw#cc%OZHbpN|PG&Z{>2uy@JP>2u*?UGa zA0&73)OvIr*P4jkEQjAOIJ#9?6(>#6RrsX(OZv7)t&*tkJKQaJB`r<0oVfF3Ar$(t zsV9*bIQ26znX@7BjfIKqK|2-R#UyQ1${pX^x8rb2K0kL=>e%hEcG2_8USn_uY-vCf z0HfE4L*^U|$IYYONXwVMf9&6yK?OWV-ZG_f$PPkNrPTO5&zHh5-8mmIH||hi=~ev= z)hIeODU|4{YyfRDk|yv~S^B(fI1|e9S1VJ+jR_%m&k(j1936|xXTl3q9p&XlPey8A zND1bep^erB*nM%R7UEBSATx0#ym8ApKG2Ys2I$CtOzC>hUyCRJocB1;dj#k`_;cY~ zqXrZ#(EBscJNj3^1Oo%~4j}gS6|r`^^@7F#M^gwE{@CLS3^0W31LkPv=;R35sfBIzw8%S%a8-VrB!RtLE73}p59U0hApf`%_v$lmrWq z`S(ph<5T`z&v)(Te}VtDM0&HN&(PN4&oc6lm~sCj&YX{!;Mq2-30o|u&SZ5LOdrtt zR#QS$qZjT!yC6+EQ}W*gU7vsCo}2!_C+}04WKTct+Z-FK7Ep6+uhLI^(ylD=9q;Q~ zpgcR!%i-76Q>IN%{EkG$u-cSrznpa698?7Upb)YG{4i}}f^q!A5U+vhlMl|H&UtY1(Pl)=7_AQ;Zg|m7 z9%YN8r)a5aq7I`gNqsI#*=P0oyw0Y;i5t=Tt0?>wTLm%f-03w*SYr?T2E(2#5E6KQ5#e}WzObVFjy^5`wh zVCtCaD};EW^i-FA;JdS;Sr*_$@$H=HuB!-lZCTQyha;w6!bv8KLPC`*~0j;M^X@2;CGsaj zN-3Zh_dj6ue$Pf8@u(>yfQFd-@+B!DpH7k<0Rx|-FFWvsz6w^?R>9oLK>Eug3md{J z6jbJmHNhQ}wld7Msg_3sC%PAz6Ibt~wUSoYSMJ9i3)N~flDc!YjF?;&7%PNW(djYn zjOu-AkFF2niahW%K=E~d@LhmksZncw!yPVVXGpb407i^?B{Lh| zriHRxviR(&1y%vt5C0AEwG%jp{k2P-x1mQ4#TMl+zYgiZFv{}Ie~5`;77ZL7E{hPW zvc%b&nf~5-5-im3bQJX|a`;tHU&Q8=)l6^;*e*5%9rhxrNgKZ~YI6P3?fLtS;$!@- z$*&I>mnqH?QWg}u`IOP5RxldAlsh9RtzhB!+Z)2d&B1Xo0*GL~Ip$DQeo-^tb$T#m z(u|?67gQ$LpL?D$K~~ZaX*Zq~cedIR8EgJPRjEum%Skqs=E4A4Vm?}4p6r;rTJ{=6 zHB7x?PrFykgon2tS2mDd(YPNPOHLe4W|^taT@&EIJ{tG`aQ7BqRju2*INe>+DP4;O zr5glk>FzFRq$Q<6q`OPHLmEUyq+42ALXglq0a4uhoW0Mz|8vg$-M`PnX3xc3YtFUS zH^%qABi`{mjpB)4j*RbO=KDrd-FVn+YV;i(G>>L6X_-o>;o{xV3nYcN@)mF`U!$jx zB8{{lJMV>&3fP33^4%Nz_CzzdU*~L4eRH{7jOd_r;tUHbB=H>SY{8EzR^{0@auPjbPu(@D zs$ZKw1dgp;UY)hs-PT*d^UDOkBe`KhSzIPs7C zdvOYoElsM^@lxCkZd(G&e7XZ9lu1A#)f}l+>$aLcW(3#+x~DQh${FwLr@4q#UII zJ;SrkVw-)amdbz|KM3h${etO3ls9ZN9nqMV)>Hq)m)vjypLD)*z%KV2*rZ1B_4&glF!A=!eLg^q=1QyEBL#xKc*A6%#XTqmV>m)Zbs zF&a;6_%qpT;~inY1bbg4mcsNePYl`|K4!-f4Te%_z;h419Y^-Tsdml17bFuNnma0< zu+Mi|*S1uXLP47I?=bwHaLIm)P0swgHz9uYa&?=9b_jPhOI68AsM~lTAZWV$+g7j~3 zm;WRA{vVPWkeGLV%J*MadR^Cn{R@5b|AkxlCzW2a`IAkd^@wMP-ekjjt*wp+Y})wE z>>G1E$<0IYU-kxv5F7P16~w*L7}Iwm43X|^!P*_Y3t$rTPgxfbvc6yXAqPvH(=`=R zWCsn?%3-c7I6u>}M&zyO48?ay!t6!<|J|PXs%gobubJHwWt>8T=!WJ7w9OBCZhu=h zm0qRI`&I%N((f60IuYO#91nvzp9NDltp%lPG z!j#LrTv%UvZzM6Nmhw}wC751}-gh}NDcobzsA7GXl*s+YPt~7dSS@P~RM)i$-aI2J zj)#huI^bY@z9eNm>#}qv^`N>$1-39UzN(s{Nyb`i?@M82JgE9YM&%i`v?`XTU+Npj zm=8G8Fi6kj-$xR|x8BKIB^$W#UqNg3&qe4mrTOBBg~ccC+!=%wLdGszjw*(fCzaXQ zl+sQF;Q1Z z9DCXjckip6yq$y)x+>no_Q*1ru6WEl%mkF_PWu%R ziJ7*T6+-skg6aC?&`4Fr53s~m7DNv>g5p_YQNAgOpE9I=4=+O$hA91E#}Fl2R2GPW$&rX|`(@&kF1PLY zGVEpb|8?v5H2~u0)*%VpDyh)r_?N3dfY_s(YT~P89S|M+Iq~H>I`;GMzkD};XWc!6 zD~_vYBTXxejI4R{E4~8#Zz!c|?ddv4j zclt4Rj`W}`rQfl1t`Y|K&ViJ*eYBThZE>D%XHLM0oPk2?N%z(?cgzyZ3S2Z&7mbk0 z*=vmuY%lwDjy72bc)r$q-13?(ECk_Iy?4cu@=L72ur#3R8ZkE&5MBrgkv}xn5Aqd~ z&vVlXnZuJ-9c5b8CRAeZdhrGy1}DeNaflrXQ!CT{gNCQdw2sk7t{~7Os1@{TS|5zh zZal$R5a^!NCUwDeuKNQB1>D65MjdS?+VKIeGAW6JwDIK;Nihb6nrsjixa%MM=py*v zoC8zA-)Rnc(k1~u@`4H<4v-NxqVN1o2l5}D@O$p~DzZ(850fx6o|4WRYytzPN;XX< zaYAQW)P~FIEQqH1D)hj865Tk0<23DDtR`(zhUGzh^tuQ`M3w8?=_MsM`>_?WIPfWz zprpG-UqVd==Tf-d6tGvACsXvLxsub{i7cr|u;}CMs&zkQehB)K!Gc54x3*}E!}huj zpCE(pWI!j3B}@-4gb@iwOR=u&E5eKS*t+Q$t)85Q+%2Lx-pf?<2we&0$e*4qwxg-m z_c>!ktL#gfPc+j(XrhXakg`l4Hri2XN9mY9@#H=5d`qi-3@djCg8Ab|k zN?Y2wk5;nbn=OeVN@~*#t^+W?C(7f6C!d%wvt8Cnq_4~$)704ZDiQBwV-9H5I_g{9 z>!|Y2GGg47EfXYD*JyP{<|*ocQJuHY?X}co&}O;sbuowuUhFfYjV;uCmqyU zzzAg8^!u@h#QchTh*V|vExNgId~6!*}B}Z(wV4qsM6IGKZl1<0nzl4{{^=;^4r?ja~`aU7eq}p zFqxHPI=N%=X)|JQ{A5D;>RPJGQ>x1_F)vXLJ@)84-`En?2Ac4vGeZxYmBRbEA+?+ zO?poVnL1?A%*u-AH}-gaYlK2yrMOUsKTCAq*-I3yoNcTx|Lvkr~|_+ z7N#~rffKWgV?$%84w{z-PcaO^PoKUM=V!;&c{eRdVoYP;VS#yw14*>iv{xEIX)lT6 zEQ1`L@r8G+AR5!~-ZpO<6AiyP6H|Kl&v=6pqDpwYO%zK2EUTzN>&^OPr|V=>d&e;7i2wh)SMFQI%q& z04?*vp(JkJ(-jhq711Rhw>Yz$rml3pkL+(i;$h14@wvOl0*L_^9INkGdS)!z^2aIQ zKEBZD)PQp?%qw?cuG4D~6}Q+;`0g10Dq=nM?1H6kOBEemEE)6}KbYs09B)3@XzzKk zwGX*nF4BTF6fMNr>x}64o@1R}@}sG0twP+J4Nkn+lga(k{_5>hN*8QRoVdwPQL$%t z7)zXE8z#ZSCD97bj<^FqPb2iS1Y}bP&5gO(J{@^VD8Epq5ZV~d@4VPBK)tVP zf3t*#(Ub{i>zD36pf)iHEHR7zKDnwyh#ID4 zZj37_Um;#f@IXfB>*F3q*h@=B=1()EN&F2~WT_6)H|6Pst=JYgSm-BpZ zLyH)6JVDvHxP?r)qk>57l`B1KOKU{fqL*z zrpL*Cm}gf`c+Y7bA*v^W)b<7HNeP9n9W7^0f8bh066tHD+^1`NtxuA^-4$b-amL<8 z&9XO=F=p~L6U)DHCx7@#E(esLIVcuFW8tyF3GH=TA`E(=lHt8sHl5KeBv>6-cT6Rp zcwyrRsUO|#`BF^KMl?12WySNy`i#hhV#7HxbYn?Q2{r;s7h;KeL%^zBN8n>aI+r15rx>{&I2LWAu z!Plt&CSFATYa|>L6Oi1d10Vp#58ST@o=_*AzfoPCH1>e`IO^>7`@1TuDf@lR{Tlgr zJzVFN{x{!fZ(e$9@&egcsE zyI##VS#@9>%5`F$8-N->3C`ue%fvbtP>=I#(9i#CZ|LvXM`ar|!`bwN=B%p+L7Iz+ z?`!^M&$5F&>Lm7TGZuSpHiP8@dlR>cLgVb*u+d08Mgif$B1_hGiq{N6v&-}6`YDT! zmE`g0BDlI;wIr%m2+(TeeVj7)JagZ_3oH-L7lpulZ10F5tPXKPfMqE>Z>!sA>6ewN zxhCC04DK-f;PLu>Z|^50P}Nk)3)LDW5_~H4<1eWcX~0$zokY%KNbn5K;q!S?iy6Q6h5hZ-|)9|z{~TOle9w8Xr{gK=0z(&9Gng&J$N zdaL>6*eqp+5?eAj^Hbr;iMSyO>+xu(uJ5(!tNs*x-1@FgDag-yDDrQ_yANF4j%%?OxTkxWX)=%mT&NyT zQesT8OyLvH&}0{f5AqM+!f{TeH5nk+&qu|lnp`&i z=yr4yD9Zeo9Lt~F1=qjRU0kOxxqv{!6;b)6^ZIMl%efU$5Q>cCIl??$-y%2qU$u_NeouE#TdeR*D$NgscMN*JmbB!a zTV!+ASY_948l1&30*Lrtly>a0oE9`3oWb=(zdxgm|5sw#x6LYP0t^JUX6gMDcPFg& zZ79+S`p2f*-#Vl@7{~C?Lb;>u`$Q)Hs39^P7}59MwO z(Gifags)$kR7i=D2Pi1#1cb@BZZyXsk?!35`38ag*rcv{OMn`5R$v$%D>uibv^XGT z_UjPz@4nvOk>uPdgm@hJrTBfFRA&|#R`=x3XwLOuNi~X`Z|+LGOvg29PWo~>>NXK& zAVbbP$IUmkQ1FS6VcHe_Q-qlUwnOg$ThMZbHlo&oDl8r7=uH=w3ymyp^lICv;)Ok< z9sgQSSLo6n4dS8mi_EYfmCAdV=XLzlToL3;1#)z|dQE(FJe?@u5{cRva?>KGs+rqg zQ(Ms`s5CKO6l>#zzT_fQO%?kd{G79`X#m#4Hpp;%h}B|D+Y0G(l?~6LB?}V|7&X=$ zcl#EeNg&30ssj$Dkz#Kl7vY5;XgdIx=&(%dyB#LjX2lVk`-3r+6X@^<<}Z(`XgIa7 zx)dB`EfGJn!%pAf8K0r?usljXV0tBYhYsyvaA1A&!m^ktDV@>J25LkRJeH9!@Z9Ts zepi?%&vLhcwCA`bBR~Eke>LRp#Nab_t3?XtNeq0l29;=A#Ar7>q*B3PM7u@zVU!6o z$tFKc9yGO26LhdW7}ILg!}nFtB%rZP4ZLs-^TY+(0sds(Wrzao4XmDvz z4#8o#jM&1paIOq6s*vGPYzH;& zJ$nH0vbOFow$EQAS)%^U_Z8cnlgcgThbUhuUqtxchyyCB zL(o8PcacM8T%cr;J*qy_*7+0%Cr6Tt;Zsz{KA`%RKH&kdv=sf0y-3AL4$q z#Yt>QC2WUo`t)F5Xi=+$?MErJL6&i!rl!7$EEFTvjjW@NLKf^{Qj4u~a&Ve=f%QJb zg@fq1y!$v+5~=6IK4e|5ha~KTG6($e;25m;wOvv|u`owC31jwFV7O-TR-=-+Beg6E z!~;cyb@{WH}=gYoj`{Buyr0!;QR5N2CAz}&PXqFkN57gLs zMI>{RGz+qHAmQdBCoR>{)I`_5lm0ueGG{|+cR}OWpc7)BXMrn$W05H5N(_4IIQsZm z{30H2c=VJlj`l;Sz|H(6v)I=Ap`t6%+0pqohD7j4AY-GDzANyl9tzofe4Wv{*>Cv9 zdT{xKv&bxUyG1*}P87Kog=6L;l6F&uLfmUFoK1A*>=>~=8T-lR{zOGpac22^F_SZTYXTIyT2|(3%JR^4tBV0CMcyP^J!Y#y@N-=U@Y$@b}XJl zOm0IyJ9DyL+o)H;fyPNXT+pcqZZ~?Wg)JN0Ti3YY^r;k@FSR#Ba-oT)5Aq`@)<{RO zUEPkHLp3<`b3B6ocr%M5IQ#alxc!19L;;6`WkRya!fQe|-3fdsGhzvd)l|Jj&IIuJ zw;VC^t){)!F6#FqQ^%sP>N@0c$cK174Dv*-8=PxX5^?lvM}7?l@Fzz*As}QMl!(zGu`WklL46}9YazJlzAyw;x`l8SF$TRxA85``x?k=cM9P8APR#9!y zICsVQr60bf#~DRbBswQC&4LhPk$Zb^3YqVgs$?-aWq9wzXZ%Cr$J~#tBP#YUIPSH^ zZj_Y@S4P<`oRi6N7_x;(jU_)E=v{jV4|Y7&yPIf`=u#U`t`nM3d|2=LVt^YjvEL!h z_05Bla=Z*j494>gp+?{6DI5)!y06vtJ{XdT4vre?O0NtR*|zihhj}3EP8TYKO=ydf z6}CajoT-mG1jZ* zG*&V^*HOudG(M?cSt{zSWy%7!dXITbUMz&;UXSJUw`nl5f(1x)A})Ej zbXjS$IF2f%_i={#q>swy6kNCH_zol`<@pij#Z|(ntvk_Vk)stwCW02UA9B`^4e0U+ zLg{LH4QBVy7YY(jat4e^9>fr{i<91cm7@?LjlU}|V6KhlE!No!9;whzvF=b--TVmN z8{;)$MLBI0Lchu&>W6ClZ3zwy>FpFpD6*lV79`9XumLKoE=#01jrh1J`J~U^bLmZ3 za{9S5=7~t^;h7Hd*G(0R<4G38`6qHZ2Q<1_eAXqC(&Nm(;muAsdioALpNrV8Vudyi zU|(ImP(`cJb#7yRiH6*LZG?W{_&~%G=da@G7gJbW>7Y5X7GhzZ^I}5^sfEu0e}W7^ zMLAGkP6Flo8syst^7YB^%J2+yw=&Yd9NB4P?r_hZFSvVtx_ zHSp^SV*|h0F8^N$_z&yl53qU|2uBW!1gB;MIJXKLQ~@Rna?P z8q_zRlk$Yj-N)`N7KV8+R=l1lcuD0akX*8ShslIjIH<;s<<*nbRsGOO1a&|sz=6#xHi9m(C5c@)}%&X^aLa4rObc9I^ zd=TfGuw+L5FA5{_BkD9Q{`-S89!i1M6I8)=;~UX$_!|b=qii&KbRHXzYkkXJyml|Q z+1StV$x1^%-OGQAjj`S?GypXGujm&r-{rCm=Z1}49sZlx*stsT9UB|()j56zJK}OC zLUwp~WWKz^3O=13NV2Ss+_>iCueiBEsE;k#L_6=+*6d# z^zW6ipYVP>uErBKHPCTc$Wqy=oix8Qdaqn`Te>@K*`3;R#c90gHRam8CBihSZ0y5= zj5i)k`|98pb;$LwzFuM61l+3#s)>9%Wb4^SNI0`$TkH`jbcA5ocf!s5hwajF;ohK| zjrLnAzUdN=oT)jOP<15!6vqSA_F;~Du3Bl!oG{C)D{`5@u5>@^>)A1mk%@V_ltR`b zHm)g*wD+i0jcgU$g8rs<>!%l}tm8xvc$qZ76{F$z=QqM9U<3_|(3ih;a;0G@3LGvD z?(4_Czc95NS1fBYD;5;VP~}i^-BUWz7lt+J8MjTA;YKzrF(~H@%W_0OJ|b%YL^3uU zAf={zb%_iNdoO0Ax7c=$tgCH6W^;XGR4<)$JgQ+yUt~J7*2`MS{RPJP#m4%5Tow#s z{Eoy^4jPYTI_K)MOL_j}TL0*T05+ziOLo-w@}43OL(EGnu>Re*q`NTt&(l8TJoa|S zIj=jaJ;NptD9&ql&_=7U-)WXxDlLJuX0m50?^-I&Zb3rNul_Dw>4AnulCLUM*(v&~*o9o}n8Xkk;=t-ajfabu5Q&a| zZ|yUl=Pw1G`wH3WMJ=v>JtMcDd{V_H-c}t@9z}>6iD6N`yF=_=1|c1Khu-;HiE_wY z204M+%KV4V3ZfIvXnQ=g7#m)j)YcVfVkkVVf%h_Swfk0Rp;t?U;)ZlFuyNPNp))~X z-=$cD9HDo!7Ae4w!N*c46s~ymWcMjII>&cPf5v_HwyBGV zZ@I6<7)ZK;mC9UDCX<}IBN&LX8%MWT%i4=dj^De>ZjQh%N*EM1>KAGz27}U3I-RG~P!(?2j}r%5OG)Rb^awW}Ddc5uEN2Vkf7hS+tRZW~ zN+GgSQn0fljRDWAZQs%AcG?kSMud2wWj+Wql=MV7k*=QyAFuoYoydWqwuagw^Agc= z;n)zuP+Hr#FZ}2JUMl307dVeHuADG7AP@Q2+}ak{TipzReeiN~ykTTiJwIJBqz8cV zTmVLZY9cRw!Kb=T<&ae4HbT4j zaXsFr;^mTN1U+q%(R`-%_DgtZ>uDOMw+2}T->QSBFgCe2$x_>e=)n~|f?YlkM)o!~ z1lL%=uQ{%x3+BHc954uHT!}=?lOHaS@1)r{Dy9qyO?pmPXj(6TG|GVFUYiD6KPv| z?rp8v1m3eucx=elVus7I=Nj(ZcAgfYO&6Fi8#P0X|0Crh1`@d?(Xf&Vgi$5b21`!i zlsN-E?$toa?p+PqA;JU){|*HQf&qR(fJ6S3qXO2{6&ZVqq#dzrt6(M!oiW=aM@<<3 z*>Y@j!8i_D%0aucIYmngG%`DX!Na7eM??6kmNTMn*8-()ff0IV6rrWN8RR+k35ha1 zP*84bFx!i}hYMY9d)AC4LT0!M{P9k!lMbvTzPH_X9Ib$Mx2N^c7v8PL=-H_45Y;D& zTs%;tg)Hv4Z5ecj7>UQ(Vtz}!I!CE#HGOHGOno_N58<#LSYOy#TjCqG7_E}jz1`bS zXKEs-+m{b^jOkch!U<1eg(!}|o&AJ{zT?GF997N_igO=gp1ktRAqakih`L z0tev-VDUQEzHguG^PrxVBB2}2zm?tXgTxII$SgkN%}d|w%$ypTu%TCqI=h?0AF!9U zuDi8DEqX+gC*U!+K)2UduEd-+01X~X9BD$xlP`O!QR@grX|n0!y{6tHok)qLyh+u2 z;Egv5vWqN#DNgsjbdz>_JiQ1iD0}2d@hca+Avm~6pxbVpHjsT=0j913m})nx%t&|@@=Gi;HYXgA2=FJ zUB562zvIJZqa&@I*iRf(or>Q+YJuuAmd;fNpB=AsErDSFttplYncn#zmbIHyC36)TgZ;%hxKRPGV zD#zl#_tp9V3mob4o-m=HLBRO$?|;~kuAB#Ssz7k+`+j_t%o6j11NEt7@F09S)xu0S zc)ICh=1SthULA`G|Deo-T`3NJ7Z@QmgSH5@dIVE}t*Tiv6Ow-AQ-?u58NmtPa+frJ z_ol4gBlbHHljwqn=L~eR*xvaD+Dw!mzz|9CiOjyiFU)tRJ2Pj! z#9|V&)D3Xp>3L*CM?#FuG8Z(Z3!$at)sKuL0+l0UMQG=E#9gyN6pe58o?xoBYY+Qd zO3HK6GDp*kq>K zX8!G`*1MGG`w95M`p41<>tPjWu&2hpm03LRj(NZ1p7ngz2vc8#DDj9HdWa>bF7^dU zher41x`5cbcjd{pX8|8vG~#Vb;sVlkn%ku#rDIP-$rf3}^yMF-O)PPE%HUPcLAPbY zr1#alStc331B2`y|5ydt4F4RBz`!WgdLcfeGY(c28Edr#`m7$%zYJUKS@U}SMXB3| zU7kj54@2Jm4ybX})v)=nvNV~#^4mwAXF83Dq;Yc|{`KFN9);JTf4cu{zv7j4Y7U|d z{bVKXQBC@TYuq~bv*$z4I;I@nFhk)Q5`g6qWwov{*e8wMR+y8*r zA$m9jwd>_c$povd5(OV)8A;|%s3D-pVR8V?x+B{#EPcNG7NfKG@mY7G_|S)$4kJ1Z znJ7$Nnzq{3wqx?>6=ktg#yTm^<)@aS1cfNtTj22G5&54a(HgfN3M5VL9cToR+55=8 zUf@m^XgXa0$-WExV2p-``#u4pg5E*ctR$o5p~e($o8=4Np^Kze9c;-r=-3VIv|82c z!N>fZWpxG@#0GPAu4sgknp4d=+>NH&D*QUC^COUgGQK&(a)$Zv!thw)v?#^{H&n?v9hkF;E17B{}3u zg0&^zWSrlf#j&^e|nK`*4GG5Ut#r_sNOMtZKtl6at}-f ztkSC%F^uq_T|SMnVF7NA)8so>?)xTXBZhwJng=mn#@|0Mer$Ny?42V-7A548$3wD_ zwv;|(Vhg=h$8O<_V*6h+KO=02rYT> zD!w-G7jl*A$`btS9CS0f+PD*!sS06stWpFv0OTh3lg<;HI# zU=j<7kc#}g6seLBsfd&~sSwcp&(6lm&2_U2AY$U*ilW}!5m2MbWb0shivf!KxYtV^ z5Ro7E`-=gxfp`GLK28?4D-}*QuAhg1M9%;98~j_K$*HoEtBngGny!=t zP}WuEBZ9FIN=uGbX5$); zBean0@88dz%0;3$ra(zyRnf6e34m5eV!T*=NXM=*Hz3Gmilj`Y;xdJRG?-B4GZV)t zeaQco6O&kJx#ZzJ=&k88hZOFxA)fmo2>dSJoGyW;AGjz0yaoRO6GQ^>UP3t%C=Q@C zge8MHgE5f)$BMa}zW#?5!+djmiD9my?7tlCQruC6?5F8WT20k3S!WpMkMFZW|3ov< zVc_)wCBXwvgu%ftr^SQ8f~YUAj}8SiHTx3YetE&0gCDE)`b3O>;Y78|S0sf}0Lg)5 zVGysBy?uk25Fo)p{J^{YAu|dABu;MLsGZ}V->9AA?{C)Zv9;B0BzD7+k%SUD@B`b) zJ=4#k4)EVYR_tPaCDVmUnd;9Cr?z??6x}Km1?~{LM*KwgOGuA)z92kbiIkYw3{yDB ztP;NQtdn`L5piC?#Y6nbJ&d~%&m(!E5p0L!Xo#$l+L?mMMgz162h$p%%l1`@?GRZfMnpHQplV$>2}0 z;zDnCg?!7@=KZtE2F6`T*1&jEZ6lUh;fZ8j*1S$&6Hn0(#B8t~vf zMuz)bEr!V8S1sAIsmWXBUoRBHhLvOM{fTa{=Vh>EcWub~p33i7Zxyh}wxnkXGGmThpOkyXSG#rLVh^F{IM3T-x$ zAtMwq8|UK{KeUG%GQW)KcC?X2LN{#3 zBc6emN6x^*KEQ%eslikhzJo({64Md3p7)~h{P_sSPO5PS3 zoYVQLK8}FP=wx(40D{#BgzV#XtnCY)Gyi=ZtW2nvh@ZagM1HhEtc0u5H-WdPfVpS< zT)lmdSlYDD80FU)=rn3|8D}2Njjy?x6aJ|0zt4lv zzdn7(CV4bqZG^?(t|s^40i0BKyb-=3iW}ci)w2P$(iCAaHye~7JT2u4-Sh4v#aPl(X>Hg!0TtGeyVFGe4?zIwBW9$iX-%QyScLG*cbyiU7EM4~H> z2<9Wzs;sK}H7+xP4kg)}$sGI@aAcWlT_Rj=tgSU`DGV~rBD9GSpbsTv3rj?&6R>Q~ z%3$Pl4YH(6c4kv6O6UGsA5!<=%sh=buvMKC)oOnG89+# zPRg@#gmhBN+54E07|#}k@Fm?HGhEDjYqh-n`iQ@&5zx>VSe5Enw?hwm)NKOuy|}(X zB%758Yj^G(4bJSzI}LN1r>uHF&D+43*UllpOR5KLdSL0f8x@(0>_+#>Y~jvY?J6-o zOZ=#_@4){~xv#}^D+g6Z?E8lX5CyDEBwAbG^T|Yw>O@t-oWUmS5T!;2Xw>DdG`tF2 z_EDMC#_eO0^1Ja+t_6@yJH9JZx}IUunNg1^^Q-tD^f5eY>FwL)Uet=&b30B6Xnf45 z4^Hak^?qLtR$VuYQABZ594YSOk%@$x%c!y@N$Fj#*J2tEGejl(CSE3XV+X*08IzJ~ zH?CjBx3|eC`0TeX#&0zSF^*vB53SLkgLdVy@gD<&-HjcM4SxxnTm?9PHb*g{|Mc9D zy@1#51OlAJKTzs#D!NzwDSo$LDo}3nqs<%viCmdc$;}bagmb)9Z8dfPiq8Ob)}Q?- zS4Aj4=3OEJGv9xne3hX9qMVn{xrvuTA`vPx0%PHgm5dEZueSt60EUoW$3%bhoB$r4 z1;`}xa9qVc*)9)nYQ}v3C*0)UB{f={V)qQlZSR&XxI33TX_>q`T3>hmW>0uw(>3^! zPI8hmkt|Z^mk;mk4i#%AVxgAA1MP*R>G$XuR3OJBI~RDkxA(Pp1w@-;`Sgn)zCmu? zBc)bD5%q?`56T^Xj>Q=@B;}FmbcS!~SaI^U5T0RQX|xkdv_3HROBg-RR5qLPZ5uKJ7A>mPmPFX&kVdF*^&-_HW02v=znk}R(6+uMQL zc;c!M7Kk}N;@gcIZJBbcRBJ7ZDt$qn@D04;r6O3p>@oq_V*;=C9UXHz%5tqSqN!oG z{(}VCg}|UWD0S4&>LW%^A0(DjLJ{|i7beX4RT9{xr+g{5XPNbY&Sm(}Fd*?|VE@Zn zX@jIdGfv$r>%$HB@~wlKHR!ZF;!q`pu~^^1$FW};rt&DEAG>LI3G(|rph5@Pd=wDs zl>p-yJ^FC0%Y0ae146}WlJPw_a`C` z>qBxq#N z*p_@3C#~Bb@Fid~xa9)=k7SL=u68p(t{L~IHpP`203@BS>|BEVs%-zSL}vaQ_waY% zu5C5S(QK}1SzDSHL=G}%+v>=tWa=m5+Yg;k@AycY&!<-Gob;y5gCGF83q*22ki(6$ z`KTwsny9$aZGQlLcv*K0npY83k)$ULJJ+ijsy6zCV3b>@ zYj15uSB(WOHzZp4#0Xao6ZDhdBR_t$XuDz+KOxrfX(&fJn`m%PGMSO)G6F~^ZC8`Y zdn}?!Y=UiY;~G*q9jk>Irn5QJ&6#pMHRx)0j_+!TM;aZ!G7ols;moJ5E$kc=l3?yG zxwnvoL12>fcwp`U@fwO@+3Lt{Ow7GBz8SD(G0KqIBvCwiulBg=VSVdOw#n}K2T)*x zPYB~XBag`{E79wky!S_+*)a&|6;&AGipJwR8+d*3d3m;Ol*jfy1omS~-Kt0qZhf8$ z_(#+@Br>8(Wm?J*s}&SQ1&PP4R3)`SPIK-hy5R!-%^!r&F*H}uy9wQ%TP=iM!h)vnsq0s1lEs&6!^iF&<48RuZic{L3Ef zw#v@d_>$kSy|$6eJd>%wTH}i{LR7gw7}yndYto;5QfSqBrz@JEGcyY;e96cR_0c<* zGP|!ANz#c8{jOR_NV?0bShSB2Ii<8k4HrsNp5vMjP$2WUeX_P3Q$88yeI+7TMkIfv zcpi1X=|!TsW$&x}0<#?Inbfb;WuT`GUrc51@kgrjkSmF9_Gv>7Y|FbpF(52oxB zp8>6>yPjK}UtVx*&5`1%5jvK|op|6Hb|^bciZ}H>fhhu*PNZ zD&>Xj`2=L`C<2kBXMBeTS>pnb;ouVp<=S^FNgsoT7kJpwo|0jM!>y~Z&e=;%(%g}k zU2TOq?yhcXAhDO7UYB;CBAYqtz zaa%g9nczY~8hPH-#H&(MXz2d(;r&A>!%w>%%3ul^p>+LkT?a=uf}n7#-j+)RmmkC= z)>f_RqYgg>Ta(9MCWueP%WO!gNUi`Q-W@`f6SnF{XxCrg#faS@{3{M3Mm zbvT1wc{@q<)$}RaJG{CuX@W@bTIzVE_1X|?&1SO^iuEvstVdt>@_aJJh^v}WQI(u% zi#9M_mgk=Nl9j@uNuZYo@xi`#YJPWM;&sioe{W|u<+zC~O8k-@g&GGUiyr{@-SoqK z<4(O#qk6L5Bu%FFX3+A!^zGGkSw-SI@Vo#1r)}*wa19gZrr(|P8m;}9P|k7>#C@57 zLd0bGIT88h%)jj}2C5k@p9_h}d-IeVc=s$=B(M z@`S4*Pa0I}D(W6oeT(#hRo&VZnKm)N)_}ntv>!bB@VpJ%CG+KB7Uh09VT>v6 z${`HYZdAiE*z)YX8ociijfcED3t^jFVOxjeCHr6Y_{wLUqS^3>2nAn1zKcf|JBjtJ zgif(HauiBxIowbpnAhC@^*4zYdM~-|_%f8#k@q6)rjAQka9%3{>5lbm2=BB7*n^&j z?kKHOVqZIC-%5~g`JWR%e*~4*H}OGmBq)85Zl-pICP*Em3R1y=2in|CjhRi&O+Wxr z0l~sR)1yE_;6q_Wu6)1`ED4XW?0n~AF(Qe};DN^kqQs&25nTjoI&7J4EcQz8+S&2y zvCCKbg<-uv80UA)ZZ++J;15K2P9e(K06D!|?ENO*7<*S5q-*eQ0{;aw-^NTLV~fj| zTsLy=t*-nGSBz2v6frxa?zIFH#uxBjy5(boG{*bjMR52Mb6>xD#%REqlOj*@Tifs= z+f1y&L<{0JoV?nepB5@wcuv~?#t56+w?A$tD}WXe(>7unX_gXVIwuTIbXdq8+FV*q zPH(`=&jJymi1H3k&9wJgfH7TsXiMOnjxUZsBii8HMS&lkH4O7bWRa<{}QkD zMVl3lEl0T)SHMhbUs(Dnx3a(P`*nu>67;cxZ9|K{WUoKVbF}aN1hW56u-EGrXzpwF z3KTzI4IcTQ#$EyaXK?b{Yx26N2lcY32jQxyr$~2-KzH3SdSy!MFl$v}<-klu51I%f zrz%IQyvM{F8gPRiVAy#%tGRz4G3!z=7oS&QXrhZqyX%yw6htZ!(c{%Vwxc>vwugKA9yr*g*?z;1nAGLy zskCMHY?YK-)ZiJPNOet159Ei`!3z`38twg;vES|Hs5W&{PB6EO406k7qx<^9^$f55 ze%qaI*eeTUmOdiM<8=ql9_z2NX{Y1`AAKLad()J@Q5=oOU0&pgEqhR#9iDp_@kO*S zf1ND8lpo9hCwLE{@m!x`l`R?MZpmrjQ3^#7pk}-{O&_{Osl;WVq}x0NZa|xrM{U1) zkKF5(Em6E3sWdG6{Kj`|pOwaO_X$lWPiS{q2`|9{7EG5ng7>QxB;1+Mmn6{VQcZ@! zMwLxY{Z%$RJ7_})KG{~iU4#zkXXbZ(YGBi^`^weuv0A$%E>a@TX3ijfjhjX|cFF`6 z@)vhl)n0OcXhz;-N-93wszMRZzP@6kNR;fh6oORH^wba@QxWgyGd3URC(VZi`QTKK znsm}X#QapbCka`qUQFJzt%JcnUEpj zQt4@Mdpr_e$Uo-{P>3dFaF!L}>mmxE7%Ca_0%6pe6nr!QCk z6^@|nS*U*0iZkgebeJc_3^sX3=t2>vjHPt?>N%!VGDGXlikfz$P-^ep`*X~j{qGo+ zLWeaEsl5$ZvL^{wJ>`Ds<&6vF9rO z1E7%gnThOeIv)XVH)B)se1-Ei241xK-+$Y^V{-ci~Uofyeu}X{uOt_4`~Kmqfd>~P*#=hJ|@s1 z!cfxeWmxlul%;Gk#00MO(qwZ$nMOk zIy;2T++FI8O~Pg1$Um~C`}PjxG{-6ZbeSH!F0pQ;O%o#Dh6W}vlAdi%dkqCS6D;EE z!pE;AnuuPF=UNUj4+~#N&E|)^Q)-+7)1Ybdn?Gs9B$5yGyNCcq(u%(%=Hr;+Ho(D^ zE?uovu$n73*aROUzWdER5(6K@tp3a-5erL9t8A^z{EjcZ23Nlh-C@sSrj~MYo-)|G zz0a?DUx4uMbK*>A+iYBvR(4G7LP%hSwtuc-DU~DqD3?->ij;cniac@1@yKVY&iYPd zDzUH8$8X(Yjir_i2<{K%9t7+>`21uR{c7(=c6HFVw!2PP-E8|zm_K&Ajq|(Ld`jgGRyk~WRuLs!fPx5hvow@S|GORg{BdC@bCF5-# zqC&ximO$RYGdk3uBoAk2sQh9^gkk+5lo)HOD>849;_0Y6a*M^nr>(6!O~M4GMU>lO z{$%H``aMSt_)qt{89!W%82<<{q=S4@K$t(qbNO*$K|sL}LxG8a3EduV75L{M#IGno zk6acXj9~zL%3FhFsFHB4noqR^S5kdIg%&diSXvph8PtK4H-$?;h1Q>zDYKe{h|pDS z7TV7{!3GCvt^Oo5eC1i9v3VfD2~5wV3BMxfX=YBzCK!CZt3*BSXY??NsZ)x8K56E> zz}1xFGtnM;OP%Br+85+`f#LqzB9$D;Mu_P-btY`h`>8ljB*wWmbY-0}IGaeuGku}N zNwXQO3)8^81Zdxa#m-;!$2f+B7^SVzFB3*H3^%-e^469m@xyBCFmco#JeHf*|03@# zz^Yo?c2T-Px{;7ZVA9>)C=Jq$ba#h@go3n$NQ=@TErN8TAdMg;B^`1GAfao0%l+;1 z?f?JxIcr^9YrV`d$2;cay`O&W=T3p#z!)j9T%ud*OfOX$YBJ#?16b((5U>F!5Ey zEAta~dEVnK*2I<})ESr)_&`*iwHDd!RVq*6W=Bu|F!=T|NQ&BX$$|0E*KilPJ zf6p~rSW$IGYp6aP0%AIL= zk9cR!U?@)lmcNVSoXPF?kmEnp{8So##h-Y*1edK6kTq?gC}fN#EOqm&L7IYrj37Yz zZlC6bC8h|AloD|&YzVz(8qi|Sxd+_I3RSpt?7_@oP9!DVqURGcD~C@ul!c@aRf(09LRu7hMSYU^YcS_1!9HVb zaI!i-!^TF|#*QvF_75)=vFg$lRv>{+7D%9z1>$EX3nY|iwY4B}wn05SwX&m1!_{qtf8`tN?eKWW^#Bd-_sHO(9Cm?1k4!R2SH1z~HfsI-0d zwCGDU+wSQWWHD#!hg4YK3LXGK#P5GJSIfZV(zY;iYX#RRZ;2B0XksSa!AJPSSiUI1jJf;Fgm;i+m1t?dH{LH1MAnM(sRj!t-Ww9h@Tl%WVd zig3MBp?*J$|8~3aOA)6OOP6y{8W%vO`+N&M;@-aJ7kZzaLfsYYv|{@HnywgO7g>t- zXil5Y8(l|{x8pVVhxOzapsBw}xnaoENPcEKupi{=_(A-M)qbZ;u$Kw^-RdM7J&rbU zm14gRUXJ-BEDLpqV%$NFa;I{zqFCO6)IUVpuC*zgb9X1~Df#aE?r(3SYi9#F#GUs3 zQj+{PG4|}-S7bfleAzEf4f-Tg!*%T6xQfn=6TXH&xg&X4N83`Z0#%Os;xW}EqQ7C^ z+9VQ(n~)Y+$(tmh2idmMLwzotcLu?xEmaw3%#o2t^BYM06h~oR=AOoJ^yUAUE!|0^ z7YhuV7l!}n2*J1)`YCa#mTE}iaSgIKp-V4j7GZT3Q%ef9BP8o73m)amC1l4+{`gks zMXF~nOeSWEwuH1g5>x{mY`+BK4=@SUFxJy!Xgb@-FUtF&2xvi)rK8uIsG~`DLbjCA z($YV}ndVZil^k@YN=>4N$!=fe8YuXjmV`(J!A4Uo829yHF-7GOp7=;5>yCF1SJEiw z=Dj_I3)1Z0-S(^4I(5S}(qYbAb^z%QAyBVc|0%V3CN;P?f&TaFyy7`C_jQTc{|7$f zKaqcT>GY#%KQQkr7_7FCO(fPx07YU4k%(JK2n+?01kPBM?q_{N+QhlVbC7dD@v*5l z&~VGsus2g;t5u67(nt`>tjV|9h)~R>?UVIR91XAHBI`$2;@lc5k(jtBF*P0U_eULB zP%F#v9LgY@x8M@YEZyU3$wub-`^iLvp$ZR7|JFYS$iMH&b=N=WOu$RQDgozmWstjN zYHOR=imDl#r6i|pMfVz6TQJ;c@Z;K~s{Rw*>=q(aefGh-RJ@Bc@YQHWj$P#rX@rt_ zwff*|vpCsEbAoU@C|_4Z9cYs_&7Z8>VZP%kw9AQsxA|fRQ(B{;M(GB&yN{g70-_$u zS5Hq(HIF$tSC1$cp6PiJ{8fyLcm5hIh5%S}0kBA1n2?B2XXYa~sk#ze=1ui@R#5rp z>UWlZulgO}@L~ZBxXT#R*(8vw<$rj$KdG)R3g>7y7*{gvoyLHQzz*u?C(zDl?_WK# zKNg1&X)6g!h0-POWRRKB^B14FUx4H%VQV>I)W5QG_r)!LJ|w!L_>UEs!KiL;c`$rk zBH(7E1}qF1iFFR2nN5XK#c`&=eB^KMq|`umn0)hLi4r9Mt94x~d;m(XLn@nk*PLg! z^8?rEG3&hlmLw6cX;nvYd>i z5`}06X1yh~&T6P9W@hu8lzL7jX)dQRnCnF(Qpc|Zu_XH;fB`i+Rq^YuTq=!{+#7V8 zFxHa|j*u;?d!{TV_4XMUyij2&MsP4SEKA431;^?dF#4h_gu$K7&0Ap& z4yMupZK58o)32>9Z6IkPS-8aBMSiJPv&=t6f3~XBO89c7Ll?iB*z-BaEI6_i^s-s~&y)THWX80bCq!ruwq!?=^$v&Z}} z7Cblc>s=ONZXSw(V8I@qyTPvQ`iz)0YoAqYrUKWX;pOV6KiT&T8YmjyUU?E9z%*J~ z-XWnbpnHKtAZ%k}&iy3NQy#rW;%~GnzcMzQ;ZOU9u98-~tUx|vY!DMsko#Hi za#`6dqQG!hR$26>g?belI%}o_rU6}aexLhZB0?9*UM>(DfI@)rnuY7!`n<9HKfV2* z5ZZ^ub-(z2Y0Y~L{}%20Mcg9BzA|lF&uTxj$B$$iMda#i%pnl&O~v!c6zyuSl9me< zbXR*jr_~?uv$Ll2H0?5>i8(&yQ}ca)5SrLAUtlp*J|+~6+B7mE9NOHl&50f-s|>xz z{zWXn1Jih997(Q!(i^FOrviBF_aSLFS%xeKX~lr~=tlfQWgsn|r%i#;LAN`>!q zbdAt!TM{8C&&3!z!Bfz0|5p56&P{|>-rm31l7DgKZWx!BX_W_M6R&uVK2ym%T@5=(lsIM6gvf{L4ATm+qlu}@Dg(k~ zNjqwHz+g#n1f;q8p`{RWDmeUJYDeU===f24=6F%9fE7LF_#1@&47Z?Mhg%?Z09~mG z=t|)WUHL0Q&yqljzakVDgnec-B4FIQfzUIm!W(YmKZ5c+>H+jX0-dfPR%Q_EIVf)~ z{}1f=CmfcR{_q*JDqhF-RC|eJEvf~l4Z!)ak>@8 zKQS*2^Q5431vE|&Z;4Vw$ ziKi5Kkx?t9VJ~|J6auckD_>7&rfTxKTS)susT{3}cmeU)?O8#-&)r_tPkd82CXk*P zrH+&B8HG>jPh!R1$4d9Q-!79&+TL3Ht@B-+UjMC)0^9wmPC4(@k;8(~Iv?e&mrHJk zqa)7b)D9Oo6aewvsFO-}kV-Tg57IK8OQvR2ORHn4g_XP+UWNHSRX?>zS-9KpYq_ zlWf82TsnZg^+ZuBY=S9Xw|$8tpGU(uK9;$?G>xCzK>fWGP`73Gk~yETHPl%|5Cg-M zFH@NUwR4kuYxMB5l>4 zHL9P-a{zsL)}(HJJ+h92?^<6f0%(*4(8zj$MmQH27%IxVV$a_!5Rm8%Py#L<=6S<0 z6ao^W(D_d1aTunbUKS)0_Kn|OcAWzca2|d<>)YW15+dxN^YG(Y7Z20V<$q|GKY?$T zK*1E2E(2=^hOCnqykyQn3`SJy?prAe%#ogVl?8VdT7{Tn)eg~fv`}WFh;Iku<>Z&s zSL-5&HNOsGb7Q7|OK8BcRjeh=*U@dKyChs6yZB-9fK1Y;b-tjL&gvEP#^EqxZ_@mR zVpceh#LGKc?(X|60#)mY@pB|iBh;b2u}9Tn-{u5y(1sj05SeD_C@qp@)rty9_-n-6 zp}*H9n`0Gu$&(niusL(XLbP+7v`MTjZaNb=U~R=`m49$|8x_u{^Lg;2hf?8%i67f* z(HiC3gdk?))CJzBI*Rh{bKFDLUHC@_0~|%aTeF-K@QVze^>kUbqc~(%p!Bif}SigRM(b=V6wp zE;Ud_#3AQH80>AucC!Esem*dvv@LMupkuc(?jy1P3)piSr3MM z?nQpX1uuSTSXaBR@0;|zxJTG_HqQ%&!iy^7hs|o~ZSt`>85Mjm9=0-`(U01o6F-d5 zt!5ayZ>BAg?II-)9%6+;)EXSiZ@+>llHh(tZ%wtiN+_WNKd5jz( zpvRFq*kR?(>W!Rc?;K8$W`A;@O-Kyl86#myl7kd!v6O_EMA^CpD_Xl;QkO>M2*h(u zdE@ffB7!NJH`tpEV9#N>ag55L%I}aSBE<2;d#+#L@3JuBYFy-n!FrK=fZe~s-x);& zR5gIXj0^ms3gWjIk(OeRLZZC=f{y>5=yx3Y)%^;@e;I4}Q^e!9a0)<4_HTq!eol>n z#9+KEgCe;ePDF$c#Z152)`BYsq z;?A8By~vo4(FpJ*Io8+EozpaAW@*Lq@m)n~9CzR$=v8Ro9zb4_(d4O@a z3$eJ)u*`l(#rFxhTfe)rCpKQ;Y;!OLiYSmud}LfIW(7)NK--| zZeE6Ec82YbhsFC1iRV7K^#1-E$c4M69$#Wz47;pl54=#2$3IlFRof~U9+Dqg21H1t z@)&+kI&fx=iz#-gEljE}{=!7s>KCV~iBjA=RGh6*xM|o~B}=3cExjI%O--y>gnHzs z$cbdxwaNEx$8fH6jT$bRf#s!OtVbyDmZhlUPVX^!t@yP%B%*#_2niM*%YH&`)6OwyF}I zR<3sY>u+){D8whwYCgWO_-{nVu0qMT)~{q7WIZ6?Y5*G}^Oxw@tzgOp(whLc{aJYI%ozvLKv&EE@NR#Ces(W8NU}l$-P#fK>J<&I**=(HT^%WlaxR@m ztgE%INN@>(mbu+BW$9i>ZgLw{^qr&~!0>;4kT(4k;a+^0X3y(_;k|tLe7(By zQQax~JUt8@Fz|NiLxHSiSV&TcNdFGV=H#fK&`$yRCH~pHzIM-(XJOS$32!%Az)WhC zCDBif;{`MPbK%7Gc|2#dKU2N*_RQyn6F1qw{`STVWg>wP{g$qDd~9mF0Zvdft(D@P8u{3WmDBs0GNSTMIPMK>O2-1+@$9HkVo>L zC+t}9o0P}I1*QjS3#^pMcDN0?No;CpLO)K#tPMWMF&+L@X+5}{uj<|-J-oT?1( zYxKn<6ljBOdp64z!u^v@K65f;bZ!kmIe=zGuoW}aS5`a-ibc62bdHnexMbu@Rd zH+B8#*>mz<;T+lq=qr8*U|s0~=V)O5sJ(9n1m`q>ZpF8T{^4LZ(S8<-Q~1@Ep*bf4 z0z7>ZsWWx}z{S5V#JUeK8tB^_nvqD@+c?=?PQ8G{K$23rFL6=%png%_zy^30tU%cn zE6dsNFP59j|NMr3Y8igDNbazW$?-k;Ih`6cVMzUW2cRo0wQe zdGp#fyrLn4`TnJG!MA8uCaN;gI~Fo+P$J$lQSy$`B4?2Tf=vKJzm!^puVcj+V;|>Z z4@g^50;eqy?_pHY#F$~h3bsOXB$|wOFXG~9xeLQ~iHG#GEII{}?PExq^cf}cRgBR! zyav=#=?H<)7y9=m3qxRIw+-!98D?#?68vUDAF#SvIWxtF`wHV*-t5XBA`_jQ)z88W-m^CJ}dVxr_)$O9?pPks~7WwawgONHRhaR|3Q zO%2zf6SFZ=<5&(PH08n@xmZhZ)btc;=$NJY*pGz1hY_OT*WjO^+#zpA%h7xxD@ZJ! zkyxI?ea||W=ed`C4mz}}#y9?Zu2_hWQIGo=p-cEZqzPplpwW+8m!}mmJVl;5mUcLP zjU)z#>jMpG*|C?w)--`Y4?5;RwA0limTxroa+ zq^d!iiYOW@J<}RXHSJjI!KqP!zPn$RVxq@!k_`yNHca9MJZ+kndDe-0J4cGBHuK(m zC_3@-PC6umS~(r{S&pG2E5<>I+6rUUQvY-e-mY9-McEbhh$U2(K9JXcYVDaCbj{tj zdo7i1@;v_aOdrS`9G>(y`(|-*p|afZm*q_hE>TG!s;3E#AWMEe`;bSC+uv4h9KJ2I z69HDva#1@Hpw2I7ds|K?FE%3GN7*%R-vMXwGHwnlV=wDl`(53aN2i6<5DfJnJ>JgQ zc|>Cm!s>`Cd2+xCM+~++OBqfqr=5<&_mwP?<$~?$lE}IrM^{sI^lE@b$!nfS&T2TL zxJ}q*L6C~=!3*k2*vDTVvqXQrlXj$-;$|s6T!vNp%-TB{=^1y`eR_crt(haZ=D;81 zxi3P;3N2=EX^V43^tA?w$9Wz*Z_zJ4cACOrvE)-a^5_)OJsr>N)90LdR#m7S&hgP{ zn58ihtteHf$=)fj9zsZ1*OXi`wm+Gw9#){bO2NV_$1|-OEi7J~;bhUS5!3cdThf^!rvIyGzML|I zfpo>-a8?nE2g12X=%b$1)m}cyc}Xq6`gT#ChI91}2}>J*9RFvT+C`xb(8~L>JpP=y z3s_w(|0^EhPbS+e&kUOVWNy~r3$=X3%qT>c==RK)Yv?`Cw7}s>y1<|2$b%}C_C#9D z!PHV`ACpalzzBl*!T0nK1T;)_^Vo^cj+nGTmpW5nM|nKI9p|-HDft3!Y~-~&5*LiI zkF+*%EzW*H{TlU0U_gXxqDGLJ^@BUiG7a*Li<(w*whb<*4aeZSL8G~%sOrplPPoJU zne`I0R2JJwEc;S$df`L5W!!-c8Trz(Ui`fZx|Ys5=`tKK&&SCYta>o6do$56k;Aj8d~4?8-Dv(U7(y`QHna#c8m#XVy^^yOBO*MHusw} zpH8{Yi|BW+sV+>mGuqW(#n(SpxzZk!p@)OF0Xq1c+PDZJ* z88Lp^YIdbwZbg_qhQ$^&DNB7)++nC!uDo}-eIMqmyvD*@`V`t!&1pFsV;Od8RP92c z25KZAe=#$@G$r9#UPcZF4O*qILu?RhP^`}TzLEfr{_4Y{)&y4F#|XTwOT%$-R+H!W zM!L3UFAMV6ZnVI<0l0PoaIO7|Z-phJ`7ZXR$~c3x+6_nxi(O(D0R!~{m**sW*3R?} z`b?q+Bem4Xi#+`x3uC}I(=E3UOL>>V%N--_{?wi;#wMe2w+YzZw5Tg;V& za;tK2+Q9HWGS{M&d!M6RWf|I)e8Af=e-gTb&ilmm*~qw~^;1#({>N?NB~B>8i+Ewf zR+huA*m0Rl9}qZKw#YFQ9F(781aTA8-exTihmNRr2`~OkOcGN7uVfwFLr};dM?!6B5 z{tDl}$)sFcA9@}p_+rTM9ifVhUQpevxu?e`GQS%3vp`F26$A|br9PO~?eTiUe+7?x& zxIWX+Ai4v~Y;EyWq3^}9c`&m3gKtu(3%1DceuI+IHr-O&NyVPO zZqR@b?q|PJD8l!|S-v({B<+((j}VAPOjKr%CEIBz&8N@NiQ#Cb@8W3ubMod*<;xFs zrF@~tfgmpl093f2%9mAsSdC=(+p5$yV-Lq?I;-ElUXWkj4X6UymTT>MvE5(F%zurx zUXiegnme020A=7nRU7+-o@Trv?znj;pg3II#-8Mi%-+TxDAF||0XTF?fRSx(S1BC~ z?29*$T+a8r0sgajo=hBP&_4(NZ-gBG7k!pL$&o+ zW}B8;l4epG9LI<(c+D<8U8wYNtb_|Ia_~8X16A$u>=2cqL*%+Jja{O#v5z4PY*sWU z&>mAKJqV0O1v2Z7CX5!G>9fPiqANtOX|OdmYfC5ch%`I=f$89zDFyq4-l18G8K3No zkzd^NSgRc%(?hP-N}Z}XzNdxwI-z8VVF_x{AY_v6!$u~(h41ButbEZ z(yhacFnO;?zKUX2H_V?m9{4N6{g@Y@loF~so}_d6%y*|S_ed4wp64)a^n zm>+RTQQn4s>FXg#zp_Xb)c!;N#CyRro@hArTAuvHd-+|!T+0&=EwSF$J)Q0+Oi%DF zD!z-as7WgRcHqzDi5ps@OleHT@y3$r5VEycy~^U#fgVVKZ(|g1h}1f7b33QYt|~e& z8rdUdlLThvxE!9h1AsFtr78T9Q247V%AabdC5Gg6;nZl!nR`|++ zBa+eeLA05~sGi=!F)SY$2~Nu%_EO6y`-2Y-o>SP2Sb6*>7E~1(E1ES4#KUuPMhK}G zAM|T>xCldPs_Hz*e0F~n3cAe~qT2995x6;rw{Vn_;s(yMgRK~nBvvo|V1aro^q~pj zK-7@rxnGa);E2Fo=Rw01jftr28I*Jep*@B;iQd{1mhZcW`rh6#bbb=eRG_sSRBeu^ z2V8Vtpf)@nt45fe%aiM&G-tBJ^eUVxgn1=RfRVBOkQ0E)e}5rOel4Il3#<4={VG0w zs&hCDSRnx2HrD!=Ayill1SLZoM@I)EeM{q;rBMG1!)qXK4zIZo23IK`<6rttBrdC> z01M$-NC0goEZ5bw|KpzcPokjiTNC8*ZqzYEv5i)m#DKSjYFNlJ%Bl+gWeE8*#WjXMCyCfUh-=6hI#y?f`l*$HHhgn>?$8@2&P`T})Guja!IbwF?)JCRTgQ<|KLKB@ z3N!U!nl)vJh=ZhL5J{D0+C?cxdRsZm;X_sf_5#J@{?Ws?JS~H6_vE{yw@qR3KJbdW zBZGUrSO03Y5Cp>BKJ_Inb<2H{@ zcq1`H9znl7FHh#3gbwvcFWR+e4F{tGc1InGj2#nb1|Kf0JlZ40@W>$Z?Q>CpdO6nn z!U4E6{zav)C4o9136udzz19!l@)uJ&qcP50pd*yC!^kHZhK4=z@VZqdKdcVua4O9|P>7)ap?HV=*kqwIGK&ij ztTGOQD+Uqe2PZ>65p>e79KX#iSfbZ=Z%65I%^tEVVQ>xUP||(o%f+l?UcrT=z7J(L zU%UdF6W< zS9NsB|KfT7?FE11d9%&YSj;nOD@W2O#-=6WV}6Npr8S7nq%wWp*Nd+svD8$O)jJ`C z4=tZ8vnFk$a=(;3g9Bm3$eEVm?)MWOr|5=5yH9O=90TRhaR$}dlc9X<9-JF_MKJQS4pg>s;pH8G}e+#)^Nrwed#FcH~xHQ)C zwN{SxcHRRy7TqWCjE_vTUEhi|&|IMxc1hr@! zX|~jk4diESqKDWV$C7n*KECv9JTo@Arg;sJgmMu!%o4*wW?2|eO>9U>fg;Bbh)zRG z7jbaAV|0%~=aw7fwLJ+a2c6u$Ed{bsNXX5pA)L0eTUS@}Q*?ju z&`}&E05aiSWUlp=GqW{SK(oX5aookSr^`;F525r(?9@V1zt^Wn-{A`*pBW?l{O$s3 zKzr?9p@s(h{fJPY5TM|s)AQ8>2X%OVEl~U$C~5YD9AMjHsmTzHCV$PO1g$$A0xEi99WV>}KH)`e^2<)3#U zJ&qk86-&`s6;?DX+SYCWS`Tf!BPS^e+_|dD#_`Aj$v~y#apk zemcmeA7&Qk>Iv(PV|XI7?}SMGuGRd)Zy-xk=+Z%k!$49q1~|#>t$_(Tw#H09s}x}| z5JjEr4fL&@%`E{t_Zo)E=gvD55s2WzJI1{D#jowBH(cPKFDsi{0lNS^5$4vWKe5Wi zan9^;Zr+}k8JsPz=*j;_J?%erfsN3qyXuPtD=o8n+)MfPW`O}0<;S#cF4bUJ`?RNq zJT{7hvR;$VZW_Z9cMv%i?`H`p!*|bHV(~SnWZl1yiV*Rly1P+)j{dQ6?<;6T2<#+{ zddjUn_HAkf=G>j2k9k7VvRQ6#=E~JFH{~+keDy0X+;5 z+Fzmf8?=DdFqx%%96#tIHw31%_d?sM$^}c!^YM|mKVsUE_$iIrGi0fZa_YdJs@PfM zC~%Dvj7zbha@{%YNdC}kA}}tCKja4^e4luRvKSkuX2I5;Npw;EvDTNtgTA>a3L~bk zwEMV^*zxbbM;2luFt4QC!Rczk(~lsMq8T^?<@ z(5s^jJp?4h`+)dQ)y2_<;;v8(|NS`MoU|k?cc~VD$m~5Q8`ZO$lgOG|nL8RAUHWK< z7=*vMT)#B?Gckbfo}Fk&42r8e{L}U4Re4N6-xwDYCkyb;#R+BscuOvpzl?(VZ+XB! z>8u9*@;Id$wd2DrQ$%k3DKuZBxU&A*Zuj0@nr95;%7Wyad-y@)>av|#RKoFnX#rvH zzQ90gfxWzU+ZM`kwrPmF;NvSESX}7HlYqiylTlm(v#=dC{DhjFZ_a$4qm!`~JTf>< z!%=m{oZNAJd@AhnQ*LTZw6{NL7)z;>BzPC-dZ#Uo$gKuA=CxUVTBDH;qJ_acP+k}u zGlmT}~2fbgh7s5h&5&5>DaJ$zsr16`QriS?ETt(QIUn{ zw->?v_dCdm-z3|2SyK6H10<--5{@YRJe<7TH59R$PdHsQL3t>Z; zp_ndzx(Ujfc<>sB|}StDwQIgK~233A+O27TNV38!mC!N;OqU+Zu;mxj^^4oqTN(E{E9gd zk~^$RnSzVS&jW+(UX8eA#@pwIP?l1Zj!^>+D_x+>g=G!0kp1`_l#}JICt27n_i>j= zdx|#~0{!M|@`9Px`{@_1=pP&mJ0OuG_5!NQ)u6&d510;6Cr1_T}7MEFd zB%GTa7K#_U0}L-BH(M-#sjg?NQvm7gd5gvQ@;|lBzo1z`2I&xBr9$>)xUp4o^FDF! zz#{B6_?|-f0G4lF`S8U^7y9xxrtV@SUI8P6ZIj32s%{iX$L$nN9xC~RpOf`8MAf}AYqLVk)eB2HG}ZXnmv!a)%a3$g&*#~(wA@eKRwW} zDKa`#pAWx1XINkE*KGq4;dXvv@Boo@F%{gw*xva}44qjG*HPXFq*rGK@;l&Mt^?Bi zuLtHo8|CEzaa~vt%u1Kfd`*V-n=SG0hkO6$KKh>wjbSA27TN>(*`0d4 zc-_cHO2xzioxN!!7*;ggA8vW^?mcwoG#q8u%T3mK0Oni5sNu64azG&x{MDS`ep2*| z-P|X7WO{WTA-BJQuXOb24S&3e5U8W428U91vQ)sxZ2RuH{+ zW3scqP6pvhd78+-;I_&uUt?vcL58einUYv3e)yPTG`YvGUre0&nQh1<1d5bMP=%$# zTcc%tp~Chd<*5+HHnI}Rt~3p0v$W%STE-R?LMK5zoFE7lu5cgd3qwN~B=~2TUBfg* z7i=mY{3D2t^HFA%m2}D$T1>!A8~8HrE5n{0=wd#>gO_G{2hIjXzx^a|T=x^ZU-u^h zA^J4Ei8g$*(SGaTnMZeeF(yTNXIV?3Qi*AX8}@ZR3^&!SJ+WbS+=}sMtn(Lke??nK z;w4w>u7ci-wOqGn5yxQZnAS^Jw)GzTMNpqT|0{`GTrX{yO zR4;wXwSfpXMV6yfvl}BhZjx4fTufbaE+h~E6U@fQMqmHRQ|7*=Wr1P`TnT`!8xsGg zSomdN*ICTTm z?Rvq~ZR?q3=MX|UG116+@ectosZF#$aET+oKmg21+KOS=Oo&M^eeZ3_x~1rq4ZA9= z&BPIvmOEw18*A<5mp8GBww0665#G6%u;|T>g{pifQ~y3W-n$+UoEZ_TkY=`(5G@`^ z)#ltV*Tr@>TnE<3*uPQoO>WGAews6hVs1>EN%@Ma0i^0i7(T`_(~H~bi6@-9yHw2T zHn2MntBx?~URUOKjTq`WM2tZv=5pqC;<&QkD_nZM?aKYQ_tvQJYywgNV;Tzh#+~I$=3A;M`nr7YqrB5FytL(RGu)irM@i?=2iJ=@p_8`u5_H@=@ z<{QL@7YE!>|9oTuaF%WyrOVQV^J0DAed)}>SF>EthSr`R%JZuJe|X>D7_4xr#}FII zi>FzKBOa}yzC8*p?{~q}yRs{4F{R9)oEI0?v{xZ+>5V*A#8se_2cKJb^Kz}B)=^og z;p7;-s!c(2J5&$aN3RlE`upx?b3;qDV{iq%*Tb%X@8T1TtrTY4DMS-z3)Gom|@3baX>+{h~4M`BGF{I_{5~;&# zGGxXy$E$QK+^EAEh@PCgd8in7hvwlZ3&`U2_M>$ihcc6t>bCOCrZcUZKUx;Qxb?hY z#q6u8DLfRJW)9F>am9QN&>u(CkF$3th^io*h z%~@m^ZD>ZPWavrNkq;Q*dhIMqO~ExbwZ*@`aecXqBJ4irL(tW?QAvG5yZpTLO*K^h z$!=R&aTRU)k@_ZHx{WP`H@~^G&n>b~>#yL@l4ufV@o*L z#Nd2Yi$$?H(o>Mx$TY&3HhqezaUA$`G{jRD81<6YPmi{b%gsGWF}gP>_$>n@fLJB?`I{v@C2v z)JSoUKBobx^xHa|k0q}-{xYt}lyldCNpQ+tvz;ZJeWp^}mJvy>JF;d?-fzJ%ZQhSs zPrii`w%Ff&;59#X7_x%QcHgw`dAu|9Q-QcP@C9=fP9IeL#P&Je><)f^L&^MEI7$Be zt&$*v?6REEI*t^xa(Y#-Hb`=4 z@}OMqyE!$#dQ}2 z9{CvoM5Caj8g=i1cLsw-6q!RM@w_hDhO{hCj=TMlla=E_cFI>Gy0!W{K#c8=#LK&p z(aLb6*7`H}wXqhdd#Bz-CtG&)BZ;|`a<7C%6D_?qvIzCa;}dZ_{9;O$aWpR1vSZNR zK~dK3tH+Wz;gJr}@QeaMr(KuEqYTF{+{zEHZ#3Wmbd-=ezPX5RTJG!VrHGhtD!NQc zz3y9ESuY?0B60nial%Bf4y?Wtp=N2Oi0a=Vb^mnKf784`KZ7|2IQS_P90&&Z4+J>W zUu+7s3ll;SBoC5Jmrj=q7XN8NoU1THVD06u(8Ly6n?%9#h z{YjpD?U$@q-ihzajq%Cu%_ePRP3v+In(6x`U;eij+%PXL9T$8i^F-n1dBLE}v?uQj zhoVTia`8xixF5fh6Z}rMC{w5)TD{N(f- z1Tn4r0;|HfPc0$I!>kWTOzAwW3Rv4|$yks-dPS`bup0?~U`w*b)9-w2x?z^nM-IL4 z4yQ(}$Wx@;*A-HZ!~yzP1T~e-T-Yg=!k;diCmMIMDev`n`nZO;oL=1c8I9S@*ZA1_ z$a@%eN>4dXcwd7NS%zc27Jn8|ao@9xUcP%~Nd{pp>4$s5SDG4vF~XYhhuu&pl!AFH z4!l1UbJ3_&>OM!xtSjJt!#b@CPB2+ZqOfDNRNO4N*J+X}cgB_s0fv5QUKqOqe;Vs+ zz7fo8el93(z`6i@7|7fI%DS+*Ap-uTbwLRtKNAdy7(`d*gV@haNp6}MWFV5W>j1vk zzh3uuRtD#J&md66eqnT+EkXa~5A-KNQn`oEvakgPMb9-Yul0?%*78<&`Qe?4p2=i1 zmiR0hS+A#Qb<(v@zd_>9MfJ{GDQb?y)5L1}zL9eO%n)}{)zQGgNq?I^M(CBTYTJf_ zz#{W=$Bqxt%1f=fw_H<`1bI;k_49Qwqn?LkSSXHvOp<>MU7Bqdn_^e0sE5#ok;(}h z43WE(Yfgz9T_F2L;Z6a!Lg1&Vp0pWro3|f@KRy)w()Mc2N=VmU(+-ZK)Xg|cBR!U5 z4~!AH`HB5lAE*6i%v`Y+ZCbC&vXPdx!fp)F%67U55ip8F{c3Xy^*d;lIV@4CP8vJi zG#|w)5$KF728C^~N~%1L)|a%EhFLX#Hl_HDgZxHsSR1eE65LtKDEAIqD3#YC>!!&& zmGS>%Z6uzF7$7nq1M-XX#r(TFWH@Bm{grq_o^F~Fclt{HYIR$H zJ>d*@!~TW08Sm+u+3PWo7wQbX-8^#Okp&AZCZfXL?e0e43q#)<65C13F(5U=A`#(D z{OVap{wXg8$KjEHDghHGZkgS!r0x$W&m$QKW+D;-I*FM!MP*x|9HgA}Znf>#;rKpe zSn85R(M)o?MS{RE{M?Z8w#t*(PyRTX9(}cKQS!lOoSG0|n3hhi=L+DWD()nRll1WHwleoRhX^|<7Ay&FT5BB27QK0KL!#5^Xm5(9WYle z3ed3zWJ*C?Y)ltq(>IsD5E}jO@A4;!P~wv#xIxx|w7XKg*@+UTCg}7!U4Xx5HJ@2+ z0k7CpBti3`XKB@;Yf650V2K`quTYO+H!%)eV%E+&KYE+&8?~edu7a+|VvomSOvkzJ zRa-eMRB}Z+bC(fA@0hC`JneOTBwe4M{Lw=~OB_npA3K|E_tWm0M~%#8 zle=aJGYO9#@rFS=Va`<*eTNH7wrCPy!hKEkQYJa;#Eoe(fe%H$BP5||KMa&|$j>tT zvIP=$X78CJE*KRjRX<&R{)~`VxXmU-^LS=_T$?Z^i?&>41>V5RCx^4T1!$ioci?H@p9Fu!2J2 z8OcLz1c(T}Y&~-eki_V@Ax{4G<-gJVnj3_DLyEmZ-V0T!Uhi!Cbl6X=Bi$J%CDDdi z8F(wjnw2SXW}0qu1XRQpRftJKF2~icq*#=3JRoKmeCY2dqJF_$3H40O^9&3{?mb5SdIozQ?jzV-%S0Z2jukhBEmKvgs9 zbv;VUa2JQK(z3M-dwmm}XF<&WZANM4{~s_){}1|%|5Qtq^9Y>O@$OfA$c==Vc-~v2 zaGNQgUHmb=(>FuUG;7ixcS|#MB-O(o)x)4ZDDXVc_h@p-Cb-B#JHPp zz68JBmIsj~+dK9Z3AD45X@Y}(H1j>f&84i5sEp6|ZbxfdpwP-V8COc?NKHnr~bTgHOP zG9&zU$H(QDbaA`ndcks{3;{1&=+g&_T+MCl$1IEBHVmI|EWB+2;k?A`_=X`cQ9e$CwRtK5hi*E5t#p#&? zEqTLw{^vrSpCgeVF|d9s%n%}BA(1$rUIz4U0xg|D+L)b%`7DoobNL_K=}#OQrJX3K zrjg5-@j1-22&(KpX0mOiHr#hGV@~mWUUn3&l{%-oFNG5nJY@QOh^Qp{`u$6MH{LMW z)50l=N%>KKW<=4+j?WQ=FRZZxc(RRTm_v`Su{ph0L}mvarJD^991td?O4SD5AOxk_ z@lL+zY-Iip&P5_Zz}Bx+TD4mY57rlJ>1ui7L-zY#d=R9rJF+X+D03S8~Mz-Yw5+EEkl}>uV4BC|gUs$llEp4wBTksB(tzWif3g z)1(R1UZdlp-Q+2sM%Ob+^zfiKHT;5yPKcs>Xw+LO`ge$(@pDX$%Fn+WTrhF`k&%4m>|7Oau z6{oK@cNs!tivT83n=YE9`eVwMo5JASbB6#2W7Qv3Ek_Sgk(+qI#w7kW3)a3?SKnaq zgibAx)|e#7{s88*9;5Q z1dJ(qOM;{N;7+TYF|8-(sa6YI%ay7t1QtmS26F55#XCYG0>2+@rkf>`-S(qQMum#8 zYlI;lPamvS&6!J}_X$Xyil9qjj%1GA0Ouv6O%rKmc$dHh%jt2f113!Tote@tAaHkn zqv;-cBjRnSJ4nheWf`U3OZQh;>mffpei$A^Zd4hHQa~Z0#oOyGRoAwJ&^or~je-L= zC>h^p_hjQ17eA{7q1UG^vZA-<+bm_-k}CbTP$(ByD-h{(g4nuGG(M#Dck^q~uvYU0 z_7ocrX~e{+I*~THJU?)s(MuhC1%Ri+lM&jb%6o@6(mAMZxR^8egN|C28?KC5G( zPQ;?d#msWJ(iswwGbB9$^Jv*9k}2UOXY}oaUIs(D4e?shlF*DsY9uLJ(zrCE#_Xr) z5XO|jtuI424nGE_+rC!7j~9Ru9d2K-NA>!`M14f4zd#Ku>hOfjJO}cTWvbza`(p`I zU*TkJ`ODhTo8FrSXtR8WL|e7!Q=09sVb_cGeGBxBXyJHI3C7+oKyD_bwtaLuLTPEK z2Sp*T;_bEG5Gy(P|5W#tVO6ee+H{9>cY}1dbazU3cXuNl(hY)iNl8eEl$3yUH`3iG zda}F~>V|%=}owv(|$9S$N%7p7(jy&G@aLrPLQk_I9Fo?VfeUj-)hj=u94vJA`u|NET~j398Txa zGF(|r)Jjp5rlQ-Qbxoq6YdG}On*(=xR0RT?c}EGnO2BL<)B1XN8we8O1rI+&2c;$X z44!Y-Uw?yiwr>f67P6X8buQ{mAQ)0S9S*i%g5P_${u`i;8irlE_YOY~2nq^=IzWq2 zTQr;&@kP4Rw8+iavnr1gGuju`sl?0eG(8!YC%Q!e$U9g_v=6!4zwj16gb27Ne|Za# zYJjJx06bWKcnhBDQ*juemD?RsZXG?o4$Hq_|NhIag8jeQReV3X`=9Dg|5shbfB&uj z2N*@Ps?BepXhQ2{J0a{y1_z%5*%ZA+)LxAOIj-eYZh2fTSCM$CH_XnH5lrFh9QhcP;Zzg-GSvzE1=x0U!I+2f4%V2|!3CQIcXvP&O#lf%3$9APR#>Xx_CX z(fFziM6|0BAQ%`U&hx5s5M)wV5oElMPX3duQF*JcGF;yA;E&NWWyd76;s^;O5(#(~N+S9@mhMSrQjhXtO4N@Q%=$AV$cCF4iA$9!oa#nN!93=KwqP^0v1AZI#$eVyrT|y z=8aFa%M=bs{VGaSlWnicy=V0KQ(SrSU%thSB%d+Pg|=>AQC@BAla}_2arM+o!(V)R zERK)RB&%$lvb3Cwo$D$;)M*&ED~6T1vtjwE{d;KjF9Yzi{H?Y4xBdQme`}I?CcQRR z@M+*YdGBVTv&r@4aQ`n;#(((+(fdv&A=te{0yhM-Ua%AJb6^kv9O3?G4-^dqFgQ3Q z2rxR~kC~{0v7N1>xs$DfJKeAS`!EdsV@E#l22wPwCU_lHykEfBSf3g{Q+^Y8`fgAM z+*2;w*`$(ZQ`P6IgBu6vG>ricP~+v6-r&F|h43DEMw?|xA541l=Wlt9{ApGLgEojm zqP>^)K~nV_@J}0PLXpvfJ}|~}Brz5=ax3w7kjATOn#b*SwZG#4THVDcZkX%w0uIb( z5^GWmO`D2I_1=1BhZBIOZ)~qhMrQ$SQ_z`Nia1>f` zeOt>y0rq|c?TxrA#_hY^O3X8Ag+tDuEYdI#iw}VBhV#Xp%iqDmc5QP|cxXmofP|uQ82G+Xi74Q4_C0~*nm2w9%S<$P{=m&3% z+JUpS`f@B%5Qe7oj_D;D)&!S1SkX^E+Q?n_t8O;swUSg;e6s~z0^w$GxyG>;7y_=^ zYUnjNoh3ivdhK*9L8{sNW>~HLn$Kf7x%a`(!vhBB_mZ^$K@^Uk6BLw{Up^j;4lt;t z2Kad7@A(TDhvCBIAI8MsU;7CCCG!3EJt(t0W8ixzX;9hk56k>m=>fJUB(J1x3ChI6$j%Co{JTHx2B^URmn@8|j7$I#-2a;&>z^v4av7;& zefj1{*Cwrpn-2htH&iQE4#h zi$u+2CV5-lk4+|Ut6y6hB-&lVB(T1{FKP+PGo|m&ZnnqLYoOhF+8gU);+e2g={8}k zq|GytQ9VQjdPVDNatDK#YQLxB_Ha*Br-4&vYQxx54rriGWs(Y z`%j#=%9{TOl3(joSR+&(Eo17pk8sjvO5J_|C@I%(%V)HCk3QL)qah^(!0q!V zRw9)mPSP(BfX1jj#qWKWhAo`9gDF=)FzBuJggzY%S|g~=W(tpMc_M4X%3a4D7qx43 zSUR5Dm}hU17ZAo1u{-LpI$Crz-`$ZMm|^uJ$5Uv5(%Lc%158YaIj&72y-LJ()t(ee#9Jon9wnL>et>d6$@dAWaB1dCo63 zs>U?kRvZdVb*Sx##!Dy7W$SrtmscPwlCi>rPeC>`bG9)xbJaJtvowD^nw4_@98>8wF~ZqEGUugjqGsHLr}_7gb@1BWg2V=;iBu+_aP z96)G>?kS&#Jl)DuIeeqbUC_U#~fNGWnnXdx@2EQ^ohpl5pK zF$0IhYXn~|EQlHFpQ0Tv-2zz%-Gvx;*I~C8N?Q}#5rs1zCp(2fBa`E1^-Uw^Ci2W| znYKaY7_CkS){7h@pPrvh)=8p)sC%cVOX)QaPQs_5Fe{DdUwMpAy49pHUjV)MgeQ~0 zQCYO>reGJY^(0V%=mLqC<%Q0=_+YYs0IM|CuJO5L7x=4vF_eiKB8r43Fw>;mO|8Nx zq73}WJX4dC7W>>Bt2ATD2#q*jASHa+sjpn!hLra-MKkmK5MBzkPiX3*Hf0w?k%~Qg z-sZqj&1S`ZmpY=2@EoSp{NjBzn23imP(`cKdEJ-gOP)N&&&9Vyibh=priadu5DBRM zD%})oz%dx@7WR{^Qa1_e-abgoRe~1=R+tk*&}MwTR_FLJ2IcvNv(SYE#DW6&wyEKK zlUMxDJlx;p1IK;NK-%M*rj5c_i=K2gupw$?j+vwBmUcWyFKpG?=++N0}9Oja^u zomo@Xwht)Y^VV`Al6jiRJE>&A zJ@?k&6-Xdcgjy1NipGdF-Ug}C{SLV<`BXPBxXHeK+R7iv1qS9#TR`%~Ht+TIE^RSo z0=Me?X9c>7_bpIpk*My2x8=hT_2cnZXt7op5?JH3 z5#B;pV)@00#VSbA$;=0@e1=hk!ASdx{YD;S9XQXBok&`^GhHDS3BfS~B1W06kWxpO zGenh^y(L&UVHr;9X+}p}FpGXaVne#5km4;1mv_S!)tpQ5M}|vOd%tdcHgP9n0j-dL zO-%a;fiU4OKI)o94N)jtg;pJ6do|dZ1{F~O+X-3o&!2RZ&#%a=GLHA`ZXUn6jl)NW z!TvgZa!K*Fw-v9}$|%MVa;zg88mJcialh=M# zK~@cnIj0uNFS>J8xw1oIzmhhhZ$hM7Qc0n(Zi9n9M(ZvAM#;6sjZ1IE2qfL>MfwDG zbom_tn(ufbK63vCY7kiaB3li5cKB-VYzoV#DP{~x(XWNQho5W5WuP4-u8u=ii)XxF zd0SRr07JKJKQ5Nk8D5{2k4jE*(8;#+*v_O+v2hBBrNs1JHP*S<%|f(kc60 z*k+!)N(Eb}ta*!re%Wy1H@K%qZITVwq0%D8>2O{o3)-Q~U9z-?`OiFr8rcIUgkb3nIG@8tbsvpjg?my3S z`r&O4+wkervAttOph)^=!oscZ;CO_tB?h4^0py{M{08PA_*+#2v$I7!F7k$+jxC;V zF9At^kbX9Pfbf-Wxq%94nqtbH!k1S*$y^)s1V}SVeouq5N?D)G66;!Y=_D&1Gl4)e zVfC7MNKHPoXd zUad+D91Jd)r$_u^s%Z0$KG^F_((y|d-}@FKWN`kk^`H_UnjgtFR+Kr;+(RPql&WQe zJs~)Ho&vO@9HPk9T@_D3am5SdbwddmY>oqjN_mrmz9`)gwaXzAr@d%+PJJ%Mixo<{ z8N8ge8w_PpWyR8qw<}J=?Y3vq8SQQ!xZoT4;Egd1^#Qt$_cD+AhGzQK2F4EeP1~QP zJOj=zpA0F(5CACw$hsN7XWY<0y@R7$m|LIB*`!PpT%yx~N6U*-S4orlKLk5{RF9^wE;ffnFkVrOAy z;$Z)IDisC=SwILt)%5do5Gv6L{L)$mMZy1pv=sXN@xG_>w1j|co#P>M=Ky>KfK-?P z3X=DkI|lL}t&j6hA?5v|>gXC*x{F9H%IsbHTdW9n0n9{2u^o&|s?caw zD+ne!F@qBcur_aDwkrF977b{}@#$dUyO(NYi#lx4b|YH$LU_|N2U?$Ra9X(P`pfeM zgbU#-h|u^OJv8H-r(Pp-gkgo(`+Z(vvQ^D^IT88eCl6p*xcJtKw& zl?MzaDMKE6okuaSH~GXBLiF>aZ#c|#RNNTc0&j`t$KLbiXJ-=#$@ji%w(>|$*&mD~ zxXDnch_Gv+$hLoPS>_#UvN@^E1^;IJL)7@k%*=M{;o&xgwbjcG4z`MkLg$UjAR+De zI5-Q)1#Y|);)$|$*4exqvO9-57rX`Qt*0hW6@!LF;y&T7uQTJ#K`JTdl>~67vD44Z z=NN5m%(qm0H44V#DeQ$<2)eV}w8ZA=%B>N+iGb$Qg;MM>=o|yLZ|gK#yL=V6$%mX? zgg?e6LvMc`MR^`rjPhB({pCCH_bf32ky6sL=E!&Q2^oZe&mH%?UL7BOo4=446;jVp zHSFW%U1wE&zJ<_7I6xy6PlfnBqyzw40GLEf9`slLdV~hl zq!|yvN~go%zYN&T_jvbd2LA~^)PMD!SN_$n{VV^E?VNx|9_`QI%6tEj2?0k=0AL3Q zFu=D!4EN1}-yL+^A02eN?~Ffx^3U_8$P)BIpD^MF z3puO=_Y|@|WIZ&OW zr6J)tqX-ly!l|9lGdm6bVN1bcCtgZ=5Nl_LMCd4S!|rYpwYYd4*7m3?n2V>G2DsI(kgO~y zlyymC#_^90?B7-jG5)m_Vh2Rs2OgjY{3he~%YRQP^sv%D#jTuH`6tm^$5zj3!7_py z8T;mF*N}?AcyLtOzo;|dk=Aw`1TREBU2f-H4A_t>7Yni4)s$$L9Df(H8&c*QkoIDi z7c^B7_%mx(+03C*uL;js83^zU=_q#*h-l?&+|^4*b(i{cgjo2x!=Zuet8VG_ju#9t zQ5Zg~J6((K5Hj9Jg{+2h1e;#-^^`85WPoF)48@;S#npZ#866t$q>xz3;z@_fi_oSf zTKdXaqBP^}GlUJT)T|8-HuOp|Cc-5E{?+B@9#mJ&ST;!ibUuh{MIFPg0-)RcEkzu~ zx;Rb;P-ugeH+twc1@mMdKuzXPz($^}$VR+4wOtR)p$O6eYd?!^Vc_jNC^B#6NPqo0 zX~ba0{5^0|MSrIP%Qjre9Z~f27)h-){j|c*Thwk*3cOrs`?=4fi?tTIu40MTLq+W8 zU#41lWb^W-cwOa%<&NNSbwUMkkm|p@oXxU2wOEXVNC7P$pghPIJ5l;Cl|p&R6DK;kPnJtXvUxC! zc=mzqG6w$GEDzoo^IFsSE1yS?u4_-<;*f3!a_2McF{Y9Ri^*{`G|?22{Eb*m%v9;s za>2X+Mz+wwt(#7;B=HE~o+#iUO=gHBQ4=RTxxMzcjkWOl1BO16bgq#}xYQ(OmceQ+ zcFZWZ%W?%Iu9(-@k(({Xy+iz`y%=*lJQT~ank}Ih zn*@+3ob4!0F{XQN)n`iWgrW~!X>){2FvRQ~?O>g{Vl!gOm?~S*+ZtcZ^6nDt>YScN z^ywKZkJuZXUKQl|GqW+T40fg+I(J?dF-OWc8qdbxdh##Pi=Lvlv(la zg(Rgd-F~h6U@UV-jm!1fp;abu>~&V`d4|J~*h~k(8Y`jFvGT~GeqaH}3K_>jFcYng zW2~aKVCN@ZspTuIAiVKh`2m3&0o0C`iC&F7LlY3c7H}?Zv5Z|Ha_O^Bg|;FRUrfk1 zuzL~YuZ%|ZC3m#47iua^L2C%+t+ma~%u98+cD>A-d$@yTv(OrA(d0T3J=7mY)UzZ^285}*!}1Jpt6KkJ~p89|@dRTVRj4eOx}s`%ef z2mO%41VthISpeOW8VcE31Ii@;t>QgJ1^^!OXC=hM`M{L%vl4o^1aJ`kANxT66jLgJ z`t8_h{4^xFtlSd3^L()h&RNG>X8YlV9r(zrV-=eWxhd6c^J}aV3Qp#Y;8f1fzZ?@2 z`{^tjbibuq_P082TlZ$NP|jh(dVWg~G0Cg66{*IRk1&aq$g&1%+K$fXA}CI2 z0^v<;c=&2a6vSvtEW%3aE1VYyGifmIVD;r86pyleU3U8shw9g_{b9y*k zRzqU3*aUWfr`ye|FavM(Q?ybm_<|my z)oQAHwDRT(a~VD#gC$$Qk!x88&KmqoHJ*)u1!LO=7dr~AJG~mtI^eD$gT)?K-`QSX zdFoD@Rt*eWRjHM9;>UEO#hY@J!!NneTA4sMt&)aeqnBxD)nM3L;je|sx)l4al}C=9 zxY1{4D9e?ofS1GnMtp{mmu5BWY`Ct)XGy*jIjn{x?FnP>Y=!Ls2Q#J%($8AR3Q!A~ z{X;EuMOubfU3cmJ{#q{-r*m2cBntzJ!GHBfEi}vEKjnw|uio>@zxK6%)&IG@1E_|6 z`L+N4YA6=Dl9=^F9_Jz5!dBguo0oVizqd%7&elv7)>m|%>{?BeEQ>E4joP7`h~+~h zhzn#~Is4Rfb+zp|OtZ=9EL<=JD|m9TTw-E!uGoP%SLY@iIeUKF?%0w58xAC83V}7E8r? z9KXnt-+gg;5Q%6Y=~Xd$rhFOoHcd4IJ%Ee3RD&4mBA{uTZb>YywoW7uoO^nXPUIQ) z_MGn~_s+;J`4sDGhAyr$wX7PKd_vDFd`WNv-~vi#z2rD}8$KC!A^1473{2udh147> zY#!ZkCT2~Bo-lf`6vRBx?(?IvMW(}Pf9$o4mUBPU!;zRUh+-)^Smi_Yqna zsfrI+lr%@zEKq_V-ZFu7DpnAGo~tGj#GZL}NY%Rw1E;kz+%zCQ$__Jt3={7Dh8NV= zTQ0JG25%wH4Z7KFMUD)7m;+?`;Z&V0USE)|+8X zj{J1_1s}|mIuY)3_u9=1Y)^>5o8gF=cRekbrBJp!ewNSpv9Owq!#7&TwMN+{51uKK z>T#HYG495k1avPns+@2$zSNkjgF-8JA%lVUqzhSrD}ZPwe`3v^xjXL`>E+jTIPEGv z{1AT;08G0M4``xC1gK2-zalS4JOBtC19+hQnE7i(q;_c}bbYp0T^J1W8z6rScV+Mo zeH5$!OsD`l@cS+#6bg*2u`8h@pwDUS{u6fupmzWM1fh_*lRM*&Lv1i9XvzS_VHG72 zMP)@5CFNhVQ@?q!A6~o1Ki!k2{zrhJ`wk<3$_`LB{bG;-#8pN{j$h2~|9furKL9&p zmnmx$Uzw47@fatyRqEE5k}B;Ue><|_XkRg(F#)IlZ8j$iv;iD-SMAfjHvbAy&8Lg8 zaQCIWKp)!VVg1gd)H$-3hNZp?d9`*cJ+GKEPTJ_Sl&W-VyqSE%SytWM{~%=LT84fx zHiY#d$;C;+Wp0VJ+t4uWpRtmU30cj67>q#}*i2ucL(Q!v0}0^Cwk{h24{?_WA{pgm zCUAlvxodgiT^lv-qYKuwg@1tV++C}Z0bvB%qShCSRFssG#k4!05T39(3v9AhZy}vl z72iJS%oX(ywcrvRDNdK^)t?#BRwU6l_*ZNF!IB7rf_ZO(Vzz|B_rL$UZ21q}J^&^6y+OmGQ2u&Z$O^!FO$bo_ z_|HHp3~WrC0Hy&priWHHKpW)O<^Rl01OO90&Xj>4*s$OMe;i;a5CCTd&_d?qwi>3y zE6trrUGW#&t@C$l2d22E#%48Y_BlGP@AlFqw(R_PRWS{rY-f%KhWUL#l!=v}OjY}j zy?5r7^6hc$;Bza*2CXBsq_?1{>I(Jge24$0eM*GFwuD{%B7}T$iuSx1qn^NlO|}4o z-#8_1mfA);_cnA1cENW&3T=6(pKE*tITwX?D!Nh^nKuUt9O|uBzA*=wNGg7wo}N>e zK)!XvYe^UkcPuXr9XR#} zaDnsi5-49Yo_lmEPoA$cs)ObZd@`6#5OJ()6X_+b8xH^}Cvi8AYIrO2X-K+{nPd|; zv?Mb^zy`gSa6K$*kbB1N!Y7rS5LKD1)W$J*Bb_J{#wq}z=<);H^%H!nq=OLnPDK5P z^8G-X#LMWoAP3GJ!tKDlz`o0%tXV>ou46vXd;9VjlABF4gf6e0^bl>rl z{Hy!;O?f5(SnYv`gpq=Q?A~7h24cUS`&kA8Zs7;IO@NT`AzT6IR~gwD0j#S`_c9L5 zk1zkNYyDG&Rw$3Pz@N%H#w>bdtDqlHT`U)48m1jItuzluIX!uJSCJ^>ka|_8?G@fG zxXdW;G~y5MHJDCRDzs8ouJgj?1Uj5RgQ+EVC*f*S3ugB&<#VNdO|4S4Y{)@ebIYzA z6GNyJhK^dN(uZ@9X61~O%}acl!o zKYTsE{w9~)qi=~V9}E5LTm3-=|Fbh|4X|&YSo%okoJerAnNg6nJAuj08RkK>vzNxj zT55N2UcHl$n99}!ISF4Qja@U4lWVVxKybn+8gU4<@Bo$=s183PaFS9fAO7TrZOUP9W4G9d13-<-{axc({`IS`el5De2JZBK(g|MB^Eu_YJ zRoMvUz%lE(galNdYA^2%BBpl)cRR`lnV4QV1okb0R|po126X+b5DM;3Kaz2rUP~2* zX=Qv~h39-B>Et#3JmdqvES#3&Der}1z|nSI**6ImY|0aXP%f1H&4s&EJl&@LP5k3V zdA;|HqmJtbtvP$luuiRngq@~Vhfts%Dt^AAvPp%f_!OS zY?GVg)2=d*W8FObLeM51oHA6L+^U+EZS26BWkQWD z-I;420U-V?LjDGbcm#kD#(F<`2obUW+Ot*g3%T;gXf~HWHq1lU556Ln>AXP7OV|ZV zAD@D9P)GTIta%kG_KIqJ6v%L^gTQPC3cb+WE`nq^8kBepax#!HD z{0VE@-VUk^V~^hH%uz_jiug)*7yAvwlWXVaCX8ZKGcIYkTLBdZX^jdOp(JVMi3U9D zDt>H}${=?~joP2aNNlG>f+RJK55nN)Q40zCG4si4;je%Xb3dN~eRPU`MJ*m}Jd4t~ zLWbeJ(;A|QL8z<{FR26yyQ|&Dyl3!Dq=#N}PCdtubK`pQ76r9nk2gV5&}pp0^||9< zjA*M0$HDdHOQ!8)bui)uj*fw5>t1Fq3m9|$oR8ZMF_DHZNLs9UlClz)B&17n8>w+J z*!DGWXd%GB5E*S<<7mWfW;SoJ;Ed{;5pC+Tmg1AC&d>_oHW zLs2-LDj|EB?>9?^h-+vN-pg!i$@3(zyL_Q&5Is8T5R%@qur3FS@hI=Ix zFi<#@N6!K@kss6_ev}K0fV7MSzyiev@V`H`7alJE9ZUR^+wDYNEY`ghn4dob0LRYD zmU-OLnx_tZJ#v_-=b{KTD`;)M#80!?S(pt)E93=hnu6s#A+^<}3}{bQn-Zr&dAr>D zQf4t2P6;y&pEOr4<}f!Hw*QpM_w#G@eR7%Jk|NKv>1&P!VKub%%gB$@;w$jz6nOX_ z1B1!xn@Us!20)3WXkSJ}Y=l*s^laVHZ1I%Me6HzOHp>XR^Z-&_I8{lsx@j_+H;7n- zb!FNy7$8qee^+7RMk!D=CSqT{ z^gF{BeFLyzpLafIa;<6BGgsyORtw?zbi#;jx$9!fw?-;ES}D(k2iK%^lnLkglr35h~_Av`Oa)9PzBwK zWuI2nC+hhYnof3Do=xTYpSkhVkLQ1#&K=VML%;wa3lG<;9j=7~m<2a!)h|>d4fvY` z-(0(E z`fegwiM#_ij*FFFVzI_2SX*#)RS&-ldlA2x-iydp=+W@%&`tFED?9yqc+kn>E!gxc zvmAQT73f5W@x;Dm(xe@)^LX%gdyZSQBhH+YUr1-CWOapNvW8JP+f?ktQ`H!2GxjA@ zK$92<^dRZ_}F8RX42h@>Cy4 zM{e&xww1D`sk~^$ET}8esSUKUG=DalJ$KY$ZrQkO4Sjl4BJTp@;q8I(N+n4!l@WSg z4=Dm;=wrV#;1z(wT>v=T?jOw_;BVFpU-SADig22Z z5LiL~LM(9ac>jm{J&Hd?H z7jx6Ryj-yz-bE*jh2~q?bn3;deK}@eH0zrOaTJPa5lhU|e2a1zg6es`Mi4!=rqGRK zDhj30{{mH+znS^g$139vJrlDj%u?aQZDs;2`~_H6R4JWV?;@d9 zi9Oa)=!B_5>0q|x)=Ad9$1+kcexb7mZgH8ZVj!Md!7b4=VxL#_7H-%a5eM;vjNFph z`oc!++EHlT2O;02ipq;A$WSG$6TUrOnE?x>Y}A{*a7UZVidAb?U=n$zSORu&$}b!` z-XOZpWQZ(uFsPx5|1~G#G)UGCvUH-?i6wZk4tn4~gXzj9O6GuOz^HW1p61=cx?>p$ zH!V;btBMn}#FT_Owr?EUhC&>|3As0}3`TSBt`PZfn}F`KVqxLa)GulwS@HyG%0+zN zoDAu!)CK8;u%&Ac$;6&tgW<=Ik&8;o`&Tcty48O*MO*Gw3&KYuS)XogVMV#Wt$?2X6gM+EH)HqE{zGBV)Wd2GlGS0Mi3x$#}zvFmsI1HluvzQweQ zjUX3i$!T@0GaaUHLL`eRvq+-vv?wf>-^iUn?gP<{JrC(u6iD1Y6`R~Q5>WAqdJd8} zZ;<(+abJVdDJVa^KDGxAa`7yT&53D(vi;2)*rg2V4%~^atLAlLqW<)`0i~mc!gife zceKn+VBh`Hd+i+*Hq z)fZdz?I92#eHW8>Xq_?}>N6R;eD8*6fc=UEU}AvosSqF*(0q>tpwNM*kMr*P28jHl zyjxi07e@pfoG|S}`-4!-+Ccp0i9S#`BEZuRJ^hFI_xHXj^F31p6QPLe{WP5O0T;{( zpk)QnMX)lk+^6AyjQscIf5VFZbo4g$D2TxBDv0pLJz-vE3DKWNK zEq7ZVeCkT3lj7Q@eD7*f;leWi=eU*>&?_eVd+PbyLyabcOB~#sSvNQ2*2U4 z7QL~9v5T#t{=MF$mARu6gJ_O$j^K-DKLDlonE1yx0HCU;|8cE*e>PV4VdLmB>l*6o zGU*z-0FK2oa6ahXJaV{z4f4={W!dx8;CY=8dVr1O3`t*e%X5vdK)w@lPPMRfG}cWD zz12RXf>X;d{R#UPLZJZdHSIyQR2v~BnZ!r|n=8wv(V}7%Qjep@AZONx@<@B4V-3s7 zMIKp|%IY9v)UZa^erm?np*-s$Tk!-YQQI%}kvRoHSnQC;pd!#TFPyy=b^RJ}sF_i@w!gpfU1p+glGJYjBcxlr9gg zxoCrY-L_+{0hDMv8~y01@V- z5paNk86fliQ!wj~0ptIY`}_~4i=bkj^v=BjXu&tmYcJ8qN)1xa_!MCwJ_!K%CNIN~ z^56l3?el()f=-58a$dW>aSf+F1J#pai$EtqFNd~2l{}@Cue}Y$wc)p-r-y#7nXd`U zoVwD%8+K*!^q_haV(X9TqO*M=mN-^(X6Ckl*kVGbxNGXD``^67-`b0Rn=Uc}9e`?j zl-0YzIMjzO-qnesLc?6=Y24O%wjpval zWcG*^6{|7yMoE{gx5Z@lQ@WieTw+aQ8|eXV+?bm`xx;(Pg|0Ro$2sv3A%9L6@3;GJ zCM8jwUq%bs3y6(f0HZDcZnS@$F5X8+|G-Cv3K)j}f2utG)Nw;W60?06K;ECp`Rn)* z^wIe7do2C!*a!ujod9ycfRXxd0wzvCWPLOT2k-^{)^+>eahLxTh1={6ppLGevs~cx zV&+}=NvLlT2ci^o5eflw2OVt$OJq}Qp9kT{=G4yGrSvcXSAp|E%I=?D%h5k)ApTN{ zrE0n22l@2`-Pw;*!E^nJ^{ta^aZ~`x+FSl&(7@~5)^Icn-v!u?w^*p)ecK&HUu z*gx`Vu}~7pVi}iz8Uj@pZt<2j{@bi91{lsyN1=WWQx$UU= zZK^giLV>-r$F63Tto1V6o_Fy{v}WnQMh(c777Y2sl=AJt zc?-CJ4Pkr24C2BcDcnPZN}bhqAHPa|ZVIY@HFj~$yr1qj9qj_qJet1@6 zf5<7)v6e${2Hf)!#JS;He$`~f`Q!phztR(*aM6)ufCxMdQfcfP3cst(Y>h4U?9#PG ztD0bpOx|fyIro>W6S`nKeu1TFAhdN)PN!mfz#O0LIgsTPXYXc}njxpEb;lrcLBDPh z{sw`@-k!e3l()yRIc0S6)Cct|$aePN9i$n?EO9Ictw+>|@{#$fE_0s;uP4lEpRX?G z`igT?mkTEIr>BSX(&ZObUp~ndb*4)yLv;Cik3%7@;NJRRG?eL_kD*r!LBq+M7h2Y) o1{wExgs!Xzw`yH% - - - - 4.0.0 - - - com.wso2.openbanking.accelerator.data.publisher - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../pom.xml - - - com.wso2.openbanking.accelerator.authentication.data.publisher - bundle - WSO2 Open Banking - Authentication Data Publisher - - - - org.eclipse.osgi - org.eclipse.osgi - - - commons-logging - commons-logging - - - org.wso2.carbon.identity.framework - org.wso2.carbon.identity.application.authentication.framework - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.data.publisher.common - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - - - org.testng - testng - test - - - org.powermock - powermock-module-testng - test - - - org.powermock - powermock-api-mockito - test - - - org.mockito - mockito-all - test - - - - - - org.apache.maven.plugins - maven-compiler-plugin - - - org.apache.felix - maven-bundle-plugin - true - - - - ${project.artifactId} - - - com.wso2.openbanking.accelerator.authentication.data.publisher.internal - - - org.osgi.framework;version="${osgi.framework.imp.pkg.version.range}", - org.osgi.service.component;version="${osgi.service.component.imp.pkg.version.range}", - com.wso2.openbanking.accelerator.common.*; version="${project.version}", - com.wso2.openbanking.accelerator.data.publisher.common.*; version="${project.version}", - org.wso2.carbon.identity.application.authentication.framework.*; - version="${carbon.identity.framework.version.range}", - org.apache.commons.logging; version="${commons.logging.version}" - - - !com.wso2.openbanking.accelerator.authentication.data.publisher.internal, - com.wso2.openbanking.accelerator.authentication.data.publisher.*;version="${project.version}", - - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - false - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - org.jacoco - jacoco-maven-plugin - - - - **/*ServiceComponent.class - **/*AuthPublisherConstants.class - **/AuthenticationDataPublisherDataHolder.class - - - - - - default-prepare-agent - - prepare-agent - - - - default-prepare-agent-integration - - prepare-agent-integration - - - - default-report - - report - - - - default-report-integration - - report-integration - - - - default-check - - check - - - - - BUNDLE - - - INSTRUCTION - COVEREDRATIO - 0.80 - - - - - - - - - - org.apache.maven.plugins - maven-surefire-plugin - - - src/test/resources/testng.xml - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/constant/AuthPublisherConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/constant/AuthPublisherConstants.java deleted file mode 100644 index 63d3307f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/constant/AuthPublisherConstants.java +++ /dev/null @@ -1,40 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.authentication.data.publisher.constant; - -/** - * Class containing the constants for Open Banking Authentication Data Publisher module. - */ -public class AuthPublisherConstants { - - public static final String AUTHENTICATION_SUCCESSFUL = "AuthenticationSuccessful"; - public static final String AUTHENTICATION_FAILED = "AuthenticationFailed"; - public static final String AUTHENTICATION_ATTEMPTED = "AuthenticationAttempted"; - public static final String AUTHENTICATED_USER = "authenticatedUser"; - public static final String BASIC_AUTHENTICATOR = "BasicAuthenticator"; - public static final String LAST_LOGIN_FAILED_USER = "lastLoginFailedUser"; - public static final String USER_ID = "userId"; - public static final String AUTHENTICATION_STATUS = "authenticationStatus"; - public static final String AUTHENTICATION_STEP = "authenticationStep"; - public static final String TIMESTAMP = "timestamp"; - public static final String AUTHENTICATION_APPROACH = "authenticationApproach"; - public static final String AUTHENTICATION_INPUT_STREAM = "AuthenticationInputStream"; - public static final String REDIRECT = "redirect"; - public static final String STREAM_VERSION = "1.0.0"; -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/extension/AbstractAuthDataPublisher.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/extension/AbstractAuthDataPublisher.java deleted file mode 100644 index 03878c22..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/extension/AbstractAuthDataPublisher.java +++ /dev/null @@ -1,31 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.authentication.data.publisher.extension; - -import org.wso2.carbon.identity.application.authentication.framework.config.model.graph.js.JsAuthenticationContext; - -import java.util.Map; - -/** - * Open Banking abstract Authentication Data Publisher. - */ -public abstract class AbstractAuthDataPublisher { - - public abstract Map getAdditionalData(JsAuthenticationContext context, String authenticationStatus); -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/extension/DefaultAuthDataPublisher.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/extension/DefaultAuthDataPublisher.java deleted file mode 100644 index 63d72b02..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/extension/DefaultAuthDataPublisher.java +++ /dev/null @@ -1,36 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.authentication.data.publisher.extension; - -import org.wso2.carbon.identity.application.authentication.framework.config.model.graph.js.JsAuthenticationContext; - -import java.util.HashMap; -import java.util.Map; - -/** - * Open Banking default Authentication Data Publisher. - */ -public class DefaultAuthDataPublisher extends AbstractAuthDataPublisher { - - @Override - public Map getAdditionalData(JsAuthenticationContext context, String authenticationStatus) { - - return new HashMap<>(); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/internal/AuthenticationDataPublisherDataHolder.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/internal/AuthenticationDataPublisherDataHolder.java deleted file mode 100644 index b41072c4..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/internal/AuthenticationDataPublisherDataHolder.java +++ /dev/null @@ -1,64 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.authentication.data.publisher.internal; - -import com.wso2.openbanking.accelerator.authentication.data.publisher.extension.AbstractAuthDataPublisher; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.common.util.OpenBankingUtils; - -/** - * Data holder for Open Banking Authentication Data Publisher. - */ -public class AuthenticationDataPublisherDataHolder { - - private static volatile AuthenticationDataPublisherDataHolder instance; - private OpenBankingConfigurationService openBankingConfigurationService; - private AbstractAuthDataPublisher authDataPublisher; - - public static AuthenticationDataPublisherDataHolder getInstance() { - - if (instance == null) { - synchronized (AuthenticationDataPublisherDataHolder.class) { - if (instance == null) { - instance = new AuthenticationDataPublisherDataHolder(); - } - } - } - return instance; - } - - public void setOpenBankingConfigurationService( - OpenBankingConfigurationService openBankingConfigurationService) { - - this.openBankingConfigurationService = openBankingConfigurationService; - AbstractAuthDataPublisher abstractAuthDataPublisher = - (AbstractAuthDataPublisher) OpenBankingUtils.getClassInstanceFromFQN(openBankingConfigurationService - .getConfigurations().get("DataPublishing.AuthDataPublisher").toString()); - this.setAuthDataPublisher(abstractAuthDataPublisher); - } - - public AbstractAuthDataPublisher getAuthDataPublisher() { - return authDataPublisher; - } - - public void setAuthDataPublisher(AbstractAuthDataPublisher authDataPublisher) { - - this.authDataPublisher = authDataPublisher; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/internal/AuthenticationDataPublisherServiceComponent.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/internal/AuthenticationDataPublisherServiceComponent.java deleted file mode 100644 index dc45d91a..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/internal/AuthenticationDataPublisherServiceComponent.java +++ /dev/null @@ -1,106 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.authentication.data.publisher.internal; - -import com.wso2.openbanking.accelerator.authentication.data.publisher.service.AuthenticationDataPublisherService; -import com.wso2.openbanking.accelerator.authentication.data.publisher.service.AuthenticationDataPublisherServiceImpl; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.osgi.service.component.ComponentContext; -import org.osgi.service.component.annotations.Activate; -import org.osgi.service.component.annotations.Component; -import org.osgi.service.component.annotations.Deactivate; -import org.osgi.service.component.annotations.Reference; -import org.osgi.service.component.annotations.ReferenceCardinality; -import org.osgi.service.component.annotations.ReferencePolicy; -import org.wso2.carbon.identity.application.authentication.framework.JsFunctionRegistry; - -/** - * Method to register authentication data publisher OSGi Services. - */ -@Component( - name = "com.wso2.open.banking.authentication.data.publisher", - immediate = true -) -public class AuthenticationDataPublisherServiceComponent { - - private AuthenticationDataPublisherServiceImpl authenticationDataPublisherService; - private JsFunctionRegistry jsFunctionRegistry; - private static final Log log = LogFactory.getLog(AuthenticationDataPublisherServiceComponent.class); - - @Activate - protected void activate(ComponentContext context) { - - try { - authenticationDataPublisherService = new AuthenticationDataPublisherServiceImpl(); - - jsFunctionRegistry.register(JsFunctionRegistry.Subsystem.SEQUENCE_HANDLER, "publishAuthData", - (AuthenticationDataPublisherService) authenticationDataPublisherService::authDataExtractor); - } catch (Throwable e) { - log.error("Custom adaptive authentication function for data publishing activation failed", e); - } - - if (log.isDebugEnabled()) { - log.debug("Authentication Data Publisher component is activated successfully."); - } - } - - @Deactivate - protected void deactivate(ComponentContext context) { - - if (jsFunctionRegistry != null) { - jsFunctionRegistry.deRegister(JsFunctionRegistry.Subsystem.SEQUENCE_HANDLER, "publishAuthData"); - } - if (log.isDebugEnabled()) { - log.debug("Authentication Data Publisher component is deactivated."); - } - } - - @Reference( - service = JsFunctionRegistry.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetJsFunctionRegistry" - ) - public void setJsFunctionRegistry(JsFunctionRegistry jsFunctionRegistry) { - - this.jsFunctionRegistry = jsFunctionRegistry; - } - public void unsetJsFunctionRegistry(JsFunctionRegistry jsFunctionRegistry) { - this.jsFunctionRegistry = null; - } - - @Reference( - service = OpenBankingConfigurationService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetConfigService" - ) - public void setConfigService(OpenBankingConfigurationService openBankingConfigurationService) { - - AuthenticationDataPublisherDataHolder.getInstance() - .setOpenBankingConfigurationService(openBankingConfigurationService); - } - public void unsetConfigService(OpenBankingConfigurationService openBankingConfigurationService) { - - AuthenticationDataPublisherDataHolder.getInstance() - .setOpenBankingConfigurationService(openBankingConfigurationService); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/service/AuthenticationDataPublisherService.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/service/AuthenticationDataPublisherService.java deleted file mode 100644 index 0a44f1b8..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/service/AuthenticationDataPublisherService.java +++ /dev/null @@ -1,35 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.authentication.data.publisher.service; - -import org.wso2.carbon.identity.application.authentication.framework.config.model.graph.js.JsAuthenticationContext; - -import java.io.UnsupportedEncodingException; -import java.util.Map; - -/** - * Functional interface for authDataExtractor method that implements custom adaptive authentication function. - */ -@FunctionalInterface -public interface AuthenticationDataPublisherService { - - void authDataExtractor(JsAuthenticationContext context, String authenticationStatus, - Map parameterMap) - throws UnsupportedEncodingException; -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/service/AuthenticationDataPublisherServiceImpl.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/service/AuthenticationDataPublisherServiceImpl.java deleted file mode 100644 index 2466812f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/java/com/wso2/openbanking/accelerator/authentication/data/publisher/service/AuthenticationDataPublisherServiceImpl.java +++ /dev/null @@ -1,111 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.authentication.data.publisher.service; - -import com.wso2.openbanking.accelerator.authentication.data.publisher.constant.AuthPublisherConstants; -import com.wso2.openbanking.accelerator.authentication.data.publisher.extension.AbstractAuthDataPublisher; -import com.wso2.openbanking.accelerator.authentication.data.publisher.internal.AuthenticationDataPublisherDataHolder; -import com.wso2.openbanking.accelerator.authentication.data.publisher.internal.AuthenticationDataPublisherServiceComponent; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.data.publisher.common.util.OBDataPublisherUtil; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.application.authentication.framework.config.model.graph.js.JsAuthenticationContext; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedUser; - -import java.time.Instant; -import java.util.HashMap; -import java.util.Map; - -/** - * This class implements the custom adaptive authentication function for publishing authentication data. - */ -public class AuthenticationDataPublisherServiceImpl implements AuthenticationDataPublisherService { - - private static final Log log = LogFactory.getLog(AuthenticationDataPublisherServiceComponent.class); - - @Override - public void authDataExtractor(JsAuthenticationContext context, String authenticationStatus, - Map parameterMap) { - - HashMap authenticationData = new HashMap<>(); - String userName = null; - String authenticationStep = null; - long unixTimestamp = Instant.now().getEpochSecond(); - - //Retrieves the user ID from context - if (AuthPublisherConstants.AUTHENTICATION_SUCCESSFUL.equals(authenticationStatus)) { - AuthenticatedUser authenticatedUser = context.getWrapped().getLastAuthenticatedUser(); - userName = authenticatedUser.getAuthenticatedSubjectIdentifier(); - } else if (AuthPublisherConstants.AUTHENTICATION_FAILED.equals(authenticationStatus)) { - if ((context.getWrapped()).getParameters().get(AuthPublisherConstants.AUTHENTICATED_USER) != null) { - userName = ((AuthenticatedUser) context.getWrapped().getParameters() - .get(AuthPublisherConstants.AUTHENTICATED_USER)).getUserName(); - } else if ((context.getWrapped()).getParameters() - .get(AuthPublisherConstants.LAST_LOGIN_FAILED_USER) != null) { - userName = ((AuthenticatedUser) context.getWrapped().getParameters() - .get(AuthPublisherConstants.LAST_LOGIN_FAILED_USER)).getUserName(); - } else { - log.error("Failed to retrieve the user name relating to the authentication"); - } - } - - //Retrieves authentication step from context - if (context.getWrapped().getCurrentAuthenticator() != null) { - authenticationStep = context.getWrapped().getCurrentAuthenticator(); - } else if (context.getWrapped().getCurrentAuthenticator() == null && - AuthPublisherConstants.AUTHENTICATION_ATTEMPTED.equalsIgnoreCase(authenticationStatus)) { - authenticationStep = null; - } else { - log.error("Failed to retrieve the authentication step relating to the authentication"); - } - - //Collects additional data from toolkit level depending on the configurations - AuthenticationDataPublisherDataHolder authenticationDataPublisherDataHolder - = getAuthenticationDataPublisherDataHolder(); - AbstractAuthDataPublisher authDataPublisher = authenticationDataPublisherDataHolder.getAuthDataPublisher(); - Map additionalData = authDataPublisher.getAdditionalData(context, authenticationStatus); - for (Map.Entry dataElement : additionalData.entrySet()) { - authenticationData.put(dataElement.getKey(), dataElement.getValue()); - } - - //Collect data from the map sent by adaptive authentication function - if (parameterMap != null) { - HashMap authProperties = new HashMap<>(parameterMap); //write a config - for (Map.Entry element : authProperties.entrySet()) { - authenticationData.put(element.getKey(), element.getValue()); - } - } - authenticationData.put(AuthPublisherConstants.USER_ID, userName); - authenticationData.put(AuthPublisherConstants.AUTHENTICATION_STATUS, authenticationStatus); - authenticationData.put(AuthPublisherConstants.AUTHENTICATION_STEP, authenticationStep); - authenticationData.put(AuthPublisherConstants.TIMESTAMP, unixTimestamp); - authenticationData.put(AuthPublisherConstants.AUTHENTICATION_APPROACH, AuthPublisherConstants.REDIRECT); - - //Publish Data - OBDataPublisherUtil.publishData(AuthPublisherConstants.AUTHENTICATION_INPUT_STREAM, - AuthPublisherConstants.STREAM_VERSION, authenticationData); - } - - @Generated(message = "Added for testing purposes") - protected AuthenticationDataPublisherDataHolder getAuthenticationDataPublisherDataHolder() { - - return AuthenticationDataPublisherDataHolder.getInstance(); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/resources/findbugs-include.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/resources/findbugs-include.xml deleted file mode 100644 index 649d044e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,23 +0,0 @@ - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/test/java/com/wso2/openbanking/accelerator/authentication/data/publisher/service/OBAuthDataPublisherTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/test/java/com/wso2/openbanking/accelerator/authentication/data/publisher/service/OBAuthDataPublisherTest.java deleted file mode 100644 index 820b9543..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/test/java/com/wso2/openbanking/accelerator/authentication/data/publisher/service/OBAuthDataPublisherTest.java +++ /dev/null @@ -1,206 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.authentication.data.publisher.service; - -import com.wso2.openbanking.accelerator.authentication.data.publisher.constant.AuthPublisherConstants; -import com.wso2.openbanking.accelerator.authentication.data.publisher.extension.DefaultAuthDataPublisher; -import com.wso2.openbanking.accelerator.authentication.data.publisher.internal.AuthenticationDataPublisherDataHolder; -import com.wso2.openbanking.accelerator.data.publisher.common.util.OBDataPublisherUtil; -import org.apache.logging.log4j.LogManager; -import org.apache.logging.log4j.Logger; -import org.mockito.ArgumentCaptor; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.application.authentication.framework.config.model.graph.js.JsAuthenticationContext; -import org.wso2.carbon.identity.application.authentication.framework.context.AuthenticationContext; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedUser; - -import java.io.ByteArrayOutputStream; -import java.io.PrintStream; -import java.util.AbstractMap; -import java.util.Map; -import java.util.stream.Collectors; -import java.util.stream.Stream; - -/** - * Test for Open Banking default Authentication Data Publisher. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({OBDataPublisherUtil.class}) -public class OBAuthDataPublisherTest { - - public static final Map SCRIPT_DATA_MAP = Stream.of( - new AbstractMap.SimpleImmutableEntry<>("key1", "value1"), - new AbstractMap.SimpleImmutableEntry<>("key2", "value2"), - new AbstractMap.SimpleImmutableEntry<>("key3", "value3")) - .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue)); - public static final Map ADDITIONAL_DATA_MAP = Stream.of( - new AbstractMap.SimpleImmutableEntry<>("_key1", "_value1"), - new AbstractMap.SimpleImmutableEntry<>("_key2", "_value2"), - new AbstractMap.SimpleImmutableEntry<>("_key3", "_value3")) - .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue)); - public static final Map ASSERT_DATA_MAP = Stream.of( - new AbstractMap.SimpleImmutableEntry<>("_key1", "_value1"), - new AbstractMap.SimpleImmutableEntry<>("_key2", "_value2"), - new AbstractMap.SimpleImmutableEntry<>("_key3", "_value3"), - new AbstractMap.SimpleImmutableEntry<>("key1", "value1"), - new AbstractMap.SimpleImmutableEntry<>("key2", "value2"), - new AbstractMap.SimpleImmutableEntry<>("key3", "value3"), - new AbstractMap.SimpleImmutableEntry<>(AuthPublisherConstants.USER_ID, "bob@wso2.com"), - new AbstractMap.SimpleImmutableEntry<>(AuthPublisherConstants - .AUTHENTICATION_APPROACH, AuthPublisherConstants.REDIRECT), - new AbstractMap.SimpleImmutableEntry<>(AuthPublisherConstants - .AUTHENTICATION_STATUS, AuthPublisherConstants.AUTHENTICATION_SUCCESSFUL), - new AbstractMap.SimpleImmutableEntry<>(AuthPublisherConstants - .AUTHENTICATION_STEP, AuthPublisherConstants.BASIC_AUTHENTICATOR)) - .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue)); - private static ByteArrayOutputStream outContent; - private static Logger logger = null; - private static PrintStream printStream; - - @BeforeClass - public void initializeConfigurations() { - - OBAuthDataPublisherTest.outContent = new ByteArrayOutputStream(); - OBAuthDataPublisherTest.printStream = new PrintStream(OBAuthDataPublisherTest.outContent); - System.setOut(OBAuthDataPublisherTest.printStream); - OBAuthDataPublisherTest.logger = LogManager.getLogger(OBAuthDataPublisherTest.class); - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - - @Test - public void invokeFunctionWithoutUserID() throws Exception { - - outContent.reset(); - PowerMockito.mockStatic(OBDataPublisherUtil.class); - AuthenticationDataPublisherServiceImpl - authenticationDataPublisherService = Mockito.spy(AuthenticationDataPublisherServiceImpl.class); - DefaultAuthDataPublisher authDataPublisher = Mockito.mock(DefaultAuthDataPublisher.class); - AuthenticationDataPublisherDataHolder - authenticationDataPublisherDataHolder = Mockito.spy(AuthenticationDataPublisherDataHolder.class); - authenticationDataPublisherDataHolder.setAuthDataPublisher(authDataPublisher); - Mockito.doReturn(authenticationDataPublisherDataHolder).when(authenticationDataPublisherService) - .getAuthenticationDataPublisherDataHolder(); - PowerMockito.doNothing().when(OBDataPublisherUtil.class, - "publishData", Mockito.anyString(), Mockito.anyString(), Mockito.any()); - AuthenticationContext authContext = new AuthenticationContext(); - - //Setting null values for all possible user ID data - authContext.addParameter(AuthPublisherConstants.AUTHENTICATED_USER, null); - authContext.addParameter(AuthPublisherConstants.LAST_LOGIN_FAILED_USER, null); - authContext.setCurrentAuthenticator(AuthPublisherConstants.BASIC_AUTHENTICATOR); - AuthenticatedUser authenticatedUser = new AuthenticatedUser(); - authenticatedUser.setAuthenticatedSubjectIdentifier(null); - authContext.setSubject(authenticatedUser); - JsAuthenticationContext context = new JsAuthenticationContext(authContext); - authenticationDataPublisherService - .authDataExtractor(context, AuthPublisherConstants.AUTHENTICATION_FAILED, SCRIPT_DATA_MAP); - - Assert.assertTrue(OBAuthDataPublisherTest.outContent.toString().contains("Failed to retrieve the " + - "user name relating to the authentication")); - } - - @Test - public void invokeFunctionWithoutAuthenticationStep() throws Exception { - - outContent.reset(); - PowerMockito.mockStatic(OBDataPublisherUtil.class); - AuthenticationDataPublisherServiceImpl - authenticationDataPublisherService = Mockito.spy(AuthenticationDataPublisherServiceImpl.class); - DefaultAuthDataPublisher authDataPublisher = Mockito.mock(DefaultAuthDataPublisher.class); - AuthenticationDataPublisherDataHolder - authenticationDataPublisherDataHolder = Mockito.spy(AuthenticationDataPublisherDataHolder.class); - authenticationDataPublisherDataHolder.setAuthDataPublisher(authDataPublisher); - Mockito.doReturn(authenticationDataPublisherDataHolder).when(authenticationDataPublisherService) - .getAuthenticationDataPublisherDataHolder(); - PowerMockito.doNothing().when(OBDataPublisherUtil.class, - "publishData", Mockito.anyString(), Mockito.anyString(), Mockito.any()); - AuthenticationContext authContext = new AuthenticationContext(); - - authContext.addParameter(AuthPublisherConstants.AUTHENTICATED_USER, "mark@wso2.com"); - authContext.addParameter(AuthPublisherConstants.LAST_LOGIN_FAILED_USER, "anne@wso2.com"); - - //Setting null value for authentication step - authContext.setCurrentAuthenticator(null); - AuthenticatedUser authenticatedUser = new AuthenticatedUser(); - authenticatedUser.setAuthenticatedSubjectIdentifier("bob@wso2.com"); - authContext.setSubject(authenticatedUser); - JsAuthenticationContext context = new JsAuthenticationContext(authContext); - authenticationDataPublisherService - .authDataExtractor(context, AuthPublisherConstants.AUTHENTICATION_SUCCESSFUL, SCRIPT_DATA_MAP); - - Assert.assertTrue(OBAuthDataPublisherTest.outContent.toString().contains("Failed to retrieve the " + - "authentication step relating to the authentication")); - } - - @Test - public void publishData() throws Exception { - - outContent.reset(); - PowerMockito.mockStatic(OBDataPublisherUtil.class); - - //Mocking classes - AuthenticationDataPublisherServiceImpl - authenticationDataPublisherService = Mockito.spy(AuthenticationDataPublisherServiceImpl.class); - DefaultAuthDataPublisher authDataPublisher = Mockito.mock(DefaultAuthDataPublisher.class); - AuthenticationDataPublisherDataHolder - authenticationDataPublisherDataHolder = Mockito.spy(AuthenticationDataPublisherDataHolder.class); - Mockito.doReturn(ADDITIONAL_DATA_MAP).when(authDataPublisher).getAdditionalData(Mockito.any(), Mockito.any()); - authenticationDataPublisherDataHolder.setAuthDataPublisher(authDataPublisher); - Mockito.doReturn(authenticationDataPublisherDataHolder).when(authenticationDataPublisherService) - .getAuthenticationDataPublisherDataHolder(); - - //Invoking the method - AuthenticationContext authContext = new AuthenticationContext(); - authContext.addParameter(AuthPublisherConstants.AUTHENTICATED_USER, "mark@wso2.com"); - authContext.addParameter(AuthPublisherConstants.LAST_LOGIN_FAILED_USER, "anne@wso2.com"); - authContext.setCurrentAuthenticator(AuthPublisherConstants.BASIC_AUTHENTICATOR); - AuthenticatedUser authenticatedUser = new AuthenticatedUser(); - authenticatedUser.setAuthenticatedSubjectIdentifier("bob@wso2.com"); - authContext.setSubject(authenticatedUser); - JsAuthenticationContext context = new JsAuthenticationContext(authContext); - ArgumentCaptor argumentStream = ArgumentCaptor.forClass(String.class); - ArgumentCaptor argumentVersion = ArgumentCaptor.forClass(String.class); - ArgumentCaptor argumentData = ArgumentCaptor.forClass(Map.class); - PowerMockito.doNothing().when(OBDataPublisherUtil.class, - "publishData", argumentStream.capture(), argumentVersion.capture(), argumentData.capture()); - authenticationDataPublisherService - .authDataExtractor(context, AuthPublisherConstants.AUTHENTICATION_SUCCESSFUL, SCRIPT_DATA_MAP); - - Map receivedMap = argumentData.getValue(); - receivedMap.remove(AuthPublisherConstants.TIMESTAMP); - - //Assert the values passed to the publish() method - Assert.assertEquals(argumentStream.getValue(), AuthPublisherConstants.AUTHENTICATION_INPUT_STREAM); - Assert.assertEquals(argumentVersion.getValue(), AuthPublisherConstants.STREAM_VERSION); - Assert.assertEquals(receivedMap, ASSERT_DATA_MAP); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/test/resources/testng.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/test/resources/testng.xml deleted file mode 100644 index 596587bd..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.authentication.data.publisher/src/test/resources/testng.xml +++ /dev/null @@ -1,27 +0,0 @@ - - - - - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/pom.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/pom.xml deleted file mode 100644 index 281e547a..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/pom.xml +++ /dev/null @@ -1,201 +0,0 @@ - - - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../../pom.xml - - 4.0.0 - - com.wso2.openbanking.accelerator.data.publisher.common - bundle - WSO2 Open Banking - Data Publisher Common Module - - - - org.wso2.carbon.analytics-common - org.wso2.carbon.databridge.agent - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - - - com.github.spotbugs - spotbugs-annotations - ${spotbugs.annotations.version} - - - org.testng - testng - test - - - org.powermock - powermock-module-testng - test - - - org.powermock - powermock-api-mockito - test - - - org.mockito - mockito-all - test - - - - - - org.apache.felix - maven-bundle-plugin - true - - - - ${project.artifactId} - - - com.wso2.openbanking.accelerator.data.publisher.common.internal - - - org.osgi.framework; version="${osgi.framework.imp.pkg.version.range}", - org.osgi.service.component; version="${osgi.service.component.imp.pkg.version.range}", - org.wso2.carbon.databridge.agent.*; version="${carbon.analytics.common.version.range}", - org.wso2.carbon.databridge.commons.*; version="${carbon.analytics.common.version.range}", - com.wso2.openbanking.accelerator.common.*; version="${project.version}", - org.apache.tomcat.dbcp.pool2.*; version="${tomcat.catalina.version}" - - - !com.wso2.openbanking.accelerator.data.publisher.common.internal, - com.wso2.openbanking.accelerator.data.publisher.common.constants; version="${project.version}", - com.wso2.openbanking.accelerator.data.publisher.common.model; version="${project.version}", - com.wso2.openbanking.accelerator.data.publisher.common.util; version="${project.version}", - com.wso2.openbanking.accelerator.data.publisher.common.*; version="${project.version}" - - <_dsannotations>* - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - true - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-exclude.xml - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - org.jacoco - jacoco-maven-plugin - - - - **/*ServiceComponent.class - **/*DataPublishingConstants.class - **/*OBAnalyticsEvent.class - **/*QueueWorker.class - **/*EventQueue.class - - - - - default-prepare-agent - - prepare-agent - - - - default-prepare-agent-integration - - prepare-agent-integration - - - - default-report - - report - - - - default-report-integration - - report-integration - - - - default-check - - check - - - - - BUNDLE - - - INSTRUCTION - COVEREDRATIO - 0.80 - - - - - - - - - - org.apache.maven.plugins - maven-surefire-plugin - - - src/test/resources/testng.xml - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/DataPublisherFactory.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/DataPublisherFactory.java deleted file mode 100644 index d7b17ed2..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/DataPublisherFactory.java +++ /dev/null @@ -1,43 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.data.publisher.common; - -import org.apache.tomcat.dbcp.pool2.BasePooledObjectFactory; -import org.apache.tomcat.dbcp.pool2.PooledObject; -import org.apache.tomcat.dbcp.pool2.impl.DefaultPooledObject; - -/** - * Data Publisher Factory class. - * - * @param - */ -public class DataPublisherFactory extends BasePooledObjectFactory { - - @Override - public OpenBankingDataPublisher create() { - - return (OpenBankingDataPublisher) new OBThriftDataPublisher(); - } - - @Override - public PooledObject wrap(OpenBankingDataPublisher openBankingDataPublisher) { - - return new DefaultPooledObject(openBankingDataPublisher); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/DataPublisherPool.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/DataPublisherPool.java deleted file mode 100644 index ade99f3a..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/DataPublisherPool.java +++ /dev/null @@ -1,36 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.data.publisher.common; - -import org.apache.tomcat.dbcp.pool2.PooledObjectFactory; -import org.apache.tomcat.dbcp.pool2.impl.GenericObjectPool; -import org.apache.tomcat.dbcp.pool2.impl.GenericObjectPoolConfig; - -/** - * Data Publisher Pool class. - * @param - */ -public class DataPublisherPool extends GenericObjectPool { - - public DataPublisherPool(PooledObjectFactory factory, - GenericObjectPoolConfig config) { - - super(factory, config); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/EventQueue.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/EventQueue.java deleted file mode 100644 index 99cd0b06..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/EventQueue.java +++ /dev/null @@ -1,65 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.data.publisher.common; - -import com.wso2.openbanking.accelerator.data.publisher.common.model.OBAnalyticsEvent; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.concurrent.ArrayBlockingQueue; -import java.util.concurrent.BlockingQueue; -import java.util.concurrent.ExecutorService; -import java.util.concurrent.Executors; -import java.util.concurrent.RejectedExecutionException; - -/** - * Event queue wrapper class wrapping the ArrayBlockingQueue. - */ -public class EventQueue { - - private static final Log log = LogFactory.getLog(EventQueue.class); - private final BlockingQueue eventQueue; - private final ExecutorService publisherExecutorService; - - public EventQueue(int queueSize, int workerThreadCount) { - - // Note : Using a fixed worker thread pool and a bounded queue to control the load on the server - publisherExecutorService = Executors.newFixedThreadPool(workerThreadCount); - eventQueue = new ArrayBlockingQueue<>(queueSize); - } - - public void put(OBAnalyticsEvent obAnalyticsEvent) { - - try { - if (eventQueue.offer(obAnalyticsEvent)) { - publisherExecutorService.submit(new QueueWorker(eventQueue, publisherExecutorService)); - } else { - log.error("Event queue is full. Starting to drop OB analytics events."); - } - } catch (RejectedExecutionException e) { - log.warn("Task submission failed. Task queue might be full", e); - } - } - - @Override - protected void finalize() throws Throwable { - publisherExecutorService.shutdown(); - super.finalize(); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/OBThriftDataPublisher.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/OBThriftDataPublisher.java deleted file mode 100644 index 94aa48f7..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/OBThriftDataPublisher.java +++ /dev/null @@ -1,288 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.data.publisher.common; - -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.data.publisher.common.constants.DataPublishingConstants; -import com.wso2.openbanking.accelerator.data.publisher.common.internal.OBAnalyticsDataHolder; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.databridge.agent.DataPublisher; -import org.wso2.carbon.databridge.agent.exception.DataEndpointAgentConfigurationException; -import org.wso2.carbon.databridge.agent.exception.DataEndpointAuthenticationException; -import org.wso2.carbon.databridge.agent.exception.DataEndpointConfigurationException; -import org.wso2.carbon.databridge.agent.exception.DataEndpointException; -import org.wso2.carbon.databridge.commons.Event; -import org.wso2.carbon.databridge.commons.exception.TransportException; -import org.wso2.carbon.databridge.commons.utils.DataBridgeCommonsUtils; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Locale; -import java.util.Map; -import java.util.stream.Collectors; - -/** - * Open Banking Thrift Data publisher. - */ -public class OBThriftDataPublisher implements OpenBankingDataPublisher { - - private DataPublisher dataPublisher; - private Map> streamAttributeMap = new HashMap<>(); - private Map> attributeValidationMap; - private static final Log log = LogFactory.getLog(OBThriftDataPublisher.class); - private Map obConfigurations; - - public OBThriftDataPublisher() { - this.init(); - } - - @Override - public void publish(String streamName, String streamVersion, Map analyticsData) { - - // Set payloads - Object[] payload = setPayload(streamName, analyticsData); - - // Log error and return if payload is not set - if (payload.length == 0) { - log.error("Error while setting payload to publish data."); - return; - } - - // Create wso2 event to publish - Event event = new Event(); - event.setStreamId(DataBridgeCommonsUtils.generateStreamId(streamName, streamVersion)); - event.setMetaData(null); - event.setCorrelationData(null); - event.setPayloadData(payload); - - try { - // Try to publish event with timeout - // If the queue is full, this will wait timeout time and retry to add to queue. If still full this - // returns false - boolean published = dataPublisher.tryPublish(event, - Long.parseLong((String) obConfigurations.get(DataPublishingConstants.THRIFT_PUBLISHING_TIMEOUT))); - if (!published) { - log.error("Unable to publish data for stream: " + streamName.replaceAll("[\r\n]", "") + - ". Queue is full."); - } - } catch (Exception e) { - // Catching exception and logging error because data publishing issues should not hinder other flows. - log.error("Error occurred while publishing the data", e); - } - - } - - /** - * Initialize OB Thrift Data publisher. - * This method initializes data publisher and read the attribute map for each data stream. - */ - protected void init() { - - log.debug("Initializing the Open Banking Thrift data publisher"); - obConfigurations = OBAnalyticsDataHolder.getInstance().getConfigurationMap(); - String serverUser = (String) obConfigurations.get(DataPublishingConstants.DATA_PUBLISHING_USERNAME); - String serverPassword = (String) obConfigurations.get(DataPublishingConstants.DATA_PUBLISHING_PASSWORD); - String serverURL = (String) obConfigurations.get(DataPublishingConstants.DATA_PUBLISHING_SERVER_URL); - String authURL = (String) obConfigurations.get(DataPublishingConstants.DATA_PUBLISHING_AUTH_URL); - - if (serverURL == null || serverPassword == null || serverUser == null) { - log.error("Error while retrieving publisher server configs"); - return; - } - - log.debug("Reading attribute list for data streams"); - buildStreamAttributeMap(); - attributeValidationMap = OBAnalyticsDataHolder.getInstance().getOpenBankingConfigurationService() - .getDataPublishingValidationMap(); - - try { - //Create new DataPublisher for the tenant. - dataPublisher = getDataPublisher(serverURL, authURL, serverUser, serverPassword); - } catch (DataEndpointConfigurationException e) { - log.error("Error while creating data publisher with the configurations", e); - } catch (DataEndpointException | DataEndpointAgentConfigurationException | TransportException | - DataEndpointAuthenticationException e) { - log.error("Error while creating data publisher", e); - } - - } - - @Generated(message = "Method to get new Thrift data publisher") - protected DataPublisher getDataPublisher(String serverURL, String authURL, String serverUser, - String serverPassword) - throws DataEndpointAuthenticationException, DataEndpointAgentConfigurationException, TransportException, - DataEndpointException, DataEndpointConfigurationException { - - return new DataPublisher(null, serverURL, authURL, serverUser, serverPassword); - } - - protected void setDataPublisher(DataPublisher dataPublisher) { - this.dataPublisher = dataPublisher; - } - - /** - * Create event payload for the given stream. - * - * @param streamName stream name - * @param analyticsData map of data to be published - * @return payload of object[] - */ - protected Object[] setPayload(String streamName, Map analyticsData) { - - if (streamAttributeMap.containsKey(streamName)) { - List attributes = streamAttributeMap.get(streamName); - boolean isValid = validateAttributes(streamName, attributes, analyticsData); - if (isValid) { - ArrayList payload = new ArrayList<>(); - for (String attribute : attributes) { - payload.add(analyticsData.get(attribute)); - } - return payload.toArray(); - } - } - return new Object[]{}; - } - - /** - * Validate whether the required parameters are present and are of correct data type. - * @param streamName stream name - * @param analyticsData data map - * @return boolean isValid - */ - private boolean validateAttributes(String streamName, List attributes, Map analyticsData) { - - Map> attributeValidations = getAttributeValidationMap(); - for (String attribute: attributes) { - String attributeNameKey = streamName + "_" + attribute; - boolean isRequired = (boolean) attributeValidations.get(attributeNameKey) - .get(OpenBankingConstants.REQUIRED); - String type = (String) attributeValidations.get(attributeNameKey).get(OpenBankingConstants.ATTRIBUTE_TYPE); - - // validation for required attributes - if (isRequired) { - if (analyticsData.containsKey(attribute)) { - if (analyticsData.get(attribute) == null) { - log.error(attribute.replaceAll("[\r\n]", "") + " is missing in data map for " + - streamName.replaceAll("[\r\n]", "") + ". This event " - + "will not be processed further."); - return false; - } - } else { - log.error(attribute.replaceAll("[\r\n]", "") + " is missing in data map for " + - streamName.replaceAll("[\r\n]", "") + ". This event " - + "will not be processed further."); - return false; - } - } - - // validation for data type - if (analyticsData.containsKey(attribute) && analyticsData.get(attribute) != null) { - if (!isValidDataType(type, attribute, analyticsData.get(attribute))) { - return false; - } - } - } - return true; - } - - /** - * Build a map of attributes to be published for each stream. - */ - protected void buildStreamAttributeMap() { - - Map> dataStreamAttributes = OBAnalyticsDataHolder.getInstance() - .getOpenBankingConfigurationService().getDataPublishingStreams(); - dataStreamAttributes.keySet().forEach(dataStream -> { - Map integerStringMap = dataStreamAttributes.get(dataStream); - List attributeList = integerStringMap.keySet().stream() - .map(integerStringMap::get).collect(Collectors.toList()); - streamAttributeMap.put(dataStream, attributeList); - }); - } - - @Generated(message = "Added for testing purposes") - protected Map> getStreamAttributeMap() { - - return streamAttributeMap; - } - - @Generated(message = "Added for testing purposes") - protected Map> getAttributeValidationMap() { - return attributeValidationMap; - } - - @SuppressFBWarnings("IMPROPER_UNICODE") - // Suppressed content - ype.toLowerCase(Locale.ENGLISH) - // Suppression reason - False Positive : Since the value is used in switch statements, it cannot be used - // maliciously - // Suppressed warning count - 1 - private boolean isValidDataType(String type, String attributeName, Object attributeValue) { - - Class attributeClass = attributeValue.getClass(); - switch (type.toLowerCase(Locale.ENGLISH)) { - case "string" : - if (!(attributeClass.equals(String.class))) { - logInvalidDataTypeError(attributeName, String.class.getName(), attributeClass.getName()); - return false; - } - break; - case "int" : - if (!(attributeClass.equals(Integer.class))) { - logInvalidDataTypeError(attributeName, Integer.class.getName(), attributeClass.getName()); - return false; - } - break; - case "long" : - if (!(attributeClass.equals(Long.class))) { - logInvalidDataTypeError(attributeName, Long.class.getName(), attributeClass.getName()); - return false; - } - break; - case "boolean" : - if (!(attributeClass.equals(Boolean.class))) { - logInvalidDataTypeError(attributeName, Boolean.class.getName(), attributeClass.getName()); - return false; - } - break; - case "double" : - if (!(attributeClass.equals(Double.class))) { - logInvalidDataTypeError(attributeName, Double.class.getName(), attributeClass.getName()); - return false; - } - break; - case "float" : - if (!(attributeClass.equals(Float.class))) { - logInvalidDataTypeError(attributeName, Float.class.getName(), attributeClass.getName()); - return false; - } - break; - } - return true; - } - - private void logInvalidDataTypeError(String attributeName, String expectedDataType, String actualDataType) { - log.error(attributeName.replaceAll("[\r\n]", "") + " is expecting a " + - expectedDataType.replaceAll("[\r\n]", "") + " type attribute while attribute of " + - "type " + actualDataType.replaceAll("[\r\n]", "") + " is present."); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/OpenBankingDataPublisher.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/OpenBankingDataPublisher.java deleted file mode 100644 index 20aab29d..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/OpenBankingDataPublisher.java +++ /dev/null @@ -1,29 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.data.publisher.common; -import java.util.Map; - -/** - * Interface for Open Banking Data publisher. - */ -public interface OpenBankingDataPublisher { - - void publish(String streamName, String streamVersion, Map analyticsData); - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/QueueWorker.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/QueueWorker.java deleted file mode 100644 index 56ad52b3..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/QueueWorker.java +++ /dev/null @@ -1,61 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.data.publisher.common; - -import com.wso2.openbanking.accelerator.data.publisher.common.model.OBAnalyticsEvent; -import com.wso2.openbanking.accelerator.data.publisher.common.util.OBDataPublisherUtil; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.concurrent.BlockingQueue; -import java.util.concurrent.ExecutorService; -import java.util.concurrent.ThreadPoolExecutor; - -/** - * Queue worker implementation for publish events in queue. - */ -public class QueueWorker implements Runnable { - - private BlockingQueue eventQueue; - private ExecutorService executorService; - private static final Log log = LogFactory.getLog(QueueWorker.class); - - public QueueWorker(BlockingQueue queue, ExecutorService executorService) { - - this.eventQueue = queue; - this.executorService = executorService; - } - - public void run() { - - ThreadPoolExecutor threadPoolExecutor = ((ThreadPoolExecutor) executorService); - - do { - OBAnalyticsEvent event = eventQueue.poll(); - if (event != null) { - OpenBankingDataPublisher dataPublisher = OBDataPublisherUtil.getDataPublisherInstance(); - if (dataPublisher != null) { - dataPublisher.publish(event.getStreamName(), event.getStreamVersion(), event.getAnalyticsData()); - OBDataPublisherUtil.releaseDataPublishingInstance(dataPublisher); - } - } - } while (threadPoolExecutor.getActiveCount() == 1 && eventQueue.size() != 0); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/constants/DataPublishingConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/constants/DataPublishingConstants.java deleted file mode 100644 index 30f8d7e1..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/constants/DataPublishingConstants.java +++ /dev/null @@ -1,41 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.data.publisher.common.constants; - -/** - * Class containing the constants for Open Banking Data Publishing module. - */ -public class DataPublishingConstants { - - public static final String DATA_PUBLISHING_USERNAME = "DataPublishing.Username"; - public static final String DATA_PUBLISHING_PASSWORD = "DataPublishing.Password"; - public static final String DATA_PUBLISHING_SERVER_URL = "DataPublishing.ServerURL"; - public static final String DATA_PUBLISHING_AUTH_URL = "DataPublishing.AuthURL"; - public static final String DATA_PUBLISHING_POOL_SIZE = "DataPublishing.PoolSize"; - public static final String DATA_PUBLISHING_POOL_WAIT_TIME = "DataPublishing.PoolWaitTimeMs"; - public static final String DATA_PUBLISHING_PROTOCOL = "DataPublishing.Protocol"; - public static final String DATA_PUBLISHING_ENABLED = "DataPublishing.Enabled"; - public static final String ELK_ANALYTICS_ENABLED = "ELKAnalytics.Enabled"; - public static final String APIM_ANALYTICS_ENABLED = "APIMAnalytics.Enabled"; - public static final String QUEUE_SIZE = "DataPublishing.QueueSize"; - public static final String WORKER_THREAD_COUNT = "DataPublishing.WorkerThreadCount"; - public static final String THRIFT_PUBLISHING_TIMEOUT = "DataPublishing.Thrift.PublishingTimeout"; - public static final String LOG_FILE_NAME = "OB_LOG"; - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/internal/OBAnalyticsDataHolder.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/internal/OBAnalyticsDataHolder.java deleted file mode 100644 index 84bb2373..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/internal/OBAnalyticsDataHolder.java +++ /dev/null @@ -1,117 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.data.publisher.common.internal; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.data.publisher.common.DataPublisherFactory; -import com.wso2.openbanking.accelerator.data.publisher.common.DataPublisherPool; -import com.wso2.openbanking.accelerator.data.publisher.common.EventQueue; -import com.wso2.openbanking.accelerator.data.publisher.common.OpenBankingDataPublisher; -import com.wso2.openbanking.accelerator.data.publisher.common.constants.DataPublishingConstants; -import org.apache.tomcat.dbcp.pool2.impl.GenericObjectPoolConfig; - -import java.util.Map; - -/** - * Data holder for Open Banking Analytics. - */ -public class OBAnalyticsDataHolder { - - private static volatile OBAnalyticsDataHolder instance; - private OpenBankingConfigurationService openBankingConfigurationService; - private Map configurationMap; - private DataPublisherPool pool; - private int poolSize; - private EventQueue eventQueue; - - public static OBAnalyticsDataHolder getInstance() { - - if (instance == null) { - synchronized (OBAnalyticsDataHolder.class) { - if (instance == null) { - instance = new OBAnalyticsDataHolder(); - } - } - } - return instance; - } - - public Map getConfigurationMap() { - - return configurationMap; - } - - public OpenBankingConfigurationService getOpenBankingConfigurationService() { - - return openBankingConfigurationService; - } - - public void setOpenBankingConfigurationService( - OpenBankingConfigurationService openBankingConfigurationService) { - - this.openBankingConfigurationService = openBankingConfigurationService; - this.configurationMap = openBankingConfigurationService.getConfigurations(); - } - - /** - * Initialize pool of data publishers. - */ - public void initializePool() { - - GenericObjectPoolConfig config = new GenericObjectPoolConfig<>(); - poolSize = Integer.parseInt((String) configurationMap.get(DataPublishingConstants.DATA_PUBLISHING_POOL_SIZE)); - int timeout = Integer.parseInt( - (String) configurationMap.get(DataPublishingConstants.DATA_PUBLISHING_POOL_WAIT_TIME)); - config.setMaxIdle(poolSize); - config.setMaxTotal(poolSize); - config.setMaxWaitMillis(timeout); - pool = new DataPublisherPool<>(new DataPublisherFactory<>(), config); - } - - public DataPublisherPool getDataPublisherPool() { - - return pool; - } - - public void closePool() { - - pool.close(); - } - - public void initializeEventQueue() { - - int queueSize = Integer.parseInt((String) configurationMap.get(DataPublishingConstants.QUEUE_SIZE)); - int workerThreadCount = - Integer.parseInt((String) configurationMap.get(DataPublishingConstants.WORKER_THREAD_COUNT)); - eventQueue = new EventQueue(queueSize, workerThreadCount); - } - - public EventQueue getEventQueue() { - - return eventQueue; - } - - @Generated(message = "Event queue setter for testing purposes") - public void setEventQueue(EventQueue eventQueue) { - - this.eventQueue = eventQueue; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/internal/OBAnalyticsServiceComponent.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/internal/OBAnalyticsServiceComponent.java deleted file mode 100644 index 9497f52e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/internal/OBAnalyticsServiceComponent.java +++ /dev/null @@ -1,73 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.data.publisher.common.internal; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.osgi.service.component.ComponentContext; -import org.osgi.service.component.annotations.Activate; -import org.osgi.service.component.annotations.Component; -import org.osgi.service.component.annotations.Deactivate; -import org.osgi.service.component.annotations.Reference; -import org.osgi.service.component.annotations.ReferenceCardinality; -import org.osgi.service.component.annotations.ReferencePolicy; - -/** - * Service class for Open Banking Data Publishing Component. - */ -@Component( - name = "com.wso2.openbanking.accelerator.data.publisher.common.internal.OBAnalyticsServiceComponent", - immediate = true -) -public class OBAnalyticsServiceComponent { - - private static final Log log = LogFactory.getLog(OBAnalyticsServiceComponent.class); - - @Activate - protected void activate(ComponentContext context) { - - OBAnalyticsDataHolder.getInstance().initializePool(); - OBAnalyticsDataHolder.getInstance().initializeEventQueue(); - log.debug("Open banking data publishing component is activated "); - } - - @Deactivate - protected void deactivate(ComponentContext context) { - - OBAnalyticsDataHolder.getInstance().closePool(); - log.debug("Open banking data publishing component is deactivated "); - } - - @Reference( - service = OpenBankingConfigurationService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetConfigService" - ) - public void setConfigService(OpenBankingConfigurationService openBankingConfigurationService) { - - OBAnalyticsDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - } - - public void unsetConfigService(OpenBankingConfigurationService openBankingConfigurationService) { - - OBAnalyticsDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/model/OBAnalyticsEvent.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/model/OBAnalyticsEvent.java deleted file mode 100644 index 2aff8b6c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/model/OBAnalyticsEvent.java +++ /dev/null @@ -1,53 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.data.publisher.common.model; - -import java.util.Map; - -/** - * Open Banking Analytics event model class. - */ -public class OBAnalyticsEvent { - - private String streamName; - private String streamVersion; - private Map analyticsData; - - - public OBAnalyticsEvent(String streamName, String streamVersion, Map analyticsData) { - this.streamName = streamName; - this.streamVersion = streamVersion; - this.analyticsData = analyticsData; - } - - public String getStreamName() { - - return streamName; - } - - public String getStreamVersion() { - - return streamVersion; - } - - public Map getAnalyticsData() { - - return analyticsData; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/util/OBDataPublisherUtil.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/util/OBDataPublisherUtil.java deleted file mode 100644 index 771b8d4d..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/java/com/wso2/openbanking/accelerator/data/publisher/common/util/OBDataPublisherUtil.java +++ /dev/null @@ -1,92 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.data.publisher.common.util; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.AnalyticsLogsUtils; -import com.wso2.openbanking.accelerator.data.publisher.common.DataPublisherPool; -import com.wso2.openbanking.accelerator.data.publisher.common.EventQueue; -import com.wso2.openbanking.accelerator.data.publisher.common.OpenBankingDataPublisher; -import com.wso2.openbanking.accelerator.data.publisher.common.constants.DataPublishingConstants; -import com.wso2.openbanking.accelerator.data.publisher.common.internal.OBAnalyticsDataHolder; -import com.wso2.openbanking.accelerator.data.publisher.common.model.OBAnalyticsEvent; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.Map; - -/** - * Utility class for Data Publishing. - */ -public class OBDataPublisherUtil { - - private static final Log log = LogFactory.getLog(OBDataPublisherUtil.class); - - public static OpenBankingDataPublisher getDataPublisherInstance() { - - DataPublisherPool pool = - OBAnalyticsDataHolder.getInstance().getDataPublisherPool(); - try { - return pool.borrowObject(); - } catch (Exception e) { - log.error("Error while receiving Thrift Data Publisher from the pool."); - } - return null; - } - - public static void releaseDataPublishingInstance(OpenBankingDataPublisher instance) { - - OBAnalyticsDataHolder.getInstance().getDataPublisherPool().returnObject(instance); - log.debug("Data publishing instance released to the pool"); - } - - /** - * Util method to publish OB analytics data. - * This method will put received data to an event queue and take care of asynchronous data publishing. - */ - public static void publishData(String streamName, String streamVersion, Map analyticsData) { - - // Analytics data will be added to the OB analytics logfile for processing if ELK is configured for the server. - if (Boolean.parseBoolean((String) OpenBankingConfigParser.getInstance().getConfiguration() - .get(DataPublishingConstants.ELK_ANALYTICS_ENABLED))) { - try { - AnalyticsLogsUtils.addAnalyticsLogs(DataPublishingConstants.LOG_FILE_NAME, streamName, - streamVersion, analyticsData); - } catch (OpenBankingException e) { - log.error("Error occurred while writing analytics logs", e); - } - } - - if (Boolean.parseBoolean((String) OBAnalyticsDataHolder.getInstance().getConfigurationMap() - .get(DataPublishingConstants.DATA_PUBLISHING_ENABLED))) { - - EventQueue eventQueue = OBAnalyticsDataHolder.getInstance().getEventQueue(); - if (!(eventQueue == null)) { - OBAnalyticsEvent event = new OBAnalyticsEvent(streamName, streamVersion, analyticsData); - eventQueue.put(event); - } else { - log.error("Unable to get the event queue. Data publishing may be disabled."); - } - } else { - log.debug("Data publishing is disabled. Failed to obtain a data publisher instance."); - } - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/resources/findbugs-exclude.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/resources/findbugs-exclude.xml deleted file mode 100644 index c96aff43..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/resources/findbugs-exclude.xml +++ /dev/null @@ -1,28 +0,0 @@ - - - - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/resources/findbugs-include.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/resources/findbugs-include.xml deleted file mode 100644 index 8932a22e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/test/java/com/wso2/openbanking/accelerator/data/publisher/common/DataPublisherPoolTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/test/java/com/wso2/openbanking/accelerator/data/publisher/common/DataPublisherPoolTest.java deleted file mode 100644 index e689f948..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/test/java/com/wso2/openbanking/accelerator/data/publisher/common/DataPublisherPoolTest.java +++ /dev/null @@ -1,87 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.data.publisher.common; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.data.publisher.common.internal.OBAnalyticsDataHolder; -import com.wso2.openbanking.accelerator.data.publisher.common.util.OBDataPublisherUtil; -import org.mockito.Mockito; -import org.testng.Assert; -import org.testng.annotations.Test; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -/** - * Data publisher pool test. - */ -public class DataPublisherPoolTest { - - private DataPublisherPool pool; - - @Test(priority = 1) - public void testInitializePool() { - - Map configs = new HashMap<>(); - configs.put("DataPublishing.PoolSize", "3"); - configs.put("DataPublishing.PoolWaitTimeMs", "500"); - configs.put("DataPublishing.Enabled", "true"); - OpenBankingConfigurationService openBankingConfigurationService = - Mockito.mock(OpenBankingConfigurationService.class); - Mockito.when(openBankingConfigurationService.getConfigurations()).thenReturn(configs); - OBAnalyticsDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - - OBAnalyticsDataHolder.getInstance().initializePool(); - pool = OBAnalyticsDataHolder.getInstance().getDataPublisherPool(); - Assert.assertEquals(pool.getCreatedCount(), 0); - Assert.assertEquals(pool.getMaxTotal(), 3); - Assert.assertEquals(pool.getMaxIdle(), 3); - Assert.assertEquals(pool.getBorrowedCount(), 0); - } - - @Test(priority = 2) - public void testBorrowInstances() { - - OpenBankingDataPublisher instance = OBDataPublisherUtil.getDataPublisherInstance(); - Assert.assertEquals(pool.getCreatedCount(), 1); - Assert.assertEquals(pool.getBorrowedCount(), 1); - Assert.assertEquals(pool.getNumIdle(), 0); - OBDataPublisherUtil.releaseDataPublishingInstance(instance); - Assert.assertEquals(pool.getNumIdle(), 1); - } - - @Test(priority = 3) - public void tryBorrowOverMaxLimit() throws InterruptedException { - - List instances = new ArrayList<>(); - for (int i = 0; i < 3; i++) { - instances.add(OBDataPublisherUtil.getDataPublisherInstance()); - } - Assert.assertEquals(pool.getCreatedCount(), 3); - Assert.assertEquals(pool.getBorrowedCount(), 4); - } - - @Test(priority = 100) - public void testPoolClose() { - OBAnalyticsDataHolder.getInstance().closePool(); - Assert.assertTrue(pool.isClosed()); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/test/java/com/wso2/openbanking/accelerator/data/publisher/common/OBAnalyticsEventQueueTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/test/java/com/wso2/openbanking/accelerator/data/publisher/common/OBAnalyticsEventQueueTest.java deleted file mode 100644 index 522ca4ef..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/test/java/com/wso2/openbanking/accelerator/data/publisher/common/OBAnalyticsEventQueueTest.java +++ /dev/null @@ -1,127 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.data.publisher.common; - -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.databind.ObjectMapper; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingRuntimeException; -import com.wso2.openbanking.accelerator.data.publisher.common.internal.OBAnalyticsDataHolder; -import com.wso2.openbanking.accelerator.data.publisher.common.util.OBDataPublisherUtil; -import org.apache.logging.log4j.LogManager; -import org.apache.logging.log4j.Logger; -import org.mockito.Mock; -import org.mockito.Mockito; -import org.mockito.MockitoAnnotations; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -import java.io.ByteArrayOutputStream; -import java.io.PrintStream; -import java.util.HashMap; -import java.util.Map; - -/** - * Open Banking analytics event queue test. - */ -@PowerMockIgnore({"jdk.internal.reflect.*"}) -@PrepareForTest({OpenBankingConfigParser.class}) -public class OBAnalyticsEventQueueTest extends PowerMockTestCase { - - @Mock - OpenBankingConfigParser openBankingConfigParser; - - private static ByteArrayOutputStream outContent; - private static Logger logger = null; - private static PrintStream printStream; - - @BeforeClass - public void beforeTests() { - - MockitoAnnotations.initMocks(this); - outContent = new ByteArrayOutputStream(); - printStream = new PrintStream(outContent); - System.setOut(printStream); - logger = LogManager.getLogger(OBAnalyticsEventQueueTest.class); - } - - @Test - public void testAddingDataToQueue() { - - outContent.reset(); - Map configs = new HashMap<>(); - configs.put("DataPublishing.WorkerThreadCount", "3"); - configs.put("DataPublishing.QueueSize", "10"); - configs.put("DataPublishing.Enabled", "true"); - configs.put("ELKAnalytics.Enabled", "true"); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - Mockito.when(OpenBankingConfigParser.getInstance()) - .thenReturn(openBankingConfigParser); - Mockito.when(openBankingConfigParser.getConfiguration()).thenReturn(configs); - - OpenBankingConfigurationService openBankingConfigurationService = - Mockito.mock(OpenBankingConfigurationService.class); - Mockito.when(openBankingConfigurationService.getConfigurations()).thenReturn(configs); - OBAnalyticsDataHolder.getInstance().setEventQueue(Mockito.mock(EventQueue.class)); - OBAnalyticsDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - - OBDataPublisherUtil.publishData("testStream", "1.0", configs); - try { - Assert.assertTrue(outContent.toString().contains("Data Stream : testStream , Data Stream Version : 1.0 , " + - "Data : {\"payload\":" + new ObjectMapper().writeValueAsString(configs) + "}")); - Assert.assertFalse(outContent.toString().contains("Data publishing is disabled. " + - "Failed to obtain a data publisher instance.")); - } catch (JsonProcessingException e) { - throw new OpenBankingRuntimeException("Error in processing JSON payload", e); - } - } - - @Test - public void tryAddingToQueueWhenDataPublishingDisabled() { - - outContent.reset(); - Map configs = new HashMap<>(); - configs.put("DataPublishing.WorkerThreadCount", "3"); - configs.put("DataPublishing.QueueSize", "10"); - configs.put("DataPublishing.Enabled", "false"); - configs.put("ELKAnalytics.Enabled", "true"); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - Mockito.when(OpenBankingConfigParser.getInstance()) - .thenReturn(openBankingConfigParser); - Mockito.when(openBankingConfigParser.getConfiguration()).thenReturn(configs); - - OpenBankingConfigurationService openBankingConfigurationService = - Mockito.mock(OpenBankingConfigurationService.class); - Mockito.when(openBankingConfigurationService.getConfigurations()).thenReturn(configs); - OBAnalyticsDataHolder.getInstance().setEventQueue(Mockito.mock(EventQueue.class)); - OBAnalyticsDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - - OBDataPublisherUtil.publishData("testStream", "1.0", configs); - Assert.assertTrue(outContent.toString().contains("Data publishing is disabled. " + - "Failed to obtain a data publisher instance.")); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/test/java/com/wso2/openbanking/accelerator/data/publisher/common/OBThriftDataPublisherTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/test/java/com/wso2/openbanking/accelerator/data/publisher/common/OBThriftDataPublisherTest.java deleted file mode 100644 index 61d5fdad..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/test/java/com/wso2/openbanking/accelerator/data/publisher/common/OBThriftDataPublisherTest.java +++ /dev/null @@ -1,387 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.data.publisher.common; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.data.publisher.common.internal.OBAnalyticsDataHolder; -import org.apache.logging.log4j.LogManager; -import org.apache.logging.log4j.Logger; -import org.mockito.Mockito; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; -import org.wso2.carbon.databridge.agent.DataPublisher; -import org.wso2.carbon.databridge.agent.exception.DataEndpointAgentConfigurationException; -import org.wso2.carbon.databridge.agent.exception.DataEndpointAuthenticationException; -import org.wso2.carbon.databridge.agent.exception.DataEndpointConfigurationException; -import org.wso2.carbon.databridge.agent.exception.DataEndpointException; -import org.wso2.carbon.databridge.commons.exception.TransportException; - -import java.io.ByteArrayOutputStream; -import java.io.PrintStream; -import java.util.AbstractMap; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.stream.Collectors; -import java.util.stream.Stream; - -/** - * Test for Open Banking thrift data publisher. - */ -public class OBThriftDataPublisherTest { - - public static final Map ATTRIBUTE_MAP = Stream.of( - new AbstractMap.SimpleImmutableEntry<>(2, "SampleStringAttribute"), - new AbstractMap.SimpleImmutableEntry<>(3, "SampleIntAttribute"), - new AbstractMap.SimpleImmutableEntry<>(1, "SampleBooleanAttribute")) - .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue)); - public static final Map ATTRIBUTE_MAP2 = Stream.of( - new AbstractMap.SimpleImmutableEntry<>(2, "SampleFloatAttribute"), - new AbstractMap.SimpleImmutableEntry<>(3, "SampleLongAttribute"), - new AbstractMap.SimpleImmutableEntry<>(1, "SampleDoubleAttribute")) - .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue)); - - private static final Map> STREAM_ATTRIBUTE_MAP = Stream.of( - new AbstractMap.SimpleImmutableEntry<>("testStream", ATTRIBUTE_MAP), - new AbstractMap.SimpleImmutableEntry<>("testStream2", ATTRIBUTE_MAP2)) - .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue)); - OBThriftDataPublisher thriftDataPublisher; - private static ByteArrayOutputStream outContent; - private static Logger logger = null; - private static PrintStream printStream; - private float floatNum = 4f; - private long longNum = 2L; - private double doubleNum = 5.5; - - @BeforeClass - public void initializeConfigurations() { - - OpenBankingConfigurationService openBankingConfigurationService = - Mockito.mock(OpenBankingConfigurationService.class); - Mockito.when(openBankingConfigurationService.getDataPublishingStreams()).thenReturn(STREAM_ATTRIBUTE_MAP); - Map configs = new HashMap<>(); - configs.put("DataPublishing.Username", "admin"); - configs.put("DataPublishing.Password", "admin"); - configs.put("DataPublishing.ServerURL", "{tcp://localhost:7612}"); - configs.put("DataPublishing.Thrift.PublishingTimeout", "2000"); - Mockito.when(openBankingConfigurationService.getConfigurations()).thenReturn(configs); - OBAnalyticsDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - OBThriftDataPublisherTest.outContent = new ByteArrayOutputStream(); - OBThriftDataPublisherTest.printStream = new PrintStream(OBThriftDataPublisherTest.outContent); - System.setOut(OBThriftDataPublisherTest.printStream); - OBThriftDataPublisherTest.logger = LogManager.getLogger(OBThriftDataPublisherTest.class); - } - - @Test - public void init() { - - OBThriftDataPublisher thriftDataPublisher = new MockedOBThriftDataPublisher(); - DataPublisher dataPublisher = Mockito.mock(DataPublisher.class); - thriftDataPublisher.setDataPublisher(dataPublisher); - thriftDataPublisher.init(); - } - - @Test - public void tryInitWithoutRequiredConfigs() - throws DataEndpointAuthenticationException, DataEndpointAgentConfigurationException, DataEndpointException, - DataEndpointConfigurationException, TransportException { - outContent.reset(); - OpenBankingConfigurationService openBankingConfigurationService = - Mockito.mock(OpenBankingConfigurationService.class); - Mockito.when(openBankingConfigurationService.getDataPublishingStreams()).thenReturn(STREAM_ATTRIBUTE_MAP); - Map configs = new HashMap<>(); - configs.put("DataPublishing.Thrift.PublishingTimeout", "2000"); - Mockito.when(openBankingConfigurationService.getConfigurations()).thenReturn(configs); - OBAnalyticsDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - OBThriftDataPublisher thriftDataPublisher = Mockito.spy(OBThriftDataPublisher.class); - Mockito.doReturn(Mockito.mock(DataPublisher.class)).when(thriftDataPublisher) - .getDataPublisher(Mockito.anyString(), Mockito.anyString(), Mockito.anyString(), Mockito.anyString()); - thriftDataPublisher.init(); - Assert.assertTrue(OBThriftDataPublisherTest.outContent.toString().contains("ERROR : Error while retrieving " + - "publisher server configs")); - } - - @Test(priority = 1) - public void testStreamAttributeMapCreation() { - - thriftDataPublisher = new MockedOBThriftDataPublisher(); - thriftDataPublisher.buildStreamAttributeMap(); - - List attributes = new ArrayList<>(); - attributes.add("SampleBooleanAttribute"); - attributes.add("SampleStringAttribute"); - attributes.add("SampleIntAttribute"); - List attributesSet2 = new ArrayList<>(); - attributesSet2.add("SampleDoubleAttribute"); - attributesSet2.add("SampleFloatAttribute"); - attributesSet2.add("SampleLongAttribute"); - Map> expectedMap = new HashMap<>(); - expectedMap.put("testStream", attributes); - expectedMap.put("testStream2", attributesSet2); - Assert.assertEquals(thriftDataPublisher.getStreamAttributeMap(), expectedMap); - - } - - @Test(priority = 2) - public void setPayload() { - - String streamName = "testStream"; - Map data = new HashMap<>(); - data.put("SampleStringAttribute", "StringValue1"); - data.put("SampleIntAttribute", 2); - data.put("SampleBooleanAttribute", true); - - Object[] result = thriftDataPublisher.setPayload(streamName, data); - Object[] expectedOutput = new Object[]{ - true, "StringValue1", 2 - }; - - Assert.assertEquals(result, expectedOutput); - } - - @Test(priority = 2) - public void setPayloadWithoutRequiredAttributes() { - - outContent.reset(); - String streamName = "testStream"; - Map data = new HashMap<>(); - data.put("SampleStringAttribute", "StringValue1"); - data.put("SampleIntAttribute", 2); - - Object[] result = thriftDataPublisher.setPayload(streamName, data); - Object[] expectedOutput = new Object[]{}; - - Assert.assertEquals(result, expectedOutput); - Assert.assertTrue(OBThriftDataPublisherTest.outContent.toString().contains("is missing in data map for ")); - } - - @Test(priority = 2) - public void setPayloadWithoutRequiredAttributes2() { - - outContent.reset(); - String streamName = "testStream"; - Map data = new HashMap<>(); - data.put("SampleStringAttribute", "StringValue1"); - data.put("SampleIntAttribute", 2); - data.put("SampleBooleanAttribute", null); - - Object[] result = thriftDataPublisher.setPayload(streamName, data); - Object[] expectedOutput = new Object[]{}; - - Assert.assertEquals(result, expectedOutput); - Assert.assertTrue(OBThriftDataPublisherTest.outContent.toString().contains("is missing in data map for ")); - } - - @Test(priority = 2) - public void setPayloadWithInvalidBooleanData() { - - outContent.reset(); - String streamName = "testStream"; - Map data = new HashMap<>(); - data.put("SampleStringAttribute", "StringValue1"); - data.put("SampleIntAttribute", 2); - data.put("SampleBooleanAttribute", "true"); - - Object[] result = thriftDataPublisher.setPayload(streamName, data); - Object[] expectedOutput = new Object[]{}; - - Assert.assertEquals(result, expectedOutput); - Assert.assertTrue(OBThriftDataPublisherTest.outContent.toString().contains(" is expecting a " + - Boolean.class.getName() + " type attribute while attribute of type ")); - } - - @Test(priority = 2) - public void setPayloadWithInvalidIntegerData() { - - outContent.reset(); - String streamName = "testStream"; - Map data = new HashMap<>(); - data.put("SampleStringAttribute", "StringValue1"); - data.put("SampleIntAttribute", 2.14); - data.put("SampleBooleanAttribute", true); - - Object[] result = thriftDataPublisher.setPayload(streamName, data); - Object[] expectedOutput = new Object[]{}; - - Assert.assertEquals(result, expectedOutput); - Assert.assertTrue(OBThriftDataPublisherTest.outContent.toString().contains(" is expecting a " + - Integer.class.getName() + " type attribute while attribute of type ")); - } - - @Test(priority = 2) - public void setPayloadWithInvalidStringData() { - - outContent.reset(); - String streamName = "testStream"; - Map data = new HashMap<>(); - data.put("SampleStringAttribute", 1); - data.put("SampleIntAttribute", 2); - data.put("SampleBooleanAttribute", true); - - Object[] result = thriftDataPublisher.setPayload(streamName, data); - Object[] expectedOutput = new Object[]{}; - - Assert.assertEquals(result, expectedOutput); - Assert.assertTrue(OBThriftDataPublisherTest.outContent.toString().contains(" is expecting a " + - String.class.getName() + " type attribute while attribute of type ")); - } - - @Test(priority = 2) - public void setPayloadWithInvalidFloatData() { - - outContent.reset(); - String streamName = "testStream2"; - Map data = new HashMap<>(); - data.put("SampleFloatAttribute", 4); - data.put("SampleLongAttribute", longNum); - data.put("SampleDoubleAttribute", doubleNum); - - - Object[] result = thriftDataPublisher.setPayload(streamName, data); - Object[] expectedOutput = new Object[]{}; - - Assert.assertEquals(result, expectedOutput); - Assert.assertTrue(OBThriftDataPublisherTest.outContent.toString().contains(" is expecting a " + - Float.class.getName() + " type attribute while attribute of type ")); - } - - @Test(priority = 2) - public void setPayloadWithInvalidLongData() { - - outContent.reset(); - String streamName = "testStream2"; - Map data = new HashMap<>(); - data.put("SampleFloatAttribute", floatNum); - data.put("SampleLongAttribute", 2.2); - data.put("SampleDoubleAttribute", doubleNum); - - Object[] result = thriftDataPublisher.setPayload(streamName, data); - Object[] expectedOutput = new Object[]{}; - - Assert.assertEquals(result, expectedOutput); - Assert.assertTrue(OBThriftDataPublisherTest.outContent.toString().contains(" is expecting a " + - Long.class.getName() + " type attribute while attribute of type ")); - } - - @Test(priority = 2) - public void setPayloadWithInvalidDoubleData() { - - outContent.reset(); - String streamName = "testStream2"; - Map data = new HashMap<>(); - data.put("SampleFloatAttribute", floatNum); - data.put("SampleLongAttribute", longNum); - data.put("SampleDoubleAttribute", 5); - - Object[] result = thriftDataPublisher.setPayload(streamName, data); - Object[] expectedOutput = new Object[]{}; - - Assert.assertEquals(result, expectedOutput); - Assert.assertTrue(OBThriftDataPublisherTest.outContent.toString().contains(" is expecting a " + - Double.class.getName() + " type attribute while attribute of type ")); - } - - @Test - public void publish() { - - outContent.reset(); - OBThriftDataPublisher thriftDataPublisher = new MockedOBThriftDataPublisher(); - DataPublisher dataPublisher = Mockito.mock(DataPublisher.class); - thriftDataPublisher.setDataPublisher(dataPublisher); - Mockito.doReturn(true).when(dataPublisher).tryPublish(Mockito.any(), Mockito.anyLong()); - Map data = new HashMap<>(); - data.put("SampleStringAttribute", "StringValue1"); - data.put("SampleIntAttribute", 2); - data.put("SampleBooleanAttribute", true); - thriftDataPublisher.publish("testStream", "1.0", data); - Assert.assertFalse(OBThriftDataPublisherTest.outContent.toString().contains("ERROR")); - } - - @Test - public void tryPublishWhenAttributesNotDefined() { - - outContent.reset(); - OpenBankingConfigurationService openBankingConfigurationService = - Mockito.mock(OpenBankingConfigurationService.class); - Mockito.when(openBankingConfigurationService.getDataPublishingStreams()).thenReturn(STREAM_ATTRIBUTE_MAP); - Map configs = new HashMap<>(); - configs.put("DataPublishing.Username", "admin"); - configs.put("DataPublishing.Password", "admin"); - configs.put("DataPublishing.ServerURL", "{tcp://localhost:7612}"); - configs.put("DataPublishing.Thrift.PublishingTimeout", "2000"); - Mockito.when(openBankingConfigurationService.getConfigurations()).thenReturn(configs); - OBAnalyticsDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - OBThriftDataPublisher thriftDataPublisher = new MockedOBThriftDataPublisher(); - DataPublisher dataPublisher = Mockito.mock(DataPublisher.class); - thriftDataPublisher.setDataPublisher(dataPublisher); - Mockito.doReturn(true).when(dataPublisher).tryPublish(Mockito.any(), Mockito.anyLong()); - Map data = new HashMap<>(); - thriftDataPublisher.publish("testStream2", "1.0", data); - Assert.assertTrue(OBThriftDataPublisherTest.outContent.toString().contains("ERROR : Error while setting " + - "payload to publish data.")); - } - - private class MockedOBThriftDataPublisher extends OBThriftDataPublisher { - - private Map> validationMap; - - public MockedOBThriftDataPublisher() { - validationMap = new HashMap<>(); - Map metadata1 = new HashMap<>(); - Map metadata2 = new HashMap<>(); - Map metadata3 = new HashMap<>(); - Map metadata4 = new HashMap<>(); - Map metadata5 = new HashMap<>(); - Map metadata6 = new HashMap<>(); - metadata1.put("required", true); - metadata1.put("type", "int"); - validationMap.put("testStream_SampleIntAttribute", metadata1); - metadata2.put("required", true); - metadata2.put("type", "boolean"); - validationMap.put("testStream_SampleBooleanAttribute", metadata2); - metadata3.put("required", false); - metadata3.put("type", "string"); - validationMap.put("testStream_SampleStringAttribute", metadata3); - metadata4.put("required", true); - metadata4.put("type", "float"); - validationMap.put("testStream2_SampleFloatAttribute", metadata4); - metadata5.put("required", true); - metadata5.put("type", "long"); - validationMap.put("testStream2_SampleLongAttribute", metadata5); - metadata6.put("required", false); - metadata6.put("type", "double"); - validationMap.put("testStream2_SampleDoubleAttribute", metadata6); - } - - @Override - protected DataPublisher getDataPublisher(String serverURL, String authURLSet, String serverUser, - String serverPassword) { - - return Mockito.mock(DataPublisher.class); - } - - @Override - protected Map> getAttributeValidationMap() { - - return validationMap; - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/test/resources/testng.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/test/resources/testng.xml deleted file mode 100644 index 732af2c3..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/com.wso2.openbanking.accelerator.data.publisher.common/src/test/resources/testng.xml +++ /dev/null @@ -1,29 +0,0 @@ - - - - - - - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/pom.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/pom.xml deleted file mode 100644 index ec348abf..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.data.publisher/pom.xml +++ /dev/null @@ -1,36 +0,0 @@ - - - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../pom.xml - - 4.0.0 - - com.wso2.openbanking.accelerator.data.publisher - WSO2 Open Banking - Data Publisher - pom - - - com.wso2.openbanking.accelerator.data.publisher.common - com.wso2.openbanking.accelerator.authentication.data.publisher - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/pom.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/pom.xml deleted file mode 100644 index c3456744..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/pom.xml +++ /dev/null @@ -1,301 +0,0 @@ - - - - - 4.0.0 - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../pom.xml - - - com.wso2.openbanking.accelerator.gateway - bundle - WSO2 Open Banking - Gateway component - - - org.wso2.eclipse.osgi - org.eclipse.osgi.services - - - org.eclipse.osgi - org.eclipse.osgi - - - commons-logging - commons-logging - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - - - org.apache.synapse - synapse-core - - - io.swagger.core.v3 - swagger-models - - - io.swagger.parser.v3 - swagger-parser - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.data.publisher.common - - - org.wso2.orbit.com.hazelcast - hazelcast - - - - - org.wso2.orbit.org.bouncycastle - bcprov-jdk18on - - - org.testng - testng - test - - - org.powermock - powermock-module-testng - test - - - org.powermock - powermock-api-mockito - test - - - org.mockito - mockito-all - test - - - org.wso2.carbon.apimgt - org.wso2.carbon.apimgt.common.gateway - - - org.wso2.orbit.com.hazelcast - hazelcast - - - - - org.wso2.carbon.apimgt - org.wso2.carbon.apimgt.keymgt - - - org.wso2.orbit.com.hazelcast - hazelcast - - - - - io.jsonwebtoken - jjwt - - - org.wso2.carbon - org.wso2.carbon.core - - - org.wso2.orbit.com.hazelcast - hazelcast - - - - - com.fasterxml.jackson.core - jackson-core - - - com.fasterxml.jackson.core - jackson-annotations - - - org.wso2.am.analytics.publisher - org.wso2.am.analytics.publisher.client - - - org.wso2.orbit.com.hazelcast - hazelcast - - - - - org.wso2.orbit.com.nimbusds - nimbus-jose-jwt - - - - - - org.apache.felix - maven-bundle-plugin - true - - - - ${project.artifactId} - - - com.wso2.openbanking.accelerator.gateway.internal - - - org.osgi.framework; version="${osgi.framework.imp.pkg.version.range}", - org.osgi.service.component; version="${osgi.service.component.imp.pkg.version.range}", - org.wso2.am.analytics.publisher.*; version="${analytics.publisher.version}", - - com.nimbusds.jwt.*;version="${nimbusds.osgi.version.range}", - com.nimbusds.jose.*;version="${nimbusds.osgi.version.range}", - - - !com.wso2.openbanking.accelerator.gateway.internal, - com.wso2.openbanking.accelerator.gateway.*; - version="${project.version}", - com.wso2.openbanking.accelerator.gateway.executor.*; - version="${project.version}", - com.wso2.openbanking.accelerator.gateway.reporter.*; - version="${project.version}", - - * - <_dsannotations>* - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - false - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-exclude.xml - ${project.basedir}/src/main/resources/findbugs-include.xml - false - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - org.jacoco - jacoco-maven-plugin - - - - **/*Constants.class - **/*Component.class - **/*APIRequestContext.class - **/*APIResponseContext.class - **/*GatewayCache.class - **/*CertificateRevocationCache.class - **/*TppValidationCache.class - **/*GatewayCacheKey.class - **/*OpenBankingExecutorError.class - **/*DataHolder.class - **/*Exception.class - - - - - default-prepare-agent - - prepare-agent - - - - default-prepare-agent-integration - - prepare-agent-integration - - - - default-report - - report - - - - default-report-integration - - report-integration - - - - default-check - - check - - - - - BUNDLE - - - INSTRUCTION - COVEREDRATIO - 0.73 - - - - - - - - - - org.apache.maven.plugins - maven-surefire-plugin - - - src/test/resources/testng.xml - - - org.wso2.orbit.com.fasterxml.jackson.core:jackson-core - org.wso2.orbit.com.hazelcast:hazelcast:3.12.2.wso2v1 - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/cache/CertificateRevocationCache.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/cache/CertificateRevocationCache.java deleted file mode 100644 index a1a41640..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/cache/CertificateRevocationCache.java +++ /dev/null @@ -1,78 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.cache; - -import com.wso2.openbanking.accelerator.common.caching.OpenBankingBaseCache; -import com.wso2.openbanking.accelerator.gateway.internal.TPPCertValidatorDataHolder; - -/** - * Cache definition to store API Resource Security Schemes. - */ -public class CertificateRevocationCache extends OpenBankingBaseCache { - - private static final String CACHE_NAME = "OPEN_BANKING_CLIENT_CERTIFICATE_CACHE"; - - private static CertificateRevocationCache clientCertificateCache; - private final Integer accessExpiryMinutes; - private final Integer modifiedExpiryMinutes; - - /** - * Initialize with unique cache name. - */ - private CertificateRevocationCache() { - - super(CACHE_NAME); - this.accessExpiryMinutes = setAccessExpiryMinutes(); - this.modifiedExpiryMinutes = setModifiedExpiryMinutes(); - } - - /** - * Singleton getInstance method to create only one object. - * - * @return TPPValidationCache object - */ - public static synchronized CertificateRevocationCache getInstance() { - if (clientCertificateCache == null) { - clientCertificateCache = new CertificateRevocationCache(); - } - return clientCertificateCache; - } - - @Override - public int getCacheAccessExpiryMinutes() { - - return accessExpiryMinutes; - } - - @Override - public int getCacheModifiedExpiryMinutes() { - - return modifiedExpiryMinutes; - } - - public int setAccessExpiryMinutes() { - - return TPPCertValidatorDataHolder.getInstance().getTppCertRevocationCacheExpiry(); - } - - public int setModifiedExpiryMinutes() { - - return TPPCertValidatorDataHolder.getInstance().getTppCertRevocationCacheExpiry(); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/cache/GatewayCache.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/cache/GatewayCache.java deleted file mode 100644 index d716c8a7..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/cache/GatewayCache.java +++ /dev/null @@ -1,65 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.cache; - -import com.wso2.openbanking.accelerator.common.caching.OpenBankingBaseCache; -import com.wso2.openbanking.accelerator.gateway.internal.GatewayDataHolder; - -/** - * Cache definition to store API Resource Security Schemes. - */ -public class GatewayCache extends OpenBankingBaseCache { - - private static final String cacheName = "OPEN_BANKING_GATEWAY_CACHE"; - - private Integer accessExpiryMinutes; - private Integer modifiedExpiryMinutes; - - /** - * Initialize with unique cache name. - */ - public GatewayCache() { - - super(cacheName); - this.accessExpiryMinutes = setAccessExpiryMinutes(); - this.modifiedExpiryMinutes = setModifiedExpiryMinutes(); - } - - @Override - public int getCacheAccessExpiryMinutes() { - - return accessExpiryMinutes; - } - - @Override - public int getCacheModifiedExpiryMinutes() { - - return modifiedExpiryMinutes; - } - - public int setAccessExpiryMinutes() { - - return GatewayDataHolder.getInstance().getGatewayCacheAccessExpiry(); - } - - public int setModifiedExpiryMinutes() { - - return GatewayDataHolder.getInstance().getGatewayCacheModifiedExpiry(); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/cache/GatewayCacheKey.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/cache/GatewayCacheKey.java deleted file mode 100644 index 3077c60a..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/cache/GatewayCacheKey.java +++ /dev/null @@ -1,62 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.cache; - -import com.wso2.openbanking.accelerator.common.caching.OpenBankingBaseCacheKey; - -import java.io.Serializable; -import java.util.Objects; - -/** - * Cache Key for Open Banking Gateway cache. - */ -public class GatewayCacheKey extends OpenBankingBaseCacheKey implements Serializable { - - private static final long serialVersionUID = 883027070771592120L; - public String gatewayCacheKey; - - public GatewayCacheKey(String gatewayCacheKey) { - - this.gatewayCacheKey = gatewayCacheKey; - } - - public static GatewayCacheKey of(String gatewayCacheKey) { - - return new GatewayCacheKey(gatewayCacheKey); - } - - @Override - public boolean equals(Object o) { - - if (this == o) { - return true; - } - if (o == null || getClass() != o.getClass()) { - return false; - } - GatewayCacheKey that = (GatewayCacheKey) o; - return Objects.equals(gatewayCacheKey, that.gatewayCacheKey); - } - - @Override - public int hashCode() { - - return Objects.hash(gatewayCacheKey); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/cache/TppValidationCache.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/cache/TppValidationCache.java deleted file mode 100644 index 610a6528..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/cache/TppValidationCache.java +++ /dev/null @@ -1,78 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.cache; - -import com.wso2.openbanking.accelerator.common.caching.OpenBankingBaseCache; -import com.wso2.openbanking.accelerator.gateway.internal.TPPCertValidatorDataHolder; - -/** - * Cache definition to store API Resource Security Schemes. - */ -public class TppValidationCache extends OpenBankingBaseCache { - - private static final String CACHE_NAME = "OPEN_BANKING_TPP_VALIDATION_CACHE"; - - private static TppValidationCache tppValidationCache; - private final Integer accessExpiryMinutes; - private final Integer modifiedExpiryMinutes; - - /** - * Initialize with unique cache name. - */ - private TppValidationCache() { - - super(CACHE_NAME); - this.accessExpiryMinutes = setAccessExpiryMinutes(); - this.modifiedExpiryMinutes = setModifiedExpiryMinutes(); - } - - /** - * Singleton getInstance method to create only one object. - * - * @return TPPValidationCache object - */ - public static synchronized TppValidationCache getInstance() { - if (tppValidationCache == null) { - tppValidationCache = new TppValidationCache(); - } - return tppValidationCache; - } - - @Override - public int getCacheAccessExpiryMinutes() { - - return accessExpiryMinutes; - } - - @Override - public int getCacheModifiedExpiryMinutes() { - - return modifiedExpiryMinutes; - } - - public int setAccessExpiryMinutes() { - - return TPPCertValidatorDataHolder.getInstance().getTppValidationCacheExpiry(); - } - - public int setModifiedExpiryMinutes() { - - return TPPCertValidatorDataHolder.getInstance().getTppValidationCacheExpiry(); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/core/AbstractRequestRouter.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/core/AbstractRequestRouter.java deleted file mode 100644 index 9c4a35bb..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/core/AbstractRequestRouter.java +++ /dev/null @@ -1,84 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.core; - -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.common.util.OpenBankingUtils; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.internal.GatewayDataHolder; - -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.stream.Collectors; - -/** - * Open Banking abstract Request Router. - */ -public abstract class AbstractRequestRouter { - - private Map> executorMap = new HashMap<>(); - - /** - * Initiation method of the Router. - */ - @Generated(message = "Ignoring since the method require OSGi services to function. This functionality is tested " + - "in other services") - public void build() { - - Map> executorConfig = - GatewayDataHolder.getInstance().getOpenBankingConfigurationService().getExecutors(); - executorConfig.keySet().forEach(consentType -> { - Map integerStringMap = executorConfig.get(consentType); - List executorList = integerStringMap.keySet().stream() - .map(integer -> (OpenBankingGatewayExecutor) OpenBankingUtils - .getClassInstanceFromFQN(integerStringMap.get(integer))).collect(Collectors.toList()); - executorMap.put(consentType, executorList); - }); - } - - /** - * Method to obtain correct executors for the given request context. ( Expected to be implemented at toolkit) - * - * @param requestContext OB Request context - * @return List of executors - */ - public abstract List getExecutorsForRequest(OBAPIRequestContext requestContext); - - /** - * Method to obtain correct executors for the given response context. ( Expected to be implemented at toolkit) - * - * @param requestContext OB Response context - * @return List of executors - */ - public abstract List getExecutorsForResponse(OBAPIResponseContext requestContext); - - public Map> getExecutorMap() { - - return executorMap; - } - - public void setExecutorMap( - Map> executorMap) { - - this.executorMap = executorMap; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/core/DefaultRequestRouter.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/core/DefaultRequestRouter.java deleted file mode 100644 index 6e52ee52..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/core/DefaultRequestRouter.java +++ /dev/null @@ -1,72 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.core; - -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; - -import java.util.ArrayList; -import java.util.List; - -/** - * Open Banking Default Request Router. - */ -public class DefaultRequestRouter extends AbstractRequestRouter { - - private static final List EMPTY_LIST = new ArrayList<>(); - - public List getExecutorsForRequest(OBAPIRequestContext requestContext) { - if (GatewayConstants.API_TYPE_NON_REGULATORY - .equals(requestContext.getOpenAPI().getExtensions().get(GatewayConstants.API_TYPE_CUSTOM_PROP))) { - requestContext.addContextProperty(GatewayConstants.API_TYPE_CUSTOM_PROP, - GatewayConstants.API_TYPE_NON_REGULATORY); - return EMPTY_LIST; - } else if (GatewayConstants.API_TYPE_CONSENT - .equals(requestContext.getOpenAPI().getExtensions().get(GatewayConstants.API_TYPE_CUSTOM_PROP))) { - requestContext.addContextProperty(GatewayConstants.API_TYPE_CUSTOM_PROP, - GatewayConstants.API_TYPE_CONSENT); - return this.getExecutorMap().get("Consent"); - } else if (requestContext.getMsgInfo().getResource().contains("/register")) { - return this.getExecutorMap().get("DCR"); - } else { - return this.getExecutorMap().get("Default"); - } - } - - public List getExecutorsForResponse(OBAPIResponseContext responseContext) { - - if (responseContext.getContextProps().containsKey(GatewayConstants.API_TYPE_CUSTOM_PROP)) { - if (GatewayConstants.API_TYPE_NON_REGULATORY - .equals(responseContext.getContextProps().get(GatewayConstants.API_TYPE_CUSTOM_PROP))) { - return EMPTY_LIST; - } else if (GatewayConstants.API_TYPE_CONSENT - .equals(responseContext.getContextProps().get(GatewayConstants.API_TYPE_CUSTOM_PROP))) { - return this.getExecutorMap().get("Consent"); - } - } - - if (responseContext.getMsgInfo().getResource().contains("/register")) { - return this.getExecutorMap().get("DCR"); - } else { - return this.getExecutorMap().get("Default"); - } - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/core/OBExtensionListenerImpl.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/core/OBExtensionListenerImpl.java deleted file mode 100644 index 22ff79a0..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/core/OBExtensionListenerImpl.java +++ /dev/null @@ -1,280 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.core; - -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.data.publisher.common.util.OBDataPublisherUtil; -import com.wso2.openbanking.accelerator.gateway.cache.GatewayCacheKey; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.internal.GatewayDataHolder; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.HttpStatus; -import org.wso2.carbon.apimgt.common.gateway.dto.ExtensionResponseDTO; -import org.wso2.carbon.apimgt.common.gateway.dto.ExtensionResponseStatus; -import org.wso2.carbon.apimgt.common.gateway.dto.RequestContextDTO; -import org.wso2.carbon.apimgt.common.gateway.dto.ResponseContextDTO; -import org.wso2.carbon.apimgt.common.gateway.extensionlistener.ExtensionListener; - -import java.io.ByteArrayInputStream; -import java.nio.charset.StandardCharsets; -import java.util.HashMap; -import java.util.Map; -import java.util.TreeMap; - -/** - * Open Banking implementation for Extension listener. - */ -public class OBExtensionListenerImpl implements ExtensionListener { - - private static final Log log = LogFactory.getLog(OBExtensionListenerImpl.class); - - @Override - @Generated(message = "Ignoring since the method has covered in other tests") - public ExtensionResponseDTO preProcessRequest(RequestContextDTO requestContextDTO) { - - OBAPIRequestContext obapiRequestContext = new OBAPIRequestContext(requestContextDTO, new HashMap<>(), - new HashMap<>()); - for (OpenBankingGatewayExecutor gatewayExecutor : - GatewayDataHolder.getInstance().getRequestRouter().getExecutorsForRequest(obapiRequestContext)) { - gatewayExecutor.preProcessRequest(obapiRequestContext); - } - - if (!obapiRequestContext.isError()) { - setPropertiesToCache(requestContextDTO.getMsgInfo().getMessageId() + - GatewayConstants.CONTEXT_PROP_CACHE_KEY, obapiRequestContext.getContextProps()); - - setPropertiesToCache(requestContextDTO.getMsgInfo().getMessageId() + - GatewayConstants.ANALYTICS_PROP_CACHE_KEY, obapiRequestContext.getAnalyticsData()); - } else { - publishAnalyticsData(obapiRequestContext.getAnalyticsData()); - } - return getResponseDTOForRequest(obapiRequestContext); - } - - @Override - @Generated(message = "Ignoring since the method has covered in other tests") - public ExtensionResponseDTO postProcessRequest(RequestContextDTO requestContextDTO) { - - Map contextProps = getPropertiesFromCache(requestContextDTO.getMsgInfo().getMessageId() + - GatewayConstants.CONTEXT_PROP_CACHE_KEY); - Map analyticsData = getPropertiesFromCache(requestContextDTO.getMsgInfo().getMessageId() + - GatewayConstants.ANALYTICS_PROP_CACHE_KEY); - - OBAPIRequestContext obapiRequestContext = - new OBAPIRequestContext(requestContextDTO, contextProps, analyticsData); - for (OpenBankingGatewayExecutor gatewayExecutor : - GatewayDataHolder.getInstance().getRequestRouter().getExecutorsForRequest(obapiRequestContext)) { - gatewayExecutor.postProcessRequest(obapiRequestContext); - } - - if (!obapiRequestContext.isError()) { - setPropertiesToCache(requestContextDTO.getMsgInfo().getMessageId() + - GatewayConstants.CONTEXT_PROP_CACHE_KEY, obapiRequestContext.getContextProps()); - - setPropertiesToCache(requestContextDTO.getMsgInfo().getMessageId() + - GatewayConstants.ANALYTICS_PROP_CACHE_KEY, obapiRequestContext.getAnalyticsData()); - } else { - publishAnalyticsData(obapiRequestContext.getAnalyticsData()); - } - return getResponseDTOForRequest(obapiRequestContext); - } - - @Override - @Generated(message = "Ignoring since the method has covered in other tests") - public ExtensionResponseDTO preProcessResponse(ResponseContextDTO responseContextDTO) { - - Map contextProps = getPropertiesFromCache(responseContextDTO.getMsgInfo().getMessageId() + - GatewayConstants.CONTEXT_PROP_CACHE_KEY); - Map analyticsData = getPropertiesFromCache(responseContextDTO.getMsgInfo().getMessageId() + - GatewayConstants.ANALYTICS_PROP_CACHE_KEY); - OBAPIResponseContext obapiResponseContext = - new OBAPIResponseContext(responseContextDTO, contextProps, analyticsData); - for (OpenBankingGatewayExecutor gatewayExecutor : - GatewayDataHolder.getInstance().getRequestRouter().getExecutorsForResponse(obapiResponseContext)) { - gatewayExecutor.preProcessResponse(obapiResponseContext); - } - - if (!obapiResponseContext.isError()) { - setPropertiesToCache(responseContextDTO.getMsgInfo().getMessageId() + - GatewayConstants.CONTEXT_PROP_CACHE_KEY, obapiResponseContext.getContextProps()); - - setPropertiesToCache(responseContextDTO.getMsgInfo().getMessageId() + - GatewayConstants.ANALYTICS_PROP_CACHE_KEY, obapiResponseContext.getAnalyticsData()); - } else { - publishAnalyticsData(obapiResponseContext.getAnalyticsData()); - } - return getResponseDTOForResponse(obapiResponseContext); - } - - @Override - @Generated(message = "Ignoring since the method has covered in other tests") - public ExtensionResponseDTO postProcessResponse(ResponseContextDTO responseContextDTO) { - - Map contextProps = getPropertiesFromCache(responseContextDTO.getMsgInfo().getMessageId() + - GatewayConstants.CONTEXT_PROP_CACHE_KEY); - Map analyticsData = getPropertiesFromCache(responseContextDTO.getMsgInfo().getMessageId() + - GatewayConstants.ANALYTICS_PROP_CACHE_KEY); - OBAPIResponseContext obapiResponseContext = - new OBAPIResponseContext(responseContextDTO, contextProps, analyticsData); - for (OpenBankingGatewayExecutor gatewayExecutor : - GatewayDataHolder.getInstance().getRequestRouter().getExecutorsForResponse(obapiResponseContext)) { - gatewayExecutor.postProcessResponse(obapiResponseContext); - } - publishAnalyticsData(obapiResponseContext.getAnalyticsData()); - ExtensionResponseDTO responseDTOForResponse = getResponseDTOForResponse(obapiResponseContext); - removePropertiesFromCache(responseContextDTO.getMsgInfo().getMessageId() + - GatewayConstants.CONTEXT_PROP_CACHE_KEY); - removePropertiesFromCache(responseContextDTO.getMsgInfo().getMessageId() + - GatewayConstants.ANALYTICS_PROP_CACHE_KEY); - return responseDTOForResponse; - } - - protected ExtensionResponseDTO getResponseDTOForRequest(OBAPIRequestContext obapiRequestContext) { - - ExtensionResponseDTO extensionResponseDTO = new ExtensionResponseDTO(); - if (obapiRequestContext.isError()) { - - int statusCode = (!obapiRequestContext.getContextProps().containsKey(GatewayConstants.ERROR_STATUS_PROP)) ? - HttpStatus.SC_INTERNAL_SERVER_ERROR : - Integer.parseInt(obapiRequestContext.getContextProperty(GatewayConstants.ERROR_STATUS_PROP)); - extensionResponseDTO.setStatusCode(statusCode); - extensionResponseDTO.setResponseStatus(ExtensionResponseStatus.RETURN_ERROR.toString()); - } else if (obapiRequestContext.getContextProps().containsKey(GatewayConstants.IS_RETURN_RESPONSE) && - Boolean.parseBoolean(obapiRequestContext.getContextProps().get(GatewayConstants.IS_RETURN_RESPONSE))) { - Map headers = obapiRequestContext.getMsgInfo().getHeaders(); - headers.put(GatewayConstants.CONTENT_TYPE_TAG, GatewayConstants.JSON_CONTENT_TYPE); - obapiRequestContext.getMsgInfo().setHeaders(headers); - extensionResponseDTO.setHeaders(headers); - if (obapiRequestContext.getContextProps().containsKey(GatewayConstants.MODIFIED_STATUS)) { - extensionResponseDTO.setStatusCode(Integer.parseInt(obapiRequestContext.getContextProps() - .get(GatewayConstants.MODIFIED_STATUS))); - } - extensionResponseDTO.setResponseStatus(ExtensionResponseStatus.RETURN_ERROR.toString()); - } else { - extensionResponseDTO.setResponseStatus(ExtensionResponseStatus.CONTINUE.toString()); - } - - String modifiedPayload = obapiRequestContext.getModifiedPayload(); - if (modifiedPayload != null) { - extensionResponseDTO.setPayload(new ByteArrayInputStream(modifiedPayload.getBytes(StandardCharsets.UTF_8))); - } - Map addedHeaders = obapiRequestContext.getAddedHeaders(); - if (addedHeaders.size() != 0) { - TreeMap headers = new TreeMap<>(); - headers.putAll(obapiRequestContext.getMsgInfo().getHeaders()); - for (Map.Entry headerEntry : addedHeaders.entrySet()) { - headers.put(headerEntry.getKey(), headerEntry.getValue()); - } - extensionResponseDTO.setHeaders(headers); - } - return extensionResponseDTO; - } - - protected ExtensionResponseDTO getResponseDTOForResponse(OBAPIResponseContext obapiResponseContext) { - - ExtensionResponseDTO extensionResponseDTO = new ExtensionResponseDTO(); - if (obapiResponseContext.isError()) { - int statusCode = (!obapiResponseContext.getContextProps().containsKey(GatewayConstants.ERROR_STATUS_PROP)) ? - HttpStatus.SC_INTERNAL_SERVER_ERROR : - Integer.parseInt(obapiResponseContext.getContextProperty(GatewayConstants.ERROR_STATUS_PROP)); - extensionResponseDTO.setStatusCode(statusCode); - extensionResponseDTO.setResponseStatus(ExtensionResponseStatus.RETURN_ERROR.toString()); - } else if (obapiResponseContext.getContextProps().containsKey(GatewayConstants.IS_RETURN_RESPONSE) && - Boolean.parseBoolean(obapiResponseContext.getContextProps().get(GatewayConstants.IS_RETURN_RESPONSE))) { - Map headers = obapiResponseContext.getMsgInfo().getHeaders(); - headers.put(GatewayConstants.CONTENT_TYPE_TAG, GatewayConstants.JSON_CONTENT_TYPE); - obapiResponseContext.getMsgInfo().setHeaders(headers); - extensionResponseDTO.setHeaders(headers); - if (obapiResponseContext.getContextProps().containsKey(GatewayConstants.MODIFIED_STATUS)) { - extensionResponseDTO.setStatusCode(Integer.parseInt(obapiResponseContext.getContextProps() - .get(GatewayConstants.MODIFIED_STATUS))); - } - extensionResponseDTO.setResponseStatus(ExtensionResponseStatus.RETURN_ERROR.toString()); - } else { - extensionResponseDTO.setResponseStatus(ExtensionResponseStatus.CONTINUE.toString()); - } - - String modifiedPayload = obapiResponseContext.getModifiedPayload(); - if (modifiedPayload != null) { - extensionResponseDTO.setPayload(new ByteArrayInputStream(modifiedPayload.getBytes(StandardCharsets.UTF_8))); - } - Map addedHeaders = obapiResponseContext.getAddedHeaders(); - if (addedHeaders.size() != 0) { - HashMap headers = new HashMap<>(); - headers.putAll(obapiResponseContext.getMsgInfo().getHeaders()); - for (Map.Entry headerEntry : addedHeaders.entrySet()) { - headers.put(headerEntry.getKey(), headerEntry.getValue()); - } - extensionResponseDTO.setHeaders(headers); - } - return extensionResponseDTO; - } - - @Override - public String getType() { - - return null; - } - - /** - * Method to store properties to cache. - * - * @param key unique cache key - * @param contextProps properties to store - */ - private void setPropertiesToCache(String key, Map contextProps) { - - GatewayDataHolder.getGatewayCache().addToCache(GatewayCacheKey.of(key), contextProps); - } - - /** - * Method to retrieve context properties from cache. - * - * @param key unique cache key - * @return context properties - */ - private Map getPropertiesFromCache(String key) { - //Need to implement after adding base cache implementation to the common module. - Object cachedObject = GatewayDataHolder.getGatewayCache().getFromCache(GatewayCacheKey.of(key)); - return cachedObject == null ? new HashMap<>() : (Map) cachedObject; - } - - /** - * Method to remove context properties from cache. - * - * @param key unique cache key - * @return context properties - */ - private void removePropertiesFromCache(String key) { - //Need to implement after adding base cache implementation to the common module. - GatewayDataHolder.getGatewayCache().removeFromCache(GatewayCacheKey.of(key)); - } - - private void publishAnalyticsData(Map analyticsData) { - - if (analyticsData != null && !analyticsData.isEmpty()) { - OBDataPublisherUtil. - publishData(GatewayConstants.API_DATA_STREAM, GatewayConstants.API_DATA_VERSION, analyticsData); - } - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/core/OpenBankingGatewayExecutor.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/core/OpenBankingGatewayExecutor.java deleted file mode 100644 index 9a659d3a..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/core/OpenBankingGatewayExecutor.java +++ /dev/null @@ -1,56 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.core; - -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; - -/** - * Open Banking executor interface. - */ -public interface OpenBankingGatewayExecutor { - - /** - * Method to handle pre request. - * - * @param obapiRequestContext OB request context object - */ - public void preProcessRequest(OBAPIRequestContext obapiRequestContext); - - /** - * Method to handle post request. - * - * @param obapiRequestContext OB request context object - */ - public void postProcessRequest(OBAPIRequestContext obapiRequestContext); - - /** - * Method to handle pre response. - * - * @param obapiResponseContext OB response context object - */ - public void preProcessResponse(OBAPIResponseContext obapiResponseContext); - - /** - * Method to handle post response. - * - * @param obapiResponseContext OB response context object - */ - public void postProcessResponse(OBAPIResponseContext obapiResponseContext); -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/dcr/DCRExecutor.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/dcr/DCRExecutor.java deleted file mode 100644 index 3eefec9c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/dcr/DCRExecutor.java +++ /dev/null @@ -1,1032 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.dcr; - -import com.google.gson.JsonArray; -import com.google.gson.JsonElement; -import com.google.gson.JsonObject; -import com.google.gson.JsonParser; -import com.nimbusds.jose.JOSEException; -import com.nimbusds.jose.proc.BadJOSEException; -import com.nimbusds.jwt.SignedJWT; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.error.OpenBankingErrorCodes; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.common.util.HTTPClientUtils; -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.common.util.OpenBankingUtils; -import com.wso2.openbanking.accelerator.gateway.cache.GatewayCacheKey; -import com.wso2.openbanking.accelerator.gateway.executor.core.OpenBankingGatewayExecutor; -import com.wso2.openbanking.accelerator.gateway.executor.exception.OpenBankingExecutorException; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OpenBankingExecutorError; -import com.wso2.openbanking.accelerator.gateway.internal.GatewayDataHolder; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import org.apache.axis2.transport.http.HTTPConstants; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.HttpEntity; -import org.apache.http.HttpHeaders; -import org.apache.http.HttpResponse; -import org.apache.http.HttpStatus; -import org.apache.http.NameValuePair; -import org.apache.http.client.entity.UrlEncodedFormEntity; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpDelete; -import org.apache.http.client.methods.HttpGet; -import org.apache.http.client.methods.HttpPost; -import org.apache.http.client.methods.HttpRequestBase; -import org.apache.http.client.utils.URIBuilder; -import org.apache.http.entity.StringEntity; -import org.apache.http.impl.client.CloseableHttpClient; -import org.apache.http.message.BasicNameValuePair; -import org.apache.http.util.EntityUtils; -import org.json.JSONException; - -import java.io.IOException; -import java.net.HttpURLConnection; -import java.net.MalformedURLException; -import java.net.URI; -import java.net.URISyntaxException; -import java.text.ParseException; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Collections; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.Optional; - -import javax.ws.rs.HttpMethod; - -/** - * Executor for signature validation, am app creation and API subscription for DCR. - */ -public class DCRExecutor implements OpenBankingGatewayExecutor { - - private static final Log log = LogFactory.getLog(DCRExecutor.class); - private static String clientIdParam = "client_id"; - private static String registrationAccessTokenParam = "registration_access_token"; - private static String clientSecret = "client_secret"; - private static String applicationIdParam = "applicationId"; - private static String userName = "userName"; - private static String obDCREndpoint = "api/openbanking/dynamic-client-registration/register"; - private static Map urlMap = GatewayDataHolder.getInstance().getUrlMap(); - - public static void setUrlMap(Map conf) { - - if (urlMap == null) { - DCRExecutor.urlMap = conf; - } - } - - @Generated(message = "Excluding from unit tests since there is an external http call") - @Override - public void preProcessRequest(OBAPIRequestContext obapiRequestContext) { - - if (obapiRequestContext.isError()) { - return; - } - boolean validateJWT = true; - Map configs = GatewayDataHolder.getInstance() - .getOpenBankingConfigurationService().getConfigurations(); - if (configs.containsKey(GatewayConstants.VALIDATE_JWT)) { - validateJWT = Boolean.parseBoolean(configs.get(GatewayConstants.VALIDATE_JWT).toString()); - } - - if (validateJWT) { - String payload = obapiRequestContext.getRequestPayload(); - try { - String httpMethod = obapiRequestContext.getMsgInfo().getHttpMethod(); - if (HttpMethod.POST.equalsIgnoreCase(httpMethod) || HttpMethod.PUT.equalsIgnoreCase(httpMethod)) { - if (payload != null) { - //decode request jwt - validateRequestSignature(payload, obapiRequestContext); - } else { - handleBadRequestError(obapiRequestContext, "Malformed request found"); - } - } - } catch (ParseException e) { - log.error("Error occurred while decoding the provided jwt", e); - handleBadRequestError(obapiRequestContext, "Malformed request JWT"); - } catch (BadJOSEException e) { - log.error("Error occurred while validating the signature", e); - handleBadRequestError(obapiRequestContext, "Invalid request signature. " + e.getMessage()); - } catch (JOSEException | MalformedURLException e) { - log.error("Error occurred while validating the signature", e); - handleBadRequestError(obapiRequestContext, "Invalid request signature"); - } catch (OpenBankingExecutorException e) { - log.error("Error occurred while validating the signature", e); - handleBadRequestError(obapiRequestContext, e.getErrorPayload()); - } - } - - } - - @Override - public void postProcessResponse(OBAPIResponseContext obapiResponseContext) { - - if (obapiResponseContext.isError()) { - return; - } - String basicAuthHeader = GatewayUtils.getBasicAuthHeader(urlMap.get(userName).toString(), - String.valueOf((char[]) urlMap.get(GatewayConstants.PASSWORD))); - Map> regulatoryAPIs = GatewayDataHolder.getInstance() - .getOpenBankingConfigurationService().getAllowedAPIs(); - - switch (obapiResponseContext.getMsgInfo().getHttpMethod().toUpperCase()) { - case HttpMethod.POST : - if (HttpStatus.SC_CREATED == obapiResponseContext.getStatusCode()) { - String fullBackEndURL = urlMap.get(GatewayConstants.IAM_HOSTNAME).toString().concat("/") - .concat(obDCREndpoint); - postProcessResponseForRegister(obapiResponseContext, basicAuthHeader, fullBackEndURL, - regulatoryAPIs); - } - break; - case HttpMethod.PUT : - if (HttpStatus.SC_OK == obapiResponseContext.getStatusCode()) { - postProcessResponseForUpdate(obapiResponseContext, basicAuthHeader, regulatoryAPIs); - } - break; - case HttpMethod.DELETE : - if (HttpStatus.SC_NO_CONTENT == obapiResponseContext.getStatusCode()) { - postProcessResponseForDelete(obapiResponseContext, basicAuthHeader); - } - } - } - - /** - * Method to handle post response. - * - * @param obapiResponseContext OB response context object - */ - @Generated(message = "Ignoring since it's implemented as an extension point") - @Override - public void preProcessResponse(OBAPIResponseContext obapiResponseContext) { - - } - - /** - * Method to handle post request. - * - * @param obapiRequestContext OB request context object - */ - @Override - public void postProcessRequest(OBAPIRequestContext obapiRequestContext) { - - if (obapiRequestContext.isError()) { - return; - } - String httpMethod = obapiRequestContext.getMsgInfo().getHttpMethod(); - - if (HttpMethod.GET.equalsIgnoreCase(httpMethod) || HttpMethod.PUT.equalsIgnoreCase(httpMethod) || - HttpMethod.DELETE.equalsIgnoreCase(httpMethod)) { - String[] contextPathValues = obapiRequestContext.getMsgInfo().getResource().split("/"); - String clientIdSentInRequest = ""; - List paramList = Arrays.asList(contextPathValues); - int count = paramList.size(); - clientIdSentInRequest = paramList.stream().skip(count - 1).findFirst().get().toString(); - String clientIdBoundToToken = obapiRequestContext.getApiRequestInfo().getConsumerKey(); - if (!clientIdSentInRequest.equals(clientIdBoundToToken)) { - obapiRequestContext.setError(true); - obapiRequestContext.addContextProperty(GatewayConstants.ERROR_STATUS_PROP, - String.valueOf(OpenBankingErrorCodes.UNAUTHORIZED_CODE)); - Map requestHeaders = obapiRequestContext.getMsgInfo().getHeaders(); - requestHeaders.remove(GatewayConstants.CONTENT_TYPE_TAG); - requestHeaders.remove(GatewayConstants.CONTENT_LENGTH); - obapiRequestContext.getMsgInfo().setHeaders(requestHeaders); - return; - } - } - char[] adminPassword = (char[]) urlMap.get(GatewayConstants.PASSWORD); - String basicAuthHeader = GatewayUtils.getBasicAuthHeader(urlMap.get(userName).toString(), - String.valueOf(adminPassword)); - Map headers = new HashMap<>(); - String bearerAccessToken = ""; - if (obapiRequestContext.getMsgInfo().getHeaders() != null && - obapiRequestContext.getMsgInfo().getHeaders().get(GatewayConstants.AUTH_HEADER) != null) { - bearerAccessToken = obapiRequestContext.getMsgInfo().getHeaders().get(GatewayConstants.AUTH_HEADER) - .replace(GatewayConstants.BEARER_TAG, "").trim(); - } - headers.put(GatewayConstants.AUTH_HEADER, basicAuthHeader); - headers.put(registrationAccessTokenParam, bearerAccessToken); - obapiRequestContext.setAddedHeaders(headers); - if (HttpMethod.DELETE.equalsIgnoreCase(httpMethod)) { - try { - //call dcr endpoint of IS to get the application name to be deleted - JsonObject createdSpDetails = callGet(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/") - .concat(obapiRequestContext.getApiRequestInfo().getConsumerKey()), - basicAuthHeader, "", "").getAsJsonObject(); - String applicationName = createdSpDetails.get("client_name").getAsString(); - - //add application name to the cache - String cacheKey = obapiRequestContext.getApiRequestInfo().getConsumerKey() - .concat(GatewayConstants.AM_APP_NAME_CACHEKEY); - GatewayDataHolder.getGatewayCache().addToCache(GatewayCacheKey.of(cacheKey), applicationName); - - } catch (IOException | OpenBankingException | URISyntaxException e) { - log.error("Error occurred while deleting application", e); - handleRequestInternalServerError(obapiRequestContext, OpenBankingErrorCodes.REGISTATION_DELETE_ERROR); - } - - } - } - - /** - * Method to handle response for DCR POST requests. - * - * @param obapiResponseContext OB response context object - * @param basicAuthHeader Basic authentication header for accessing the DCR endpoint. - * @param fullBackEndURL URL of the OB DCR Endpoint - * @param regulatoryAPIs A map containing regulatory API names and the related authorized roles - */ - private void postProcessResponseForRegister(OBAPIResponseContext obapiResponseContext, String basicAuthHeader, - String fullBackEndURL, Map> regulatoryAPIs) { - - try { - JsonParser jsonParser = new JsonParser(); - JsonObject createdDCRAppDetails = ((JsonObject) jsonParser - .parse(obapiResponseContext.getResponsePayload())); - //get software statement from dcr app details - String softwareStatement = createdDCRAppDetails.has(OpenBankingConstants.SOFTWARE_STATEMENT) ? - createdDCRAppDetails.get(OpenBankingConstants.SOFTWARE_STATEMENT).toString() : null; - - //call IS DCR endpoint to create application for obtaining a token to invoke devportal REST APIs - JsonElement registrationResponse = createServiceProvider(basicAuthHeader, - createdDCRAppDetails.get("software_id").getAsString()); - if (registrationResponse == null) { - log.error("Error while creating AM app for invoking APIM rest apis"); - String clientId = createdDCRAppDetails.get(clientIdParam).getAsString(); - //delete service provider - callDelete(fullBackEndURL.concat("/").concat(clientId), basicAuthHeader); - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTRATION_INTERNAL_ERROR); - return; - } - //call token endpoint to retrieve a token for invoking the devportal REST apis - String amRestAPIInvokeClientId = registrationResponse.getAsJsonObject() - .get(clientIdParam).getAsString(); - - String authHeaderForTokenRequest = GatewayUtils - .getBasicAuthHeader(registrationResponse.getAsJsonObject().get(clientIdParam).getAsString(), - registrationResponse.getAsJsonObject().get(clientSecret).getAsString()); - - JsonElement tokenResponse = getToken(authHeaderForTokenRequest, - urlMap.get(GatewayConstants.TOKEN_URL).toString(), amRestAPIInvokeClientId); - - if (tokenResponse == null || tokenResponse.getAsJsonObject().get("access_token") == null) { - log.error("Error while creating tokens"); - String clientId = createdDCRAppDetails.get(clientIdParam).getAsString(); - //delete service provider - callDelete(fullBackEndURL.concat("/").concat(clientId), basicAuthHeader); - //delete SP created for calling dev portal REST APIs - callDelete(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/") - .concat(amRestAPIInvokeClientId), basicAuthHeader); - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTRATION_INTERNAL_ERROR); - return; - } - String token = tokenResponse.getAsJsonObject().get("access_token").getAsString(); - String getSPDetails = urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/") - .concat(createdDCRAppDetails.get(clientIdParam).getAsString()); - //call IS dcr api to get client secret and client name - JsonElement createdSpDetails = callGet(getSPDetails, basicAuthHeader, "", ""); - if (createdSpDetails == null) { - log.error("Error while retrieving client id and secret"); - String clientId = createdDCRAppDetails.get(clientIdParam).getAsString(); - //delete service provider - callDelete(fullBackEndURL.concat("/").concat(clientId), basicAuthHeader); - //delete SP created for calling dev portal REST APIs - callDelete(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/") - .concat(amRestAPIInvokeClientId), basicAuthHeader); - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTRATION_INTERNAL_ERROR); - return; - } - - //create am application - JsonObject amAppCreatePayload = getAppCreatePayload(createdSpDetails.getAsJsonObject() - .get("client_name").getAsString()); - JsonElement amApplicationCreateResponse = - callPost(urlMap.get(GatewayConstants.APP_CREATE_URL).toString(), - amAppCreatePayload.toString(), GatewayConstants.BEARER_TAG.concat(token)); - - if (amApplicationCreateResponse == null) { - log.error("Error while creating AM app"); - String clientId = createdDCRAppDetails.get(clientIdParam).getAsString(); - //delete service provider - callDelete(fullBackEndURL.concat("/").concat(clientId), basicAuthHeader); - //delete SP created for calling dev portal REST APIs - callDelete(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/") - .concat(amRestAPIInvokeClientId), basicAuthHeader); - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTRATION_INTERNAL_ERROR); - return; - } - String keyMapURL = urlMap.get(GatewayConstants.KEY_MAP_URL).toString() - .replace("application-id", amApplicationCreateResponse.getAsJsonObject() - .get(applicationIdParam).getAsString()); - String keyManagerName = GatewayDataHolder.getInstance().getOpenBankingConfigurationService() - .getConfigurations().get(OpenBankingConstants.OB_KM_NAME).toString(); - - //map keys to am application - JsonObject keyMapPayload = getKeyMapPayload(createdDCRAppDetails.get(clientIdParam).getAsString(), - createdSpDetails.getAsJsonObject().get(clientSecret).getAsString(), - OpenBankingUtils.getSoftwareEnvironmentFromSSA(softwareStatement), keyManagerName); - - JsonElement amKeyMapResponse = callPost(keyMapURL, keyMapPayload.toString(), - GatewayConstants.BEARER_TAG.concat(token)); - if (amKeyMapResponse == null) { - log.error("Error while mapping keys to AM app"); - String clientId = createdDCRAppDetails.get(clientIdParam).getAsString(); - //delete service provider - callDelete(fullBackEndURL.concat("/").concat(clientId), basicAuthHeader); - //delete SP created for calling dev portal REST APIs - callDelete(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/") - .concat(amRestAPIInvokeClientId), basicAuthHeader); - //delete AM application - callDelete(urlMap.get(GatewayConstants.APP_CREATE_URL).toString() - .concat("/").concat(amApplicationCreateResponse.getAsJsonObject() - .get(applicationIdParam).getAsString()), GatewayConstants.BEARER_TAG.concat(token)); - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTRATION_INTERNAL_ERROR); - return; - } - //get list of published APIs - JsonElement publishedAPIsResponse = callGet(urlMap.get(GatewayConstants.API_RETRIEVE_URL).toString(), - GatewayConstants.BEARER_TAG.concat(token), "", ""); - if (publishedAPIsResponse == null) { - log.error("Error while retrieving published APIs"); - String clientId = createdDCRAppDetails.get(clientIdParam).getAsString(); - //delete service provider - callDelete(fullBackEndURL.concat("/").concat(clientId), basicAuthHeader); - //delete SP created for calling dev portal REST APIs - callDelete(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/") - .concat(amRestAPIInvokeClientId), basicAuthHeader); - //delete AM application - callDelete(urlMap.get(GatewayConstants.APP_CREATE_URL).toString() - .concat("/").concat(amApplicationCreateResponse.getAsJsonObject() - .get(applicationIdParam).getAsString()), GatewayConstants.BEARER_TAG.concat(token)); - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTRATION_INTERNAL_ERROR); - return; - } - - List apiIDList = new ArrayList<>(); - if (regulatoryAPIs != null) { - if (StringUtils.isEmpty(softwareStatement)) { - apiIDList = filterRegulatoryAPIs(regulatoryAPIs, publishedAPIsResponse.getAsJsonObject() - .get(OpenBankingConstants.API_LIST).getAsJsonArray(), Collections.emptyList()); - } else { - apiIDList = filterRegulatoryAPIs(regulatoryAPIs, publishedAPIsResponse.getAsJsonObject() - .get(OpenBankingConstants.API_LIST).getAsJsonArray(), getRolesFromSSA(softwareStatement)); - } - } else { - log.warn("No regulatory APIs configured. Application will be subscribed to all published APIs"); - //subscribe to all APIs if there are no configured regulatory APIs - for (JsonElement apiInfo : publishedAPIsResponse.getAsJsonObject().get("list").getAsJsonArray()) { - apiIDList.add(apiInfo.getAsJsonObject().get("id").getAsString()); - } - } - //subscribe to apis - JsonArray subscribeAPIsPayload = getAPISubscriptionPayload(amApplicationCreateResponse - .getAsJsonObject().get(applicationIdParam).getAsString(), apiIDList); - JsonElement subscribeAPIsResponse = callPost(urlMap.get(GatewayConstants.API_SUBSCRIBE_URL).toString(), - subscribeAPIsPayload.toString(), "Bearer ".concat(token)); - if (subscribeAPIsResponse == null) { - log.error("Error while subscribing to APIs"); - String clientId = createdDCRAppDetails.get(clientIdParam).getAsString(); - //delete service provider - callDelete(fullBackEndURL.concat("/").concat(clientId), basicAuthHeader); - //delete SP created for calling dev portal REST APIs - callDelete(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/") - .concat(amRestAPIInvokeClientId), basicAuthHeader); - //delete AM application - callDelete(urlMap.get(GatewayConstants.APP_CREATE_URL).toString() - .concat("/").concat(amApplicationCreateResponse.getAsJsonObject() - .get(applicationIdParam).getAsString()), GatewayConstants.BEARER_TAG.concat(token)); - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTRATION_INTERNAL_ERROR); - return; - } - - //delete IAM application used to invoke am rest endpoints - if (!callDelete(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/") - .concat(amRestAPIInvokeClientId), basicAuthHeader)) { - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTATION_DELETE_ERROR); - } - - } catch (IOException | OpenBankingException | URISyntaxException | ParseException e) { - log.error("Error occurred while creating application", e); - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTRATION_INTERNAL_ERROR); - } - } - - /** - * Method to handle response for DCR PUT requests. - * - * @param obapiResponseContext OB response context object - * @param basicAuthHeader Basic authentication header for accessing the DCR endpoint. - * @param regulatoryAPIs A map containing regulatory API names and the related authorized roles - */ - private void postProcessResponseForUpdate(OBAPIResponseContext obapiResponseContext, String basicAuthHeader, - Map> regulatoryAPIs) { - - JsonParser jsonParser = new JsonParser(); - JsonObject createdDCRAppDetails = ((JsonObject) jsonParser.parse(obapiResponseContext - .getResponsePayload())); - try { - JsonObject dcrPayload = getIAMDCRPayload(createdDCRAppDetails.get("software_id").getAsString()); - JsonElement registrationResponse = callPost(urlMap.get(GatewayConstants.IAM_DCR_URL).toString(), - dcrPayload.toString(), basicAuthHeader); - if (registrationResponse == null) { - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTRATION_UPDATE_ERROR); - return; - } - //call token endpoint to retrieve a token for invoking the devportal REST apis - String clientId = registrationResponse.getAsJsonObject().get(clientIdParam).getAsString(); - String authHeaderForTokenRequest = GatewayUtils.getBasicAuthHeader(clientId, - registrationResponse.getAsJsonObject().get(clientSecret).getAsString()); - - JsonElement tokenResponse = getToken(authHeaderForTokenRequest, - urlMap.get(GatewayConstants.TOKEN_URL).toString(), clientId); - if (tokenResponse == null || tokenResponse.getAsJsonObject().get("access_token") == null) { - log.error("Error while creating tokens"); - //delete SP created to call dev portal REST APIs - callDelete(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/") - .concat(clientId), basicAuthHeader); - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTRATION_UPDATE_ERROR); - return; - } - String token = tokenResponse.getAsJsonObject().get("access_token").getAsString(); - - String applicationName = getApplicationName(obapiResponseContext.getResponsePayload(), - GatewayDataHolder.getInstance().getOpenBankingConfigurationService().getConfigurations()); - if (StringUtils.isEmpty(applicationName)) { - log.error("Error while retrieving application name during update"); - //delete SP created to call dev portal REST APIs - callDelete(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/") - .concat(clientId), basicAuthHeader); - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTRATION_UPDATE_ERROR); - return; - } - //call application get endpoint to retrieve the application id - JsonElement applicationSearchResponse = - callGet(urlMap.get(GatewayConstants.APP_CREATE_URL).toString(), - GatewayConstants.BEARER_TAG.concat(token), "query", applicationName); - if (applicationSearchResponse == null) { - log.error("Error while searching for created application during update"); - //delete SP created to call dev portal REST APIs - callDelete(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/") - .concat(clientId), basicAuthHeader); - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTRATION_UPDATE_ERROR); - return; - } - String applicationId = applicationSearchResponse.getAsJsonObject().get("list").getAsJsonArray().get(0) - .getAsJsonObject().get(applicationIdParam).getAsString(); - - //get list of subscribed APIs - JsonElement subscribedAPIsResponse = callGet(urlMap.get(GatewayConstants.API_GET_SUBSCRIBED).toString(), - GatewayConstants.BEARER_TAG.concat(token), "applicationId", applicationId); - if (subscribedAPIsResponse == null) { - log.error("Error while retrieving subscribed APIs"); - //delete SP created to call dev portal REST APIs - callDelete(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/") - .concat(clientId), basicAuthHeader); - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTRATION_UPDATE_ERROR); - return; - } - List subscribedAPIIdList = new ArrayList<>(); - for (JsonElement subscribedAPI : subscribedAPIsResponse.getAsJsonObject().get("list") - .getAsJsonArray()) { - String apiId = subscribedAPI.getAsJsonObject().get("apiId").getAsString(); - subscribedAPIIdList.add(apiId); - } - - //get software statement from dcr app details - String softwareStatement = createdDCRAppDetails.has(OpenBankingConstants.SOFTWARE_STATEMENT) ? - createdDCRAppDetails.get(OpenBankingConstants.SOFTWARE_STATEMENT).getAsString() : null; - if (StringUtils.isNotEmpty(softwareStatement)) { - final JsonArray subscribedAPIs = subscribedAPIsResponse.getAsJsonObject() - .get("list").getAsJsonArray(); - //check whether the ssa still contains the roles related to the subscribed APIs and unsubscribe if not - Optional.of(getRolesFromSSA(softwareStatement)) - .map(ssaRoles -> getUnAuthorizedAPIs(subscribedAPIs, regulatoryAPIs, ssaRoles)) - .flatMap(unAuthorizedApis -> unAuthorizedApis.stream() - .map(unAuthorizedApi -> String.format("%s/%s", - urlMap.get(GatewayConstants.API_GET_SUBSCRIBED).toString(), unAuthorizedApi)) - .filter(endpoint -> isSubscriptionDeletionFailed(endpoint, GatewayConstants.BEARER_TAG - .concat(token))) - .findAny()) - .ifPresent(endpoint -> { - log.error("Error while unsubscribing from API: " + endpoint); - //delete SP created to call dev portal REST APIs - try { - callDelete(String.format("%s/%s", urlMap.get(GatewayConstants.IAM_DCR_URL).toString(), - clientId), basicAuthHeader); - } catch (OpenBankingException | IOException e) { - handleInternalServerError(obapiResponseContext, - OpenBankingErrorCodes.REGISTRATION_INTERNAL_ERROR); - } - }); - } - //subscribe to new APIs if new roles were added to the SSA - //get list of published APIs - JsonElement publishedAPIsResponse = callGet(urlMap.get(GatewayConstants.API_RETRIEVE_URL).toString(), - GatewayConstants.BEARER_TAG.concat(token), "", ""); - if (publishedAPIsResponse == null) { - log.error("Error while retrieving published APIs"); - //delete SP created to call dev portal REST APIs - callDelete(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/") - .concat(clientId), basicAuthHeader); - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTRATION_UPDATE_ERROR); - return; - } - List apiIDList = new ArrayList<>(); - if (StringUtils.isEmpty(softwareStatement)) { - filterRegulatoryAPIs(regulatoryAPIs, publishedAPIsResponse.getAsJsonObject() - .get(OpenBankingConstants.API_LIST).getAsJsonArray(), Collections.emptyList()); - } else { - filterRegulatoryAPIs(regulatoryAPIs, publishedAPIsResponse.getAsJsonObject() - .get(OpenBankingConstants.API_LIST).getAsJsonArray(), getRolesFromSSA(softwareStatement)); - } - - List newApisListToSubscribe = getNewAPIsToSubscribe(apiIDList, subscribedAPIIdList); - if (!newApisListToSubscribe.isEmpty()) { - JsonArray subscribeAPIsPayload = getAPISubscriptionPayload(applicationId, newApisListToSubscribe); - JsonElement subscribeAPIsResponse = callPost(urlMap.get(GatewayConstants.API_SUBSCRIBE_URL) - .toString(), subscribeAPIsPayload.toString(), "Bearer ".concat(token)); - if (subscribeAPIsResponse == null) { - log.error("Error while subscribing to APIs"); - //delete SP created to call dev portal REST APIs - callDelete(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/") - .concat(clientId), basicAuthHeader); - handleInternalServerError(obapiResponseContext, - OpenBankingErrorCodes.REGISTRATION_UPDATE_ERROR); - return; - } - } - //delete IAM application used to invoke am rest endpoints - if (!callDelete(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/") - .concat(clientId), basicAuthHeader)) { - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTRATION_UPDATE_ERROR); - } - } catch (ParseException | IOException | URISyntaxException | OpenBankingException e) { - log.error("Error occurred while creating application", e); - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTRATION_UPDATE_ERROR); - } - } - - - /** - * Method to handle response for DCR DELETE requests. - * - * @param obapiResponseContext OB response context object - * @param basicAuthHeader Basic authentication header for accessing the DCR endpoint. - */ - private void postProcessResponseForDelete(OBAPIResponseContext obapiResponseContext, String basicAuthHeader) { - - try { - JsonObject dcrPayload = getIAMDCRPayload(obapiResponseContext.getApiRequestInfo().getConsumerKey()); - JsonElement registrationResponse = callPost(urlMap.get(GatewayConstants.IAM_DCR_URL).toString(), - dcrPayload.toString(), basicAuthHeader); - if (registrationResponse == null) { - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTATION_DELETE_ERROR); - return; - } - - //call token endpoint to retrieve a token for invoking the devportal REST apis - String clientId = registrationResponse.getAsJsonObject().get(clientIdParam).getAsString(); - String authHeaderForTokenRequest = GatewayUtils.getBasicAuthHeader(clientId, - registrationResponse.getAsJsonObject().get(clientSecret).getAsString()); - - JsonElement tokenResponse = getToken(authHeaderForTokenRequest, - urlMap.get(GatewayConstants.TOKEN_URL).toString(), clientId); - if (tokenResponse == null || tokenResponse.getAsJsonObject().get("access_token") == null) { - log.error("Error while creating tokens during delete"); - //delete IAM application used to invoke am rest endpoints - callDelete(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/").concat(clientId), - basicAuthHeader); - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTATION_DELETE_ERROR); - return; - } - String token = tokenResponse.getAsJsonObject().get("access_token").getAsString(); - - //get application id of the sent request - String cacheKey = obapiResponseContext.getApiRequestInfo().getConsumerKey() - .concat(GatewayConstants.AM_APP_NAME_CACHEKEY); - String applicationName = GatewayDataHolder.getGatewayCache() - .getFromCache(GatewayCacheKey.of(cacheKey)).toString(); - - //Adding applicationName to contextProps for use in next steps - Map contextProps = obapiResponseContext.getContextProps(); - contextProps.put("AppName", applicationName); - obapiResponseContext.setContextProps(contextProps); - - //call application get endpoint to retrieve the application id - JsonElement applicationSearchResponse = - callGet(urlMap.get(GatewayConstants.APP_CREATE_URL).toString(), - GatewayConstants.BEARER_TAG.concat(token), "query", applicationName); - if (applicationSearchResponse == null) { - log.error("Error while searching application during delete"); - //delete IAM application used to invoke am rest endpoints - callDelete(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/").concat(clientId), - basicAuthHeader); - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTATION_DELETE_ERROR); - return; - } - - String applicationId = applicationSearchResponse.getAsJsonObject().get("list").getAsJsonArray().get(0) - .getAsJsonObject().get(applicationIdParam).getAsString(); - - if (!callDelete(urlMap.get(GatewayConstants.APP_CREATE_URL).toString() - .concat("/").concat(applicationId), GatewayConstants.BEARER_TAG.concat(token))) { - log.error("Error while deleting AM application"); - //delete IAM application used to invoke am rest endpoints - callDelete(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/").concat(clientId), - basicAuthHeader); - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTATION_DELETE_ERROR); - return; - } - - //delete IAM application used to invoke am rest endpoints - if (!callDelete(urlMap.get(GatewayConstants.IAM_DCR_URL).toString().concat("/").concat(clientId), - basicAuthHeader)) { - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTATION_DELETE_ERROR); - } - } catch (IOException | OpenBankingException | URISyntaxException e) { - log.error("Error occurred while deleting application", e); - handleInternalServerError(obapiResponseContext, OpenBankingErrorCodes.REGISTATION_DELETE_ERROR); - } - } - - private JsonObject getIAMDCRPayload(String uniqueId) { - - JsonObject jsonObject = new JsonObject(); - JsonElement jsonElement = new JsonArray(); - /* Concatenating the unique id (software id/client id) to the rest api invoking SP name to avoid - issues in parallel requests - */ - String restApiInvokerName = "AM_RESTAPI_INVOKER_".concat(uniqueId); - ((JsonArray) jsonElement).add("client_credentials"); - jsonObject.addProperty("client_name", restApiInvokerName); - jsonObject.add("grant_types", jsonElement); - return jsonObject; - } - - private JsonObject getAppCreatePayload(String applicationName) { - - JsonObject jsonObject = new JsonObject(); - jsonObject.addProperty("name", applicationName); - jsonObject.addProperty("throttlingPolicy", "Unlimited"); - return jsonObject; - - } - - private JsonObject getKeyMapPayload(String consumerKey, String consumerSecret, String keyType, - String keyManagerName) { - - JsonObject jsonObject = new JsonObject(); - jsonObject.addProperty("consumerKey", consumerKey); - jsonObject.addProperty("consumerSecret", consumerSecret); - jsonObject.addProperty("keyType", keyType); - jsonObject.addProperty("keyManager", keyManagerName); - return jsonObject; - - } - - private JsonArray getAPISubscriptionPayload(String applicationId, List apiIdList) { - - JsonArray jsonArray = new JsonArray(); - for (String apiID : apiIdList) { - JsonObject apiInfo = new JsonObject(); - apiInfo.addProperty(applicationIdParam, applicationId); - apiInfo.addProperty("apiId", apiID); - apiInfo.addProperty("throttlingPolicy", "Unlimited"); - jsonArray.add(apiInfo); - } - return jsonArray; - } - - @Generated(message = "Excluding since it requires an Http response") - private JsonElement getResponse(HttpResponse response) throws IOException { - - HttpEntity entity = response.getEntity(); - if (response.getStatusLine().getStatusCode() == HttpStatus.SC_OK || - response.getStatusLine().getStatusCode() == HttpStatus.SC_CREATED) { - String responseStr = EntityUtils.toString(entity); - JsonParser parser = new JsonParser(); - return parser.parse(responseStr); - - } else { - String error = String.format("Error while invoking rest api : %s %s", - response.getStatusLine().getStatusCode(), response.getStatusLine().getReasonPhrase()); - log.error(error); - return null; - } - - } - - @Generated(message = "Excluding from test coverage since it is an HTTP call") - protected JsonElement callPost(String endpoint, String payload, String authenticationHeader) - throws IOException, OpenBankingException { - - try (CloseableHttpClient httpClient = HTTPClientUtils.getHttpsClient()) { - HttpPost httpPost = new HttpPost(endpoint); - StringEntity entity = new StringEntity(payload); - httpPost.setEntity(entity); - httpPost.setHeader(GatewayConstants.ACCEPT, GatewayConstants.JSON_CONTENT_TYPE); - httpPost.setHeader(GatewayConstants.CONTENT_TYPE_TAG, GatewayConstants.JSON_CONTENT_TYPE); - httpPost.setHeader(HttpHeaders.AUTHORIZATION, authenticationHeader); - CloseableHttpResponse httpResponse = httpClient.execute(httpPost); - return getResponse(httpResponse); - } - } - - @Generated(message = "Excluding from test coverage since it is an HTTP call") - protected JsonElement getToken(String authHeader, String url, String clientId) throws IOException, JSONException, - OpenBankingException { - - try (CloseableHttpClient client = HTTPClientUtils.getHttpsClient()) { - HttpPost request = new HttpPost(url); - List params = new ArrayList<>(); - params.add(new BasicNameValuePair("grant_type", "client_credentials")); - params.add(new BasicNameValuePair("scope", "apim:subscribe apim:api_key apim:app_manage " + - "apim:sub_manage openid")); - //params.add(new BasicNameValuePair("client_id", clientId)); - request.setEntity(new UrlEncodedFormEntity(params)); - request.addHeader(HTTPConstants.HEADER_AUTHORIZATION, authHeader); - HttpResponse response = client.execute(request); - if (response.getStatusLine().getStatusCode() != HttpURLConnection.HTTP_OK) { - log.error("Obtaining access token failed with status code: " + - response.getStatusLine().getStatusCode()); - return new JsonObject(); - } - return getResponse(response); - } - } - - /** - * Filters the regulatory APIs based on the given software roles. - * - * @param regulatoryAPIs A map containing regulatory API names and their allowed roles - * @param publishedAPIs The list of published APIs as JSON array - * @param softwareRoles The list of software roles provided in the request - * @return A list of API IDs that the application is authorized to access - */ - protected List filterRegulatoryAPIs(Map> regulatoryAPIs, JsonArray publishedAPIs, - List softwareRoles) { - - List filteredAPIs = new ArrayList<>(); - for (JsonElement publishedAPIInfo : publishedAPIs) { - String publishedAPIName = publishedAPIInfo.getAsJsonObject().get(OpenBankingConstants.API_NAME) - .getAsString(); - if (regulatoryAPIs.containsKey(publishedAPIName)) { - List allowedRolesForAPI = regulatoryAPIs.get(publishedAPIName); - // Check if no specific roles are configured for the API or if software roles contain any of the - // allowed roles - if (allowedRolesForAPI.isEmpty() || allowedRolesForAPI.stream().anyMatch(softwareRoles::contains)) { - filteredAPIs.add(publishedAPIInfo.getAsJsonObject().get(OpenBankingConstants.API_ID).getAsString()); - } - } - } - return filteredAPIs; - } - - @Generated(message = "Excluding from test coverage since it is an HTTP call") - protected JsonElement callGet(String endpoint, String authHeader, String queryParamKey, String paramValue) - throws IOException, OpenBankingException, URISyntaxException { - - try (CloseableHttpClient httpClient = HTTPClientUtils.getHttpsClient()) { - HttpGet httpGet = new HttpGet(endpoint); - List nameValuePairs = new ArrayList(); - if (StringUtils.isNotEmpty(queryParamKey)) { - nameValuePairs.add(new BasicNameValuePair(queryParamKey, paramValue)); - URI uri = new URIBuilder(httpGet.getURI()).addParameters(nameValuePairs).build(); - ((HttpRequestBase) httpGet).setURI(uri); - } - httpGet.setHeader("Accept", "application/json"); - httpGet.setHeader(HttpHeaders.AUTHORIZATION, authHeader); - CloseableHttpResponse restAPIResponse = httpClient.execute(httpGet); - return getResponse(restAPIResponse); - } - } - - private void handleInternalServerError(OBAPIResponseContext obapiResponseContext, String message) { - - //catch errors and set to context - OpenBankingExecutorError error = new OpenBankingExecutorError(OpenBankingErrorCodes.SERVER_ERROR_CODE, - "Internal server error", message, OpenBankingErrorCodes.SERVER_ERROR_CODE); - ArrayList executorErrors = obapiResponseContext.getErrors(); - executorErrors.add(error); - obapiResponseContext.setError(true); - obapiResponseContext.setErrors(executorErrors); - - } - - private void handleRequestInternalServerError(OBAPIRequestContext obapiResponseContext, String message) { - - //catch errors and set to context - OpenBankingExecutorError error = new OpenBankingExecutorError(OpenBankingErrorCodes.SERVER_ERROR_CODE, - "Internal server error", message, OpenBankingErrorCodes.SERVER_ERROR_CODE); - ArrayList executorErrors = obapiResponseContext.getErrors(); - executorErrors.add(error); - obapiResponseContext.setError(true); - obapiResponseContext.setErrors(executorErrors); - - } - - @Generated(message = "Excluding from test coverage since it is an HTTP call") - protected boolean callDelete(String endpoint, String authHeader) throws OpenBankingException, IOException { - - try (CloseableHttpClient httpClient = HTTPClientUtils.getHttpsClient()) { - HttpDelete httpDelete = new HttpDelete(endpoint); - httpDelete.setHeader(HttpHeaders.AUTHORIZATION, authHeader); - CloseableHttpResponse appDeletedResponse = httpClient.execute(httpDelete); - int status = appDeletedResponse.getStatusLine().getStatusCode(); - return (status == 204 || status == 200); - } - } - - /** - * Check if the deletion of subscription at a given endpoint failed. - * - * @param endpoint The URL of the endpoint where the subscription deletion is attempted - * @param authHeader The authorization header to be used in the HTTP request - * @return True if the subscription deletion fails or an exception occurs, false otherwise - */ - protected boolean isSubscriptionDeletionFailed(String endpoint, String authHeader) { - - try { - return !callDelete(endpoint, GatewayConstants.BEARER_TAG.concat(authHeader)); - } catch (OpenBankingException | IOException e) { - return true; - } - } - - private void handleBadRequestError(OBAPIRequestContext obapiRequestContext, String message) { - - //catch errors and set to context - OpenBankingExecutorError error = new OpenBankingExecutorError("Bad request", - "invalid_client_metadata", message, "400"); - ArrayList executorErrors = obapiRequestContext.getErrors(); - executorErrors.add(error); - obapiRequestContext.setError(true); - obapiRequestContext.setErrors(executorErrors); - - } - - @Generated(message = "Excluding from unit tests since there is an external http call") - private void validateRequestSignature(String payload, OBAPIRequestContext obapiRequestContext) - throws ParseException, JOSEException, BadJOSEException, MalformedURLException, - OpenBankingExecutorException { - - String jwksEndpointName = GatewayDataHolder.getInstance().getOpenBankingConfigurationService() - .getConfigurations().get(OpenBankingConstants.JWKS_ENDPOINT_NAME).toString(); - //decode request jwt - JSONObject decodedSSA; - JSONObject decodedRequest = JWTUtils.decodeRequestJWT(payload, "body"); - - //Check whether decodedRequest is null - if (decodedRequest == null) { - throw new OpenBankingExecutorException("invalid_client_metadata", OpenBankingErrorCodes.BAD_REQUEST_CODE, - "Provided jwt is malformed and cannot be decoded"); - } - - //Check whether the SSA exists and decode the SSA - if (decodedRequest.containsKey(OpenBankingConstants.SOFTWARE_STATEMENT) && - decodedRequest.getAsString(OpenBankingConstants.SOFTWARE_STATEMENT) != null) { - decodedSSA = JWTUtils.decodeRequestJWT(decodedRequest - .getAsString(OpenBankingConstants.SOFTWARE_STATEMENT), "body"); - } else { - //Throwing an exception whn SSA is not found - throw new OpenBankingExecutorException("invalid_client_metadata", OpenBankingErrorCodes.BAD_REQUEST_CODE, - "Required parameter software statement cannot be null"); - } - - //validate request signature - String jwksEndpoint = decodedSSA.getAsString(jwksEndpointName); - SignedJWT signedJWT = SignedJWT.parse(payload); - String alg = signedJWT.getHeader().getAlgorithm().getName(); - JWTUtils.validateJWTSignature(payload, jwksEndpoint, alg); - obapiRequestContext.setModifiedPayload(decodedRequest.toJSONString()); - Map requestHeaders = obapiRequestContext.getMsgInfo().getHeaders(); - requestHeaders.remove("Content-Type"); - Map addedHeaders = obapiRequestContext.getAddedHeaders(); - addedHeaders.put(GatewayConstants.CONTENT_TYPE_TAG, GatewayConstants.JSON_CONTENT_TYPE); - obapiRequestContext.setAddedHeaders(addedHeaders); - obapiRequestContext.getMsgInfo().setHeaders(requestHeaders); - } - - /** - * Extract roles from SSA. - * - * @param softwareStatement software statement extracted from request payload - * @return list of roles - * @throws ParseException - */ - public List getRolesFromSSA(String softwareStatement) throws ParseException { - - List softwareRoleList = new ArrayList<>(); - // decode software statement and get payload - JSONObject softwareStatementBody = JWTUtils.decodeRequestJWT(softwareStatement, "body"); - Object softwareRolesStr = softwareStatementBody.get(OpenBankingConstants.SOFTWARE_ROLES); - if (softwareRolesStr instanceof JSONArray) { - JSONArray softwareRoles = (JSONArray) softwareRolesStr; - for (Object role : softwareRoles) { - softwareRoleList.add(role.toString()); - } - } else if (softwareRolesStr instanceof String) { - softwareRoleList = Arrays.asList(softwareRolesStr.toString().split(" ")); - } - return softwareRoleList; - } - - protected String getApplicationName(String responsePayload, Map configurations) - throws ParseException { - - JsonParser jsonParser = new JsonParser(); - JsonObject createdDCRAppDetails = ((JsonObject) jsonParser.parse(responsePayload)); - String softwareStatement = createdDCRAppDetails.has(OpenBankingConstants.SOFTWARE_STATEMENT) ? - createdDCRAppDetails.get(OpenBankingConstants.SOFTWARE_STATEMENT).getAsString() : null; - boolean isSoftwareIdAppName = Boolean.parseBoolean(configurations - .get(OpenBankingConstants.DCR_USE_SOFTWAREID_AS_APPNAME).toString()); - String applicationNameKey = configurations.get(OpenBankingConstants.DCR_APPLICATION_NAME_KEY).toString(); - - // If a software statement is not provided, get the software id directly from created app details - if (StringUtils.isEmpty(softwareStatement)) { - if (isSoftwareIdAppName) { - return createdDCRAppDetails.get(OpenBankingConstants.SOFTWARE_ID).getAsString(); - } - } else { - JSONObject softwareStatementBody = JWTUtils.decodeRequestJWT(softwareStatement, - OpenBankingConstants.JWT_BODY); - if (isSoftwareIdAppName) { - //get software id form the software statement - return softwareStatementBody.get(OpenBankingConstants.SOFTWARE_ID).toString(); - } else if (softwareStatementBody.containsKey(applicationNameKey)) { - return softwareStatementBody.get(applicationNameKey).toString(); - } - } - return createdDCRAppDetails.get(applicationNameKey).getAsString(); - } - - protected List getUnAuthorizedAPIs(JsonArray subscribedAPIs, Map> configuredAPIs, - List allowedRoles) { - - List apisToUnsubscribe = new ArrayList<>(); - for (JsonElement apiName : subscribedAPIs) { - for (Map.Entry> entry : configuredAPIs.entrySet()) { - if (entry.getKey().equals(apiName.getAsJsonObject().get("apiInfo").getAsJsonObject().get("name") - .getAsString())) { - List allowedRolesForAPI = entry.getValue(); - boolean allowedAPI = false; - for (String allowedRole : allowedRolesForAPI) { - if (allowedRoles.contains(allowedRole)) { - allowedAPI = true; - break; - } - } - if (!allowedAPI) { - apisToUnsubscribe.add(apiName.getAsJsonObject().get("subscriptionId").getAsString()); - } - } - } - } - return apisToUnsubscribe; - } - - protected List getNewAPIsToSubscribe(List filteredAPIs, List subscribedAPIs) { - - List apisToSubscribe = new ArrayList<>(); - for (String publishedAPI : filteredAPIs) { - if (!subscribedAPIs.contains(publishedAPI)) { - apisToSubscribe.add(publishedAPI); - } - } - return apisToSubscribe; - } - - protected JsonElement createServiceProvider(String basicAuthHeader, String softwareId) - throws IOException, OpenBankingException { - - JsonObject dcrPayload = getIAMDCRPayload(softwareId); - return callPost(urlMap.get(GatewayConstants.IAM_DCR_URL).toString(), - dcrPayload.toString(), basicAuthHeader); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/exception/OpenBankingExecutorException.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/exception/OpenBankingExecutorException.java deleted file mode 100644 index 9b14ddf8..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/exception/OpenBankingExecutorException.java +++ /dev/null @@ -1,78 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.exception; - -/** - * Open Banking executor exception class. - */ -public class OpenBankingExecutorException extends Exception { - - private String errorCode; - private String errorPayload; - - public OpenBankingExecutorException(String message, String errorCode, String errorPayload) { - - super(message); - this.errorCode = errorCode; - this.errorPayload = errorPayload; - } - - public OpenBankingExecutorException(String message, Throwable cause) { - - super(message, cause); - } - - public OpenBankingExecutorException(String message) { - super(message); - } - - public OpenBankingExecutorException(Throwable cause, String errorCode, String errorPayload) { - - super(cause); - this.errorCode = errorCode; - this.errorPayload = errorPayload; - } - - public OpenBankingExecutorException(String message, Throwable cause, String errorCode, String errorPayload) { - - super(message, cause); - this.errorCode = errorCode; - this.errorPayload = errorPayload; - } - - public String getErrorCode() { - - return errorCode; - } - - public void setErrorCode(String errorCode) { - - this.errorCode = errorCode; - } - - public String getErrorPayload() { - - return errorPayload; - } - - public void setErrorPayload(String errorPayload) { - - this.errorPayload = errorPayload; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/api/resource/access/validation/APIResourceAccessValidationExecutor.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/api/resource/access/validation/APIResourceAccessValidationExecutor.java deleted file mode 100644 index b808fe96..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/api/resource/access/validation/APIResourceAccessValidationExecutor.java +++ /dev/null @@ -1,104 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.impl.api.resource.access.validation; - -import com.wso2.openbanking.accelerator.common.error.OpenBankingErrorCodes; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.gateway.executor.core.OpenBankingGatewayExecutor; -import com.wso2.openbanking.accelerator.gateway.executor.exception.OpenBankingExecutorException; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OpenBankingExecutorError; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.ArrayList; -import java.util.List; -import java.util.Map; - -/** - * API Resource Access Validation executor. - * This executor validates the grant type. - */ -public class APIResourceAccessValidationExecutor implements OpenBankingGatewayExecutor { - - private static final Log log = LogFactory.getLog(APIResourceAccessValidationExecutor.class); - - @Override - public void preProcessRequest(OBAPIRequestContext obapiRequestContext) { - - } - - @Override - public void preProcessResponse(OBAPIResponseContext obapiResponseContext) { - - } - - /** - * Method to handle post response. - * - * @param obapiResponseContext OB response context object - */ - @Override - public void postProcessResponse(OBAPIResponseContext obapiResponseContext) { - - } - - /** - * Method to handle post request. - * - * @param obapiRequestContext OB request context object - */ - @Generated(message = "Ignoring since all cases are covered from other unit tests") - @Override - public void postProcessRequest(OBAPIRequestContext obapiRequestContext) { - // Skip the executor if previous executors failed. - if (obapiRequestContext.isError()) { - return; - } - - // Get allowed security definitions - List allowedOAuthFlows = GatewayUtils.getAllowedOAuthFlows(obapiRequestContext); - - // Return if the end point is not secured - if (allowedOAuthFlows.isEmpty()) { - log.debug("Requested resource does not require authentication."); - return; - } - - // Retrieve grant types of the access token - Map transportHeaders = obapiRequestContext.getMsgInfo().getHeaders(); - try { - String bearerTokenPayload = GatewayUtils.getBearerTokenPayload(transportHeaders); - String tokenType = GatewayUtils.getTokenType(bearerTokenPayload); - - //validation - GatewayUtils.validateGrantType(tokenType, allowedOAuthFlows); - } catch (OpenBankingExecutorException e) { - //catch errors and set to context - OpenBankingExecutorError error = new OpenBankingExecutorError(e.getErrorCode(), e.getMessage(), - e.getErrorPayload(), OpenBankingErrorCodes.UNAUTHORIZED_CODE); - ArrayList executorErrors = obapiRequestContext.getErrors(); - executorErrors.add(error); - obapiRequestContext.setError(true); - obapiRequestContext.setErrors(executorErrors); - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/common/reporting/data/executor/CommonReportingDataExecutor.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/common/reporting/data/executor/CommonReportingDataExecutor.java deleted file mode 100644 index 67dd3263..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/common/reporting/data/executor/CommonReportingDataExecutor.java +++ /dev/null @@ -1,150 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.impl.common.reporting.data.executor; - -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.gateway.cache.GatewayCacheKey; -import com.wso2.openbanking.accelerator.gateway.executor.core.OpenBankingGatewayExecutor; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.internal.GatewayDataHolder; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.time.Instant; -import java.util.Map; - -/** - * Common Reporting Data Executor. - */ -public class CommonReportingDataExecutor implements OpenBankingGatewayExecutor { - - private static final Log log = LogFactory.getLog(CommonReportingDataExecutor.class); - - private static final String CLIENT_USER_AGENT = "User-Agent"; - private static final String USER_AGENT = "userAgent"; - private static final String TIMESTAMP = "timestamp"; - private static final String ELECTED_RESOURCE = "electedResource"; - private static final String RESPONSE_PAYLOAD_SIZE = "responsePayloadSize"; - private static final String HTTP_METHOD = "httpMethod"; - private static final String STATUS_CODE = "statusCode"; - private static final String CONSENT_ID = "consentId"; - private static final String CONSUMER_ID = "consumerId"; - private static final String API_NAME = "apiName"; - private static final String API_SPEC_VERSION = "apiSpecVersion"; - private static final String CLIENT_ID = "clientId"; - private static final String MESSAGE_ID = "messageId"; - private static final String NAME_TAG = "_name"; - - /** - * Method to handle pre request. - * - * @param obapiRequestContext OB request context object - */ - @Override - public void preProcessRequest(OBAPIRequestContext obapiRequestContext) { - - Map analyticsData = obapiRequestContext.getAnalyticsData(); - - String httpMethod = obapiRequestContext.getMsgInfo().getHttpMethod(); - analyticsData.put(HTTP_METHOD, httpMethod); - - Map headers = obapiRequestContext.getMsgInfo().getHeaders(); - - String userAgent = headers.get(CLIENT_USER_AGENT); - analyticsData.put(USER_AGENT, userAgent); - - String electedResource = obapiRequestContext.getMsgInfo().getElectedResource(); - analyticsData.put(ELECTED_RESOURCE, electedResource); - - String apiName = getApiName(obapiRequestContext); - - analyticsData.put(API_NAME, apiName); - - String apiSpecVersion = obapiRequestContext.getApiRequestInfo().getVersion(); - analyticsData.put(API_SPEC_VERSION, apiSpecVersion); - - analyticsData.put(MESSAGE_ID, obapiRequestContext.getMsgInfo().getMessageId()); - analyticsData.put(TIMESTAMP, Instant.now().getEpochSecond()); - - // Add analytics data to a map - obapiRequestContext.setAnalyticsData(analyticsData); - } - - @Override - public void postProcessRequest(OBAPIRequestContext obapiRequestContext) { - - Map analyticsData = obapiRequestContext.getAnalyticsData(); - String consentId = obapiRequestContext.getConsentId(); - analyticsData.put(CONSENT_ID, consentId); - analyticsData.put(CLIENT_ID, obapiRequestContext.getApiRequestInfo().getConsumerKey()); - analyticsData.put(CONSUMER_ID, obapiRequestContext.getApiRequestInfo().getUsername()); - - // Add analytics data to a map - obapiRequestContext.setAnalyticsData(analyticsData); - } - - @Override - public void preProcessResponse(OBAPIResponseContext obapiResponseContext) { - - Map analyticsData = obapiResponseContext.getAnalyticsData(); - - analyticsData.put(STATUS_CODE, obapiResponseContext.getStatusCode()); - - String payload = obapiResponseContext.getModifiedPayload() != null ? - obapiResponseContext.getModifiedPayload() : obapiResponseContext.getResponsePayload(); - long responsePayloadSize = payload != null ? payload.length() : 0; - analyticsData.put(RESPONSE_PAYLOAD_SIZE, responsePayloadSize); - - // Add data to analytics data map - obapiResponseContext.setAnalyticsData(analyticsData); - - } - - /** - * Method to handle post response. - * - * @param obapiResponseContext OB response context object - */ - @Override - public void postProcessResponse(OBAPIResponseContext obapiResponseContext) { - - } - - /** - * Method to get api name from cache. - * @param obapiRequestContext ob api request context - * @return api name - */ - @Generated(message = "Ignoring tests since this method is used to get name from cache") - protected String getApiName(OBAPIRequestContext obapiRequestContext) { - - String apiName; - String apiNameCacheKey = obapiRequestContext.getApiRequestInfo().getApiId() + NAME_TAG; - Object cacheObject = GatewayDataHolder.getGatewayCache().getFromCache(GatewayCacheKey.of(apiNameCacheKey)); - - if (cacheObject == null) { - apiName = obapiRequestContext.getOpenAPI().getInfo().getTitle(); - GatewayDataHolder.getGatewayCache().addToCache(GatewayCacheKey.of(apiNameCacheKey), apiName); - } else { - apiName = (String) cacheObject; - } - return apiName; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/consent/ConsentEnforcementExecutor.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/consent/ConsentEnforcementExecutor.java deleted file mode 100644 index 8ccb1415..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/consent/ConsentEnforcementExecutor.java +++ /dev/null @@ -1,312 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.impl.consent; - -import com.wso2.openbanking.accelerator.common.error.OpenBankingErrorCodes; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.gateway.executor.core.OpenBankingGatewayExecutor; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OpenBankingExecutorError; -import com.wso2.openbanking.accelerator.gateway.internal.GatewayDataHolder; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import io.jsonwebtoken.Jwts; -import io.jsonwebtoken.SignatureAlgorithm; -import org.apache.commons.io.IOUtils; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.HttpResponse; -import org.apache.http.client.methods.HttpPost; -import org.apache.http.entity.StringEntity; -import org.json.JSONObject; - -import java.io.FileInputStream; -import java.io.IOException; -import java.io.InputStream; -import java.nio.charset.StandardCharsets; -import java.security.Key; -import java.security.KeyStore; -import java.security.KeyStoreException; -import java.security.NoSuchAlgorithmException; -import java.security.UnrecoverableKeyException; -import java.security.cert.CertificateException; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Map; - -/** - * Consent Enforcement executor. - */ -public class ConsentEnforcementExecutor implements OpenBankingGatewayExecutor { - - protected static final String ERROR_TITLE = "Consent Enforcement Error"; - protected static final String HEADERS_TAG = "headers"; - protected static final String BODY_TAG = "body"; - protected static final String CONTEXT_TAG = "context"; - protected static final String RESOURCE_TAG = "resource"; - protected static final String ELECTED_RESOURCE_TAG = "electedResource"; - protected static final String HTTP_METHOD = "httpMethod"; - protected static final String CONSENT_ID_TAG = "consentId"; - protected static final String USER_ID_TAG = "userId"; - protected static final String CLIENT_ID_TAG = "clientId"; - protected static final String RESOURCE_PARAMS = "resourceParams"; - private static final Log log = LogFactory.getLog(ConsentEnforcementExecutor.class); - private static final GatewayDataHolder dataHolder = GatewayDataHolder.getInstance(); - private static final String INFO_HEADER_TAG = "Account-Request-Information"; - private static final String IS_VALID = "isValid"; - private static final String ERROR_CODE = "errorCode"; - private static final String ERROR_MESSAGE = "errorMessage"; - private static final String HTTP_CODE = "httpCode"; - private static final String MODIFIED_PAYLOAD = "modifiedPayload"; - private static final String CONSENT_INFO = "consentInformation"; - private static volatile String consentValidationEndpoint; - private static volatile Key key; - - private static String getValidationEndpoint() { - - if (consentValidationEndpoint == null) { - synchronized (ConsentEnforcementExecutor.class) { - if (consentValidationEndpoint == null) { - consentValidationEndpoint = dataHolder - .getOpenBankingConfigurationService().getConfigurations() - .get(GatewayConstants.CONSENT_VALIDATION_ENDPOINT_TAG).toString(); - } - } - } - return consentValidationEndpoint; - - } - - /** - * Method to obtain signing key. - * - * @return Key as an Object. - */ - @SuppressFBWarnings("PATH_TRAVERSAL_IN") - // Suppressed content - dataHolder.getKeyStoreLocation() - // Suppression reason - False Positive : Keystore location is obtained from deployment.toml. So it can be marked - // as a trusted filepath - // Suppressed warning count - 1 - protected static Key getJWTSigningKey() { - - if (key == null) { - synchronized (ConsentEnforcementExecutor.class) { - if (key == null) { - try (FileInputStream is = new FileInputStream(dataHolder.getKeyStoreLocation())) { - KeyStore keystore = KeyStore.getInstance(KeyStore.getDefaultType()); - keystore.load(is, dataHolder.getKeyStorePassword()); - key = keystore.getKey(dataHolder.getKeyAlias(), dataHolder.getKeyPassword().toCharArray()); - } catch (IOException | CertificateException | KeyStoreException | NoSuchAlgorithmException - | UnrecoverableKeyException e) { - log.error("Error occurred while retrieving private key from keystore ", e); - } - } - } - } - return key; - } - - /** - * Method to handle request. - * - * @param obapiRequestContext OB request context object - */ - @Generated(message = "Unit testable components are covered") - @Override - public void preProcessRequest(OBAPIRequestContext obapiRequestContext) { - - } - - /** - * Method to handle response. - * - * @param obapiResponseContext OB response context object - */ - @Override - public void preProcessResponse(OBAPIResponseContext obapiResponseContext) { - - } - - protected String generateJWT(String payload) { - - return Jwts.builder() - .setPayload(payload) - .signWith(SignatureAlgorithm.RS512, getJWTSigningKey()) - .compact(); - } - - /** - * Method to invoke consent validation service when the JWT payload is provided. - * - * @param enforcementJWTPayload JWT Payload - * @return Response as a String - * @throws IOException When failed to invoke the validation endpoint or failed to parse the response. - */ - @Generated(message = "Ignoring from unit tests since this method require calling external component to function") - private String invokeConsentValidationService(String enforcementJWTPayload) throws IOException, - OpenBankingException { - - HttpPost httpPost = new HttpPost(getValidationEndpoint()); - StringEntity params; - params = new StringEntity(enforcementJWTPayload); - httpPost.setEntity(params); - httpPost.setHeader(GatewayConstants.CONTENT_TYPE_TAG, GatewayConstants.JWT_CONTENT_TYPE); - String userName = GatewayUtils.getAPIMgtConfig(GatewayConstants.API_KEY_VALIDATOR_USERNAME); - String password = GatewayUtils.getAPIMgtConfig(GatewayConstants.API_KEY_VALIDATOR_PASSWORD); - httpPost.setHeader(GatewayConstants.AUTH_HEADER, GatewayUtils.getBasicAuthHeader(userName, password)); - HttpResponse response = GatewayDataHolder.getHttpClient().execute(httpPost); - InputStream in = response.getEntity().getContent(); - return IOUtils.toString(in, String.valueOf(StandardCharsets.UTF_8)); - } - - /** - * Method to handle errors. - * - * @param obapiRequestContext API Context - * @param errorCode Error Code - * @param errorMessage Error Message - * @param httpCode HTTP status code ( in 4XX range) - */ - protected void handleError(OBAPIRequestContext obapiRequestContext, String errorCode, String errorMessage, - String httpCode) { - - obapiRequestContext.setError(true); - ArrayList errors = obapiRequestContext.getErrors(); - errors.add(new OpenBankingExecutorError(errorCode, ERROR_TITLE, errorMessage, httpCode)); - obapiRequestContext.setErrors(errors); - obapiRequestContext.addContextProperty(GatewayConstants.ERROR_STATUS_PROP, httpCode); - } - - /** - * Method to create validation payload. - * - * @param requestHeaders Request headers of original request - * @param requestPayload Request payload of original request - * @return JSON Object with added attributes. - */ - protected JSONObject createValidationRequestPayload(Map requestHeaders, String requestPayload, - Map additionalParams) { - - JSONObject validationRequest = new JSONObject(); - JSONObject headers = new JSONObject(); - requestHeaders.forEach(headers::put); - validationRequest.put(HEADERS_TAG, headers); - /*requestContextDTO.getMsgInfo().getPayloadHandler().consumeAsString() method sets the request payload as a - null string, hence adding string null check to the validation*/ - if (requestPayload != null && !requestPayload.isEmpty() && !requestPayload.equals("null")) { - //This assumes all input payloads are in Content-Type : Application/JSON - validationRequest.put(BODY_TAG, new JSONObject(requestPayload)); - } - additionalParams.forEach(validationRequest::put); - return validationRequest; - } - - /** - * Method to handle post response. - * - * @param obapiResponseContext OB response context object - */ - @Override - public void postProcessResponse(OBAPIResponseContext obapiResponseContext) { - - } - - /** - * Method to handle post request. - * - * @param obapiRequestContext OB request context object - */ - @Override - public void postProcessRequest(OBAPIRequestContext obapiRequestContext) { - // Consent ID is required for consent enforcement. If the consent ID is null, we are assume this is a - // pre-consent creation call. Therefore consent enforcement is not required. - if (obapiRequestContext.isError() || obapiRequestContext.getConsentId() == null) { - return; - } - - Map requestHeaders = obapiRequestContext.getMsgInfo().getHeaders(); - Map additionalParams = new HashMap<>(); - additionalParams.put(ELECTED_RESOURCE_TAG, obapiRequestContext.getMsgInfo().getElectedResource()); - additionalParams.put(CONSENT_ID_TAG, obapiRequestContext.getConsentId()); - additionalParams.put(USER_ID_TAG, obapiRequestContext.getApiRequestInfo().getUsername()); - additionalParams.put(CLIENT_ID_TAG, obapiRequestContext.getApiRequestInfo().getConsumerKey()); - additionalParams.put(RESOURCE_PARAMS, getResourceParamMap(obapiRequestContext)); - - JSONObject validationRequest; - if (StringUtils.isNotBlank(obapiRequestContext.getModifiedPayload())) { - validationRequest = createValidationRequestPayload(requestHeaders, - obapiRequestContext.getModifiedPayload(), additionalParams); - } else { - validationRequest = createValidationRequestPayload(requestHeaders, - obapiRequestContext.getRequestPayload(), additionalParams); - } - String enforcementJWTPayload = generateJWT(validationRequest.toString()); - JSONObject jsonResponse; - try { - String response = invokeConsentValidationService(enforcementJWTPayload); - jsonResponse = new JSONObject(response); - } catch (IOException | OpenBankingException e) { - handleError(obapiRequestContext, OpenBankingErrorCodes.CONSENT_VALIDATION_REQUEST_FAILURE, e.getMessage(), - OpenBankingErrorCodes.SERVER_ERROR_CODE); - return; - } - - boolean isValid = (boolean) jsonResponse.get(IS_VALID); - if (!isValid) { - String errorCode = jsonResponse.get(ERROR_CODE).toString(); - String errorMessage = jsonResponse.get(ERROR_MESSAGE).toString(); - String httpCode = jsonResponse.get(HTTP_CODE).toString(); - obapiRequestContext.setError(true); - handleError(obapiRequestContext, errorCode, errorMessage, httpCode); - return; - } else if (!jsonResponse.isNull(MODIFIED_PAYLOAD)) { - Object modifiedPayloadObj = jsonResponse.get(MODIFIED_PAYLOAD); - if (modifiedPayloadObj != null) { - obapiRequestContext.setModifiedPayload(modifiedPayloadObj.toString()); - } - } else if (!jsonResponse.isNull(CONSENT_INFO)) { - Object consentInformationObj = jsonResponse.get(CONSENT_INFO); - if (consentInformationObj != null) { - requestHeaders.put(INFO_HEADER_TAG, consentInformationObj.toString()); - obapiRequestContext.setAddedHeaders(requestHeaders); - } - } - } - - /** - * Method to construct resource parameter map to invoke the validation service. - * - * @param obapiRequestContext - * @return A Map containing resource path(ex: /aisp/accounts/{AccountId}?queryParam=urlEncodedQueryParamValue), - * http method and context(ex: /open-banking/v3.1/aisp) - */ - private Map getResourceParamMap(OBAPIRequestContext obapiRequestContext) { - - Map resourceMap = new HashMap(); - resourceMap.put(RESOURCE_TAG, obapiRequestContext.getMsgInfo().getResource()); - resourceMap.put(HTTP_METHOD, obapiRequestContext.getMsgInfo().getHttpMethod()); - resourceMap.put(CONTEXT_TAG, obapiRequestContext.getApiRequestInfo().getContext()); - - return resourceMap; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/error/handler/OBDefaultErrorHandler.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/error/handler/OBDefaultErrorHandler.java deleted file mode 100644 index 5ffd258c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/error/handler/OBDefaultErrorHandler.java +++ /dev/null @@ -1,193 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.impl.error.handler; - -import com.wso2.openbanking.accelerator.gateway.executor.core.OpenBankingGatewayExecutor; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OpenBankingExecutorError; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import org.apache.http.HttpStatus; -import org.json.JSONArray; -import org.json.JSONObject; - -import java.util.ArrayList; -import java.util.HashSet; -import java.util.List; -import java.util.Map; - -/** - * Default Executor to handle gateway errors. - */ -public class OBDefaultErrorHandler implements OpenBankingGatewayExecutor { - - private static final String ERRORS_TAG = "errors"; - private static final String STATUS_CODE = "statusCode"; - private static final String RESPONSE_PAYLOAD_SIZE = "responsePayloadSize"; - - /** - * Method to handle pre request. - * - * @param obapiRequestContext OB request context object - */ - @Override - public void preProcessRequest(OBAPIRequestContext obapiRequestContext) { - - handleRequestError(obapiRequestContext); - - } - - /** - * Method to handle post request. - * - * @param obapiRequestContext OB request context object - */ - @Override - public void postProcessRequest(OBAPIRequestContext obapiRequestContext) { - - handleRequestError(obapiRequestContext); - } - - /** - * Method to handle pre response. - * - * @param obapiResponseContext OB response context object - */ - @Override - public void preProcessResponse(OBAPIResponseContext obapiResponseContext) { - - handleResponseError(obapiResponseContext); - } - - /** - * Method to handle post response. - * - * @param obapiResponseContext OB response context object - */ - @Override - public void postProcessResponse(OBAPIResponseContext obapiResponseContext) { - - handleResponseError(obapiResponseContext); - } - - private void handleRequestError(OBAPIRequestContext obapiRequestContext) { - - if (!obapiRequestContext.isError()) { - return; - } - JSONObject payload = new JSONObject(); - ArrayList errors = obapiRequestContext.getErrors(); - JSONArray errorList = getErrorJSON(errors); - HashSet statusCodes = new HashSet<>(); - - for (OpenBankingExecutorError error : errors) { - statusCodes.add(error.getHttpStatusCode()); - } - - payload.put(ERRORS_TAG, errorList); - if (errorList.length() != 0) { - obapiRequestContext.setModifiedPayload(payload.toString()); - Map addedHeaders = obapiRequestContext.getAddedHeaders(); - addedHeaders.put(GatewayConstants.CONTENT_TYPE_TAG, GatewayConstants.JSON_CONTENT_TYPE); - obapiRequestContext.setAddedHeaders(addedHeaders); - } - int statusCode; - if (obapiRequestContext.getContextProps().containsKey(GatewayConstants.ERROR_STATUS_PROP)) { - statusCode = Integer.parseInt(obapiRequestContext.getContextProperty(GatewayConstants.ERROR_STATUS_PROP)); - } else if (isAnyClientErrors(statusCodes)) { - statusCode = HttpStatus.SC_BAD_REQUEST; - } else { - statusCode = HttpStatus.SC_INTERNAL_SERVER_ERROR; - } - obapiRequestContext.addContextProperty(GatewayConstants.ERROR_STATUS_PROP, - String.valueOf(statusCode)); - - // Add error data to analytics map - Map analyticsData = obapiRequestContext.getAnalyticsData(); - analyticsData.put(STATUS_CODE, statusCode); - analyticsData.put(RESPONSE_PAYLOAD_SIZE, (long) payload.toString().length()); - obapiRequestContext.setAnalyticsData(analyticsData); - } - - private void handleResponseError(OBAPIResponseContext obapiResponseContext) { - - if (!obapiResponseContext.isError()) { - return; - } - JSONObject payload = new JSONObject(); - ArrayList errors = obapiResponseContext.getErrors(); - JSONArray errorList = getErrorJSON(errors); - HashSet statusCodes = new HashSet<>(); - - for (OpenBankingExecutorError error : errors) { - statusCodes.add(error.getHttpStatusCode()); - } - - payload.put(ERRORS_TAG, errorList); - obapiResponseContext.setModifiedPayload(payload.toString()); - Map addedHeaders = obapiResponseContext.getAddedHeaders(); - addedHeaders.put(GatewayConstants.CONTENT_TYPE_TAG, GatewayConstants.JSON_CONTENT_TYPE); - obapiResponseContext.setAddedHeaders(addedHeaders); - int statusCode; - if (obapiResponseContext.getContextProps().containsKey(GatewayConstants.ERROR_STATUS_PROP)) { - statusCode = Integer.parseInt(obapiResponseContext.getContextProperty(GatewayConstants.ERROR_STATUS_PROP)); - } else if (isAnyClientErrors(statusCodes)) { - statusCode = HttpStatus.SC_BAD_REQUEST; - } else { - statusCode = HttpStatus.SC_INTERNAL_SERVER_ERROR; - } - obapiResponseContext.addContextProperty(GatewayConstants.ERROR_STATUS_PROP, - String.valueOf(statusCode)); - - // Add error data to analytics map - Map analyticsData = obapiResponseContext.getAnalyticsData(); - analyticsData.put(STATUS_CODE, statusCode); - analyticsData.put(RESPONSE_PAYLOAD_SIZE, (long) payload.toString().length()); - obapiResponseContext.setAnalyticsData(analyticsData); - } - - private JSONArray getErrorJSON(List errors) { - - JSONArray errorList = new JSONArray(); - for (OpenBankingExecutorError error : errors) { - JSONObject errorObj = new JSONObject(); - errorObj.put("Code", error.getCode()); - errorObj.put("Title", error.getTitle()); - errorObj.put("Message", error.getMessage()); - Map links = error.getLinks(); - if (links != null && links.size() > 0) { - JSONObject linksObj = new JSONObject(); - links.forEach(linksObj::put); - errorObj.put("Links", linksObj); - } - errorList.put(errorObj); - } - return errorList; - } - - private boolean isAnyClientErrors(HashSet statusCodes) { - - for (String statusCode : statusCodes) { - if (statusCode.startsWith("4")) { - return true; - } - } - return false; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/mtls/cert/validation/executor/CertRevocationValidationExecutor.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/mtls/cert/validation/executor/CertRevocationValidationExecutor.java deleted file mode 100644 index f6ba7956..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/mtls/cert/validation/executor/CertRevocationValidationExecutor.java +++ /dev/null @@ -1,260 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.impl.mtls.cert.validation.executor; - -import com.wso2.openbanking.accelerator.common.error.OpenBankingErrorCodes; -import com.wso2.openbanking.accelerator.common.exception.CertificateValidationException; -import com.wso2.openbanking.accelerator.common.util.CertificateUtils; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.gateway.cache.CertificateRevocationCache; -import com.wso2.openbanking.accelerator.gateway.cache.GatewayCacheKey; -import com.wso2.openbanking.accelerator.gateway.executor.core.OpenBankingGatewayExecutor; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OpenBankingExecutorError; -import com.wso2.openbanking.accelerator.gateway.executor.service.CertValidationService; -import com.wso2.openbanking.accelerator.gateway.executor.util.CertificateValidationUtils; -import com.wso2.openbanking.accelerator.gateway.internal.TPPCertValidatorDataHolder; -import org.apache.commons.codec.digest.DigestUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.security.cert.Certificate; -import java.security.cert.CertificateEncodingException; -import java.security.cert.CertificateException; -import java.security.cert.X509Certificate; -import java.util.List; -import java.util.Optional; - -/** - * This executor will be used to validate the certificate revocation (CRL and OCSP validation) of the client - * certificate during a mutual tls session. The immediate issuer of the client certificate must be present in the - * truststore to continue with the validation. - */ -public class CertRevocationValidationExecutor implements OpenBankingGatewayExecutor { - - private static final Log LOG = LogFactory.getLog(CertRevocationValidationExecutor.class); - - @Generated(message = "Ignoring since all cases are covered from other unit tests") - @Override - public void preProcessRequest(OBAPIRequestContext obapiRequestContext) { - - LOG.info("Starting certificate revocation validation process"); - - // Skip the executor if previous executors failed. - if (obapiRequestContext.isError()) { - return; - } - - try { - Certificate[] clientCerts = obapiRequestContext.getClientCertsLatest(); - // enforcement executor validates the certificate presence - if (clientCerts != null && clientCerts.length > 0) { - Optional transportCert = - CertificateValidationUtils.convertCertToX509Cert(clientCerts[0]); - - if (!transportCert.isPresent()) { - LOG.error("Invalid mutual TLS request. Client certificate is invalid"); - OpenBankingExecutorError error = new OpenBankingExecutorError( - OpenBankingErrorCodes.INVALID_MTLS_CERT_CODE, - "Invalid mutual TLS request. Client certificate is invalid", - "", OpenBankingErrorCodes.UNAUTHORIZED_CODE); - - CertificateValidationUtils.handleExecutorErrors(error, obapiRequestContext); - } else { - X509Certificate transportCertificate = transportCert.get(); - if (CertificateUtils.isExpired(transportCertificate)) { - LOG.error("Certificate with the serial number " + - transportCertificate.getSerialNumber() + " issued by the CA " + - transportCertificate.getIssuerDN().toString() + " is expired"); - OpenBankingExecutorError error = new OpenBankingExecutorError( - OpenBankingErrorCodes.EXPIRED_MTLS_CERT_CODE, - "Invalid mutual TLS request. Client certificate is expired", - "Certificate with the serial number " + - transportCertificate.getSerialNumber() + " issued by the CA " + - transportCertificate.getIssuerDN().toString() + " is expired", - OpenBankingErrorCodes.UNAUTHORIZED_CODE); - - CertificateValidationUtils.handleExecutorErrors(error, obapiRequestContext); - } else { - LOG.debug("Client certificate expiry validation completed successfully"); - if (isCertRevoked(transportCertificate)) { - LOG.error("Invalid mutual TLS request. Client certificate is revoked"); - OpenBankingExecutorError error = new OpenBankingExecutorError( - OpenBankingErrorCodes.REVOKED_MTLS_CERT_CODE, - "Invalid mutual TLS request. Client certificate is revoked", - "", OpenBankingErrorCodes.UNAUTHORIZED_CODE); - - CertificateValidationUtils.handleExecutorErrors(error, obapiRequestContext); - } else { - LOG.debug("Certificate revocation validation success"); - } - } - } - } - } catch (CertificateValidationException e) { - LOG.error("Unable to validate the client certificate, caused by ", e); - - //catch errors and set to context - CertificateValidationUtils.handleExecutorErrors(e, obapiRequestContext); - } catch (CertificateEncodingException e) { - LOG.error("Unable to generate the client certificate thumbprint, caused by ", e); - OpenBankingExecutorError error = new OpenBankingExecutorError( - OpenBankingErrorCodes.INVALID_MTLS_CERT_CODE, - "Unable to generate the client certificate thumbprint", - "", OpenBankingErrorCodes.UNAUTHORIZED_CODE); - - //catch errors and set to context - CertificateValidationUtils.handleExecutorErrors(error, obapiRequestContext); - } catch (CertificateException e) { - String errorMsg = "Error occurred while converting the client certificate to X509Certificate "; - LOG.error(errorMsg, e); - OpenBankingExecutorError error = new OpenBankingExecutorError( - OpenBankingErrorCodes.INVALID_MTLS_CERT_CODE, errorMsg, - e.getMessage(), OpenBankingErrorCodes.UNAUTHORIZED_CODE); - CertificateValidationUtils.handleExecutorErrors(error, obapiRequestContext); - } - } - - @Override - public void preProcessResponse(OBAPIResponseContext obapiResponseContext) { - // Do not need to handle the response - } - - /** - * Checks the certificate validity of a given certificate. For this validation, the immediate issuer - * of the peer certificate must be present in the trust store. - * JSONObject jsonObject; - * - * @param peerCertificate peer certificate - * @return validity of the certificate - * @throws CertificateValidationException when an error occurs while validating the certificate - */ - private boolean isCertRevoked(X509Certificate peerCertificate) - throws CertificateValidationException, CertificateEncodingException { - - // Initializing certificate cache and cache key - CertificateRevocationCache certificateRevocationCache = CertificateRevocationCache.getInstance(); - // Generating the certificate thumbprint to use as cache key - String certificateValidationCacheKeyStr = DigestUtils.sha256Hex(peerCertificate.getEncoded()); - GatewayCacheKey certificateValidationCacheKey = - GatewayCacheKey.of(certificateValidationCacheKeyStr); - - // Executing certificate revocation process or retrieve last status from cache - if (certificateRevocationCache.getFromCache(certificateValidationCacheKey) != null) { - // previous result is present in cache, return result - return !certificateRevocationCache.getFromCache(certificateValidationCacheKey); - } else { - final boolean result = isCertRevocationSuccess(peerCertificate); - if (result) { - // Adding result to cache - certificateRevocationCache.addToCache(certificateValidationCacheKey, true); - return false; - } - } - return true; - } - - private boolean isCertRevocationSuccess(X509Certificate peerCertificate) { - - TPPCertValidatorDataHolder tppCertValidatorDataHolder = TPPCertValidatorDataHolder.getInstance(); - - Integer certificateRevocationValidationRetryCount = - tppCertValidatorDataHolder.getCertificateRevocationValidationRetryCount(); - - int connectTimeout = tppCertValidatorDataHolder.getConnectTimeout(); - int connectionRequestTimeout = tppCertValidatorDataHolder.getConnectionRequestTimeout(); - int socketTimeout = tppCertValidatorDataHolder.getSocketTimeout(); - - boolean isValid; - // Check certificate revocation status. - if (tppCertValidatorDataHolder.isCertificateRevocationValidationEnabled()) { - LOG.debug("Client certificate revocation validation is enabled"); - - // Skip certificate revocation validation if the certificate is self-signed. - if (peerCertificate.getSubjectDN().getName().equals(peerCertificate.getIssuerDN().getName())) { - if (LOG.isDebugEnabled()) { - LOG.debug("Client certificate is self signed. Hence, excluding the certificate revocation" + - " validation"); - } - return true; - } - - /* - * Skip certificate revocation validation if the certificate issuer is listed to exclude from - * revocation validation in open-banking.xml under - * CertificateManagement.RevocationValidationExcludedIssuers configuration. - * - * This option can be used to skip certificate revocation validation for certificates which have been - * issued by a trusted locally generated CA. - */ - List revocationValidationExcludedIssuers = - tppCertValidatorDataHolder.getCertificateRevocationValidationExcludedIssuers(); - if (revocationValidationExcludedIssuers.contains(peerCertificate.getIssuerDN().getName())) { - if (LOG.isDebugEnabled()) { - LOG.debug("The issuer of the client certificate has been configured to exclude from " + - "certificate revocation validation. Hence, excluding the certificate " + - "revocation validation"); - } - return true; - } - - // Get issuer certificate from the truststore to continue with the certificate validation. - X509Certificate issuerCertificate; - try { - issuerCertificate = CertificateValidationUtils - .getIssuerCertificateFromTruststore(peerCertificate); - } catch (CertificateValidationException e) { - LOG.error("Issuer certificate retrieving failed for client certificate with" + - " serial number " + peerCertificate.getSerialNumber() + " issued by the CA " + - peerCertificate.getIssuerDN().toString(), e); - return false; - } - - isValid = CertValidationService.getInstance().verify(peerCertificate, issuerCertificate, - certificateRevocationValidationRetryCount, connectTimeout, connectionRequestTimeout, socketTimeout); - } else { - isValid = true; - } - - LOG.debug("Stored certificate validation status in cache"); - - return isValid; - } - - /** - * Method to handle post response. - * - * @param obapiResponseContext OB response context object - */ - @Override - public void postProcessResponse(OBAPIResponseContext obapiResponseContext) { - - } - - /** - * Method to handle post request. - * - * @param obapiRequestContext OB request context object - */ - @Override - public void postProcessRequest(OBAPIRequestContext obapiRequestContext) { - - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/mtls/cert/validation/executor/MTLSEnforcementExecutor.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/mtls/cert/validation/executor/MTLSEnforcementExecutor.java deleted file mode 100644 index fa5d0efa..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/mtls/cert/validation/executor/MTLSEnforcementExecutor.java +++ /dev/null @@ -1,116 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.impl.mtls.cert.validation.executor; - -import com.wso2.openbanking.accelerator.common.error.OpenBankingErrorCodes; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.gateway.executor.core.OpenBankingGatewayExecutor; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OpenBankingExecutorError; -import com.wso2.openbanking.accelerator.gateway.executor.util.CertificateValidationUtils; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.security.cert.Certificate; -import java.security.cert.CertificateException; -import java.security.cert.X509Certificate; -import java.util.Optional; - -/** - * Mutual TLS Enforcement Executor - * Enforces whether the Request is sent with MTLS cert as a header. - */ -public class MTLSEnforcementExecutor implements OpenBankingGatewayExecutor { - - private static final Log LOG = LogFactory.getLog(MTLSEnforcementExecutor.class); - - @Generated(message = "Ignoring since all cases are covered from other unit tests") - @Override - public void preProcessRequest(OBAPIRequestContext obapiRequestContext) { - - LOG.info("Starting mutual TLS enforcement process"); - - // Skip the executor if previous executors failed. - if (obapiRequestContext.isError()) { - return; - } - - Certificate[] clientCerts = obapiRequestContext.getClientCertsLatest(); - if (clientCerts != null && clientCerts.length > 0) { - Optional transportCert = Optional.empty(); - try { - transportCert = CertificateValidationUtils.convertCertToX509Cert(clientCerts[0]); - } catch (CertificateException e) { - String errorMsg = "Error occurred while converting the client certificate to X509Certificate "; - LOG.error(errorMsg, e); - OpenBankingExecutorError error = new OpenBankingExecutorError( - OpenBankingErrorCodes.INVALID_MTLS_CERT_CODE, errorMsg, - e.getMessage(), OpenBankingErrorCodes.UNAUTHORIZED_CODE); - CertificateValidationUtils.handleExecutorErrors(error, obapiRequestContext); - } - - if (transportCert.isPresent()) { - LOG.debug("Mutual TLS enforcement success"); - } else { - LOG.error(GatewayConstants.CLIENT_CERTIFICATE_INVALID); - OpenBankingExecutorError error = new OpenBankingExecutorError( - OpenBankingErrorCodes.INVALID_MTLS_CERT_CODE, GatewayConstants.INVALID_CLIENT, - GatewayConstants.CLIENT_CERTIFICATE_INVALID, OpenBankingErrorCodes.UNAUTHORIZED_CODE); - - CertificateValidationUtils.handleExecutorErrors(error, obapiRequestContext); - } - } else { - LOG.error(GatewayConstants.CLIENT_CERTIFICATE_MISSING); - OpenBankingExecutorError error = new OpenBankingExecutorError( - OpenBankingErrorCodes.MISSING_MTLS_CERT_CODE, GatewayConstants.INVALID_CLIENT, - GatewayConstants.CLIENT_CERTIFICATE_MISSING, OpenBankingErrorCodes.UNAUTHORIZED_CODE); - - CertificateValidationUtils.handleExecutorErrors(error, obapiRequestContext); - } - - } - - @Override - public void preProcessResponse(OBAPIResponseContext obapiResponseContext) { - // Do not need to handle the response - } - - /** - * Method to handle post response. - * - * @param obapiResponseContext OB response context object - */ - @Override - public void postProcessResponse(OBAPIResponseContext obapiResponseContext) { - - } - - /** - * Method to handle post request. - * - * @param obapiRequestContext OB request context object - */ - @Override - public void postProcessRequest(OBAPIRequestContext obapiRequestContext) { - - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/selfcare/portal/UserPermissionValidationExecutor.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/selfcare/portal/UserPermissionValidationExecutor.java deleted file mode 100644 index 1336f90f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/selfcare/portal/UserPermissionValidationExecutor.java +++ /dev/null @@ -1,173 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.impl.selfcare.portal; - -import com.wso2.openbanking.accelerator.common.error.OpenBankingErrorCodes; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.gateway.executor.core.OpenBankingGatewayExecutor; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OpenBankingExecutorError; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import net.minidev.json.JSONObject; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.HttpHeaders; - -import java.text.ParseException; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Map; -import java.util.Optional; - -/** - * UserPermissionValidationExecutor. - *

- * Validates access token scopes against users - */ -public class UserPermissionValidationExecutor implements OpenBankingGatewayExecutor { - - private static final Log LOG = LogFactory.getLog(UserPermissionValidationExecutor.class); - - @Generated(message = "Ignoring since all cases are covered from other unit tests") - @Override - public void preProcessRequest(OBAPIRequestContext obapiRequestContext) { - try { - // Skip the executor if previous executors failed. - if (obapiRequestContext.isError()) { - return; - } - - String authToken = obapiRequestContext.getMsgInfo().getHeaders().get(HttpHeaders.AUTHORIZATION); - JSONObject tokenBody = JWTUtils.decodeRequestJWT(authToken.replace("Bearer ", ""), "body"); - String tokenScopes = tokenBody.getAsString("scope"); - - if (!isCustomerCareOfficer(tokenScopes)) { - // user is not a customer care officer - Optional optUserId = getUserIdsFromQueryParams(obapiRequestContext.getMsgInfo().getResource()); - String tokenSubject = GatewayUtils.getUserNameWithTenantDomain(tokenBody.getAsString("sub")); - if (!optUserId.isPresent() || !isUserIdMatchesTokenSub(optUserId.get(), tokenSubject)) { - // token subject and user id do not match, invalid request - final String errorMsg = "Invalid self care portal request received. " + - "UserId and token subject do not match."; - LOG.error(errorMsg + " userIDs: " + optUserId.orElse(" ") + " sub: " + tokenSubject); - OpenBankingExecutorError error = new OpenBankingExecutorError( - OpenBankingErrorCodes.SCP_USER_VALIDATION_FAILED_CODE, "Unauthorized Request", errorMsg, - OpenBankingErrorCodes.UNAUTHORIZED_CODE); - - obapiRequestContext.setError(true); - - ArrayList executorErrors = obapiRequestContext.getErrors(); - executorErrors.add(error); - obapiRequestContext.setErrors(executorErrors); - - Map contextProps = new HashMap<>(); - contextProps.put(GatewayConstants.ERROR_STATUS_PROP, OpenBankingErrorCodes.UNAUTHORIZED_CODE); - obapiRequestContext.setContextProps(contextProps); - } - } - } catch (ParseException e) { - final String errorMsg = "Error occurred while validating self care portal user permissions"; - - LOG.error(errorMsg + ". Caused by, ", e); - //catch errors and set to context - OpenBankingExecutorError error = new OpenBankingExecutorError( - OpenBankingErrorCodes.SCP_USER_VALIDATION_FAILED_CODE, e.getMessage(), errorMsg, - OpenBankingErrorCodes.BAD_REQUEST_CODE); - - ArrayList executorErrors = obapiRequestContext.getErrors(); - executorErrors.add(error); - - obapiRequestContext.setError(true); - obapiRequestContext.setErrors(executorErrors); - } - - } - - @Generated(message = "Ignoring since empty") - @Override - public void postProcessRequest(OBAPIRequestContext obapiRequestContext) { - // do not need to handle - } - - @Generated(message = "Ignoring since empty") - @Override - public void preProcessResponse(OBAPIResponseContext obapiResponseContext) { - // do not need to handle - } - - @Generated(message = "Ignoring since empty") - @Override - public void postProcessResponse(OBAPIResponseContext obapiResponseContext) { - // do not need to handle - } - - /** - * Method to extract userID from the request URL. - * - * @param url requested URL - * @return Optional String: if userID found return userID, else return empty - */ - protected Optional getUserIdsFromQueryParams(String url) { - if (StringUtils.isNotEmpty(url) && url.contains("?")) { - final String queryParams = url.split("\\?")[1]; - final String[] queryParamPairs = queryParams.split("&"); - - for (String pair : queryParamPairs) { - if (pair.contains("userIDs") || pair.contains("userID")) { - final String[] userIds = pair.split("="); - if (userIds.length > 1) { // to prevent indexOutOfBoundException - return Optional.of(GatewayUtils.getUserNameWithTenantDomain(userIds[1])); - } - } - } - } - return Optional.empty(); - } - - /** - * Method to match customer care officer scopes. - * - * @param scopes scopes received from access token - * @return if customer care officer scope found return true else false - */ - protected boolean isCustomerCareOfficer(String scopes) { - if (StringUtils.isNotEmpty(scopes)) { - return scopes.contains(GatewayConstants.CUSTOMER_CARE_OFFICER_SCOPE); - } - return false; - } - - /** - * Method to match user id and token subject. - * - * @param userId received from query parameter - * @param tokenSub received from access token body - * @return if user id matches with token subject return true else false - */ - protected boolean isUserIdMatchesTokenSub(String userId, String tokenSub) { - if (StringUtils.isNotEmpty(tokenSub)) { - return tokenSub.equalsIgnoreCase(userId); - } - return false; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/tpp/validation/executor/APITPPValidationExecutor.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/tpp/validation/executor/APITPPValidationExecutor.java deleted file mode 100644 index af4b24e9..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/tpp/validation/executor/APITPPValidationExecutor.java +++ /dev/null @@ -1,196 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.impl.tpp.validation.executor; - -import com.wso2.openbanking.accelerator.common.error.OpenBankingErrorCodes; -import com.wso2.openbanking.accelerator.common.exception.CertificateValidationException; -import com.wso2.openbanking.accelerator.common.exception.TPPValidationException; -import com.wso2.openbanking.accelerator.common.model.PSD2RoleEnum; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.gateway.executor.core.OpenBankingGatewayExecutor; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OpenBankingExecutorError; -import com.wso2.openbanking.accelerator.gateway.executor.service.CertValidationService; -import com.wso2.openbanking.accelerator.gateway.executor.util.CertificateValidationUtils; -import com.wso2.openbanking.accelerator.gateway.internal.GatewayDataHolder; -import io.swagger.v3.oas.models.PathItem; -import io.swagger.v3.oas.models.security.SecurityRequirement; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.security.cert.Certificate; -import java.security.cert.CertificateException; -import java.security.cert.X509Certificate; -import java.util.ArrayList; -import java.util.HashSet; -import java.util.List; -import java.util.Map; -import java.util.Optional; -import java.util.Set; - -/** - * TPP validation handler used to validate the TPP status using external validation - * services for regular API requests. - */ -public class APITPPValidationExecutor implements OpenBankingGatewayExecutor { - - private static final String GET = "GET"; - private static final String POST = "POST"; - private static final String PUT = "PUT"; - private static final String PATCH = "PATCH"; - private static final String DELETE = "DELETE"; - private static final Log log = LogFactory.getLog(APITPPValidationExecutor.class); - - @Override - public void preProcessRequest(OBAPIRequestContext obapiRequestContext) { - - } - - @Override - public void preProcessResponse(OBAPIResponseContext obapiResponseContext) { - // Do not need to handle the response - } - - /** - * Method to handle post response. - * - * @param obapiResponseContext OB response context object - */ - @Override - public void postProcessResponse(OBAPIResponseContext obapiResponseContext) { - - } - - /** - * Method to handle post request. - * - * @param obapiRequestContext OB request context object - */ - @Generated(message = "Ignoring since all cases are covered from other unit tests") - @Override - public void postProcessRequest(OBAPIRequestContext obapiRequestContext) { - // Skip the executor if previous executors failed. - if (obapiRequestContext.isError()) { - return; - } - - try { - Certificate[] clientCerts = obapiRequestContext.getClientCertsLatest(); - if (clientCerts != null && clientCerts.length > 0) { - Optional transportCert = - CertificateValidationUtils.convertCertToX509Cert(clientCerts[0]); - - // Only Do Validation if Mutual TLS is used. - if (transportCert.isPresent()) { - - // extracting scopes from api swagger - final PathItem electedPath = obapiRequestContext.getOpenAPI().getPaths() - .get(obapiRequestContext.getMsgInfo().getElectedResource()); - final String httpMethod = obapiRequestContext.getMsgInfo().getHttpMethod(); - - final Set scopes = extractScopesFromSwaggerAPI(electedPath, httpMethod); - - // retrieving allowed scopes from open-banking.xml - final Map> allowedScopes = GatewayDataHolder.getInstance() - .getOpenBankingConfigurationService().getAllowedScopes(); - - List requiredPSD2Roles = getRolesFromScopes(allowedScopes, scopes); - - if (requiredPSD2Roles.isEmpty()) { - throw new TPPValidationException("No roles found associated with the request. Hence, cannot " + - "continue with TPP validation"); - } - - if (CertValidationService.getInstance().validateTppRoles(transportCert.get(), requiredPSD2Roles)) { - log.debug("TPP validation service returned a success response"); - } else { - log.error("TPP validation service returned invalid TPP status"); - throw new TPPValidationException("TPP validation service returned invalid TPP status"); - } - } // cert validation executor validates the certificate validity - } // enforcement executor validates the certificate presence - } catch (TPPValidationException | CertificateValidationException e) { - final String errorMsg = "Error occurred while validating the TPP status "; - log.error(errorMsg, e); - - //catch errors and set to context - OpenBankingExecutorError error = new OpenBankingExecutorError( - OpenBankingErrorCodes.TPP_VALIDATION_FAILED_CODE, - errorMsg, e.getMessage(), OpenBankingErrorCodes.FORBIDDEN_CODE); - CertificateValidationUtils.handleExecutorErrors(error, obapiRequestContext); - } catch (CertificateException e) { - String errorMsg = "Error occurred while converting the client certificate to X509Certificate "; - log.error(errorMsg, e); - OpenBankingExecutorError error = new OpenBankingExecutorError( - OpenBankingErrorCodes.TPP_VALIDATION_FAILED_CODE, errorMsg, - e.getMessage(), OpenBankingErrorCodes.FORBIDDEN_CODE); - CertificateValidationUtils.handleExecutorErrors(error, obapiRequestContext); - } - } - - private List getRolesFromScopes(Map> allowedScopes, Set scopes) { - List requiredPSD2Roles = new ArrayList<>(); - - Set distinctRoles = new HashSet<>(); - - for (String scope : scopes) { - for (Map.Entry> allowedScopeEntry : allowedScopes.entrySet()) { - if (scope.equalsIgnoreCase(allowedScopeEntry.getKey())) { - distinctRoles.addAll(allowedScopeEntry.getValue()); - } - } - - } - - for (String distinctRole : distinctRoles) { - requiredPSD2Roles.add(PSD2RoleEnum.fromValue(distinctRole)); - } - - return requiredPSD2Roles; - } - - private Set extractScopesFromSwaggerAPI(PathItem electedPath, String httpMethod) { - - List securityRequirements = null; - Set scopes = new HashSet<>(); - - if (GET.equalsIgnoreCase(httpMethod)) { - securityRequirements = electedPath.getGet().getSecurity(); - } else if (POST.equalsIgnoreCase(httpMethod)) { - securityRequirements = electedPath.getPost().getSecurity(); - } else if (PUT.equalsIgnoreCase(httpMethod)) { - securityRequirements = electedPath.getPut().getSecurity(); - } else if (PATCH.equalsIgnoreCase(httpMethod)) { - securityRequirements = electedPath.getPatch().getSecurity(); - } else if (DELETE.equalsIgnoreCase(httpMethod)) { - securityRequirements = electedPath.getDelete().getSecurity(); - } - - if (securityRequirements != null) { - for (SecurityRequirement securityRequirement : securityRequirements) { - for (Map.Entry> securityRequirementEntry : securityRequirement.entrySet()) { - scopes.addAll(securityRequirementEntry.getValue()); - } - } - } - return scopes; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/tpp/validation/executor/DCRTPPValidationExecutor.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/tpp/validation/executor/DCRTPPValidationExecutor.java deleted file mode 100644 index 6f4ab744..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/impl/tpp/validation/executor/DCRTPPValidationExecutor.java +++ /dev/null @@ -1,191 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.impl.tpp.validation.executor; - -import com.wso2.openbanking.accelerator.common.error.OpenBankingErrorCodes; -import com.wso2.openbanking.accelerator.common.exception.CertificateValidationException; -import com.wso2.openbanking.accelerator.common.exception.TPPValidationException; -import com.wso2.openbanking.accelerator.common.model.PSD2RoleEnum; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.gateway.executor.core.OpenBankingGatewayExecutor; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OpenBankingExecutorError; -import com.wso2.openbanking.accelerator.gateway.executor.service.CertValidationService; -import com.wso2.openbanking.accelerator.gateway.executor.util.CertificateValidationUtils; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import net.minidev.json.JSONValue; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.security.cert.Certificate; -import java.security.cert.CertificateException; -import java.security.cert.X509Certificate; -import java.text.ParseException; -import java.util.ArrayList; -import java.util.List; -import java.util.Objects; -import java.util.Optional; - -/** - * TPP validation handler used to validate the TPP status using external validation services - * for DCR API requests. - */ -public class DCRTPPValidationExecutor implements OpenBankingGatewayExecutor { - - private static final String BODY = "body"; - private static final String GET_METHOD_TYPE = "GET"; - private static final String DELETE_METHOD_TYPE = "DELETE"; - private static final String SOFTWARE_ROLES = "software_roles"; - private static final String SOFTWARE_STATEMENT = "software_statement"; - - private static final Log log = LogFactory.getLog(DCRTPPValidationExecutor.class); - - @Generated(message = "Ignoring since all cases are covered from other unit tests") - @Override - public void preProcessRequest(OBAPIRequestContext obapiRequestContext) { - // Skip the executor if previous executors failed. - if (obapiRequestContext.isError()) { - return; - } - - try { - Certificate[] clientCerts = obapiRequestContext.getClientCertsLatest(); - if (clientCerts != null && clientCerts.length > 0) { - Optional transportCert = - CertificateValidationUtils.convertCertToX509Cert(clientCerts[0]); - - // Only Do Validation if Mutual TLS is used. - if (transportCert.isPresent()) { - - String httpMethod = obapiRequestContext.getMsgInfo().getHttpMethod(); - - // During DCR request, skip validation if the method is GET or DELETE as the application roles - // cannot be updated through GET or DELETE. - // Since there is no SSA during these calls, we cannot find the applicable roles as well. - if (GET_METHOD_TYPE.equals(httpMethod) || DELETE_METHOD_TYPE.equals(httpMethod)) { - return; - } - - String softwareStatement = getSSAFromPayload(obapiRequestContext.getRequestPayload()); - List requiredPSD2Roles = getRolesFromSSA(softwareStatement); - - if (requiredPSD2Roles.isEmpty()) { - throw new TPPValidationException("No roles found associated with the request. Hence, cannot " + - "continue with TPP validation"); - } - - if (CertValidationService.getInstance().validateTppRoles(transportCert.get(), requiredPSD2Roles)) { - log.debug("TPP validation service returned a success response"); - } else { - log.error("TPP validation service returned invalid TPP status"); - throw new TPPValidationException("TPP validation service returned invalid TPP status"); - } - } // cert validation executor validates the certificate validity - } // enforcement executor validates the certificate presence - } catch (TPPValidationException | CertificateValidationException | ParseException e) { - final String errorMsg = "Error occurred while validating the TPP status "; - log.error(errorMsg, e); - - //catch errors and set to context - OpenBankingExecutorError error = new OpenBankingExecutorError( - OpenBankingErrorCodes.TPP_VALIDATION_FAILED_CODE, - e.getMessage(), errorMsg, OpenBankingErrorCodes.FORBIDDEN_CODE); - CertificateValidationUtils.handleExecutorErrors(error, obapiRequestContext); - } catch (CertificateException e) { - String errorMsg = "Error occurred while converting the client certificate to X509Certificate "; - log.error(errorMsg, e); - OpenBankingExecutorError error = new OpenBankingExecutorError( - OpenBankingErrorCodes.TPP_VALIDATION_FAILED_CODE, errorMsg, - e.getMessage(), OpenBankingErrorCodes.FORBIDDEN_CODE); - CertificateValidationUtils.handleExecutorErrors(error, obapiRequestContext); - } - } - - @Override - public void preProcessResponse(OBAPIResponseContext obapiResponseContext) { - // Do not need to handle the response - } - - /** - * Method to handle post response. - * - * @param obapiResponseContext OB response context object - */ - @Override - public void postProcessResponse(OBAPIResponseContext obapiResponseContext) { - - } - - /** - * Method to handle post request. - * - * @param obapiRequestContext OB request context object - */ - @Override - public void postProcessRequest(OBAPIRequestContext obapiRequestContext) { - - } - - private String getSSAFromPayload(String requestPayload) throws ParseException { - // decode request body and get payload - JSONObject requestBody = JWTUtils.decodeRequestJWT(requestPayload, BODY); - // extract software statement - return requestBody.getAsString(SOFTWARE_STATEMENT); - } - - /** - * Extract PSD2 roles from SSA. - * - * @param softwareStatement software statement extracted from request payload - * @return list of PSD2RoleEnum - * @throws TPPValidationException when an error occurs when generating PSD2 roles list - */ - public List getRolesFromSSA(String softwareStatement) throws TPPValidationException { - - List requiredPSD2Roles = new ArrayList<>(); - try { - // decode software statement and get payload - JSONObject softwareStatementBody = JWTUtils.decodeRequestJWT(softwareStatement, BODY); - - String softwareRolesStr = softwareStatementBody.getAsString(SOFTWARE_ROLES); - - if (StringUtils.isNotBlank(softwareRolesStr) && softwareRolesStr.contains("[")) { - JSONArray softwareRoles = (JSONArray) JSONValue.parseStrict(softwareRolesStr); - - softwareRoles.stream() - .map(softwareRole -> PSD2RoleEnum.fromValue((String) softwareRole)) - .filter(Objects::nonNull) - .forEach(requiredPSD2Roles::add); - } else { - log.error("Invalid SSA software roles received. Expected array of software roles. Received: " - + softwareRolesStr); - } - } catch (net.minidev.json.parser.ParseException | ParseException e) { - log.error("Error while parsing the message to json", e); - throw new TPPValidationException("Error while parsing the message to json", e); - - } - - return requiredPSD2Roles; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/jws/JwsRequestSignatureHandlingExecutor.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/jws/JwsRequestSignatureHandlingExecutor.java deleted file mode 100644 index f300e722..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/jws/JwsRequestSignatureHandlingExecutor.java +++ /dev/null @@ -1,627 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.jws; - -import com.nimbusds.jose.JOSEException; -import com.nimbusds.jose.JWSAlgorithm; -import com.nimbusds.jose.JWSHeader; -import com.nimbusds.jose.JWSObject; -import com.nimbusds.jose.JWSVerifier; -import com.nimbusds.jose.crypto.ECDSAVerifier; -import com.nimbusds.jose.crypto.RSASSAVerifier; -import com.nimbusds.jose.jwk.ECKey; -import com.nimbusds.jose.jwk.JWK; -import com.nimbusds.jose.jwk.JWKMatcher; -import com.nimbusds.jose.jwk.JWKSelector; -import com.nimbusds.jose.jwk.JWKSet; -import com.nimbusds.jose.jwk.KeyOperation; -import com.nimbusds.jose.jwk.KeyUse; -import com.nimbusds.jose.jwk.RSAKey; -import com.nimbusds.jose.util.Base64URL; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.error.OpenBankingErrorCodes; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.identity.retriever.JWKRetriever; -import com.wso2.openbanking.accelerator.common.identity.retriever.sp.CommonServiceProviderRetriever; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.gateway.executor.core.OpenBankingGatewayExecutor; -import com.wso2.openbanking.accelerator.gateway.executor.exception.OpenBankingExecutorException; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OpenBankingExecutorError; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.net.MalformedURLException; -import java.net.URL; -import java.nio.charset.StandardCharsets; -import java.security.cert.X509Certificate; -import java.text.ParseException; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashSet; -import java.util.List; -import java.util.Map; -import java.util.Optional; -import java.util.Set; - -import javax.ws.rs.HttpMethod; - -/** - * Class to handle JWS Signature validation for requests. - */ -public class JwsRequestSignatureHandlingExecutor implements OpenBankingGatewayExecutor { - - private static final Log log = LogFactory.getLog(JwsRequestSignatureHandlingExecutor.class); - private static final String B64_CLAIM_KEY = "b64"; - private static final String XML_DECLARATION = "\n"; - private static final String DOT_SYMBOL = "."; - - private String xWso2ApiVersion = null; - private String xWso2ApiType = null; - private String signatureHeaderName = getSignatureHeaderName(); - - /** - * Method to handle pre request. - * - * @param obapiRequestContext OB request context object - */ - @Override - @Generated(message = "Excluded from code coverage since it is covered by other methods") - public void preProcessRequest(OBAPIRequestContext obapiRequestContext) { - - } - - /** - * Method to handle post request. - * - * @param obapiRequestContext OB request context object - */ - @Override - @Generated(message = "Excluded from code coverage since it is covered by other methods") - public void postProcessRequest(OBAPIRequestContext obapiRequestContext) { - - if (obapiRequestContext.isError()) { - return; - } - - String messageID = obapiRequestContext.getMsgInfo().getMessageId(); - - log.info(String.format("Executing JwsSignatureHandlingExecutor postProcessRequest for request %s", messageID)); - - if (StringUtils.isEmpty(getXWso2ApiVersion())) { - setXWso2ApiVersion(obapiRequestContext.getApiRequestInfo().getVersion()); - } - - if (StringUtils.isEmpty(getXWso2ApiType())) { - setXWso2ApiType(obapiRequestContext.getApiRequestInfo().getContext()); - } - - // Retrieve headers and payload. - Map requestHeaders = obapiRequestContext.getMsgInfo().getHeaders(); - - Boolean isApplicable = preProcessValidation(obapiRequestContext, requestHeaders); - if (!isApplicable) { - return; - } - - Optional payload = Optional.empty(); - - if (requestHeaders.containsKey(GatewayConstants.CONTENT_TYPE_TAG)) { - if (requestHeaders.get(GatewayConstants.CONTENT_TYPE_TAG).contains(GatewayConstants.TEXT_XML_CONTENT_TYPE) - || requestHeaders.get(GatewayConstants.CONTENT_TYPE_TAG).contains( - GatewayConstants.APPLICATION_XML_CONTENT_TYPE)) { - try { - payload = Optional.of(GatewayUtils.getXMLPayloadToSign(obapiRequestContext.getMsgInfo() - .getPayloadHandler().consumeAsString())); - } catch (Exception e) { - GatewayUtils.handleRequestInternalServerError(obapiRequestContext, - "Internal Server Error, Unable to process Payload", - OpenBankingErrorCodes.SERVER_ERROR_CODE); - } - } else { - payload = Optional.ofNullable(obapiRequestContext.getRequestPayload()); - } - } else { - payload = Optional.ofNullable(obapiRequestContext.getRequestPayload()); - } - - // If the payload can be parsed. - if (log.isDebugEnabled()) { - log.debug(String.format("Request %s is Applicable for JWS Validation", messageID)); - } - - - // Retrieve consumer key from headers. - Optional clientID = Optional.ofNullable(obapiRequestContext.getApiRequestInfo().getConsumerKey()); - // Retrieve x-jws-signature from headers. - String jwsSignature = requestHeaders.get(signatureHeaderName); - // Retrieve context properties. - Map contextProps = obapiRequestContext.getContextProps(); - - // The sent header value should not be empty. - if (StringUtils.isEmpty(jwsSignature)) { - log.error(OpenBankingErrorCodes.EXECUTOR_JWS_SIGNATURE_NOT_FOUND); - handleJwsSignatureErrors(obapiRequestContext, "Empty JWS Signature", - OpenBankingErrorCodes.INVALID_SIGNATURE_CODE); - return; - } - - // Adding jws signature to the context properties - contextProps.put(signatureHeaderName, jwsSignature); - obapiRequestContext.setContextProps(contextProps); - // Now JWS Signature is part of the context properties of the req object. - - if (clientID.isPresent()) { - if (payload.isPresent() && !payload.get().matches("\"\"")) { - if (log.isDebugEnabled()) { - log.debug(String.format("Built ClientID %s for request", clientID.get())); - log.debug("Payload extracted from request"); - } - - boolean verified = false; - try { - verified = validateMessageSignature(clientID.get(), jwsSignature, obapiRequestContext, - payload.get()); - } catch (OpenBankingException | OpenBankingExecutorException e) { - log.error("Unable to validate message signature for the client ID " + clientID.get(), e); - handleJwsSignatureErrors(obapiRequestContext, e.getMessage(), - OpenBankingErrorCodes.INVALID_SIGNATURE_CODE); - return; - } - if (!verified) { - log.error("Signature validation failed for the client ID " + clientID.get()); - handleJwsSignatureErrors(obapiRequestContext, "Invalid JWS Signature", - OpenBankingErrorCodes.INVALID_SIGNATURE_CODE); - } - } else { - if (HttpMethod.POST.equals(obapiRequestContext.getMsgInfo().getMessageId()) || - HttpMethod.PUT.equals(obapiRequestContext.getMsgInfo().getMessageId())) { - handleJwsSignatureErrors(obapiRequestContext, "Request payload cannot be empty", - OpenBankingErrorCodes.MISSING_REQUEST_PAYLOAD); - } - } - } - } - - /** - * Method to validate if the request mandates to execute JWS Signature Validation. - * - * @param obapiRequestContext OB request context object - * @param requestHeaders OB request header Map - */ - @Generated(message = "Removed from unit test coverage since Common module is required") - public Boolean preProcessValidation(OBAPIRequestContext obapiRequestContext, Map requestHeaders) { - // Return if the request contains an error. - // Check mandated apis at toolkit level. - return !obapiRequestContext.isError() && OpenBankingConfigParser.getInstance().isJwsSignatureValidationEnabled() - && !HttpMethod.GET.equals(obapiRequestContext.getMsgInfo().getHttpMethod()) - && !HttpMethod.DELETE.equals(obapiRequestContext.getMsgInfo().getHttpMethod()); - } - - /** - * Method to handle pre response. - * - * @param obapiResponseContext OB response context object - */ - - @Override - @Generated(message = "Excluded from code coverage since it is covered by other methods") - public void preProcessResponse(OBAPIResponseContext obapiResponseContext) { - - } - - /** - * Method to handle post response. - * - * @param obapiResponseContext OB response context object - */ - @Override - @Generated(message = "Excluded from code coverage since it is covered by other methods") - public void postProcessResponse(OBAPIResponseContext obapiResponseContext) { - - } - - /** - * Claims to be validated in JWS header. - * - * @param obapiRequestContext OB request context object - * @param claims jose header claims - * @param appName application name - * @param jwksURI jwksUrl in URL format - * @return boolean - */ - public boolean validateClaims(OBAPIRequestContext obapiRequestContext, - JWSHeader claims, String appName, String jwksURI) { - // Implemented in toolkit level. - - // RSA PSS & EC digital signature algorithms are allowed. - List allowedSigningAlgorithms = OpenBankingConfigParser.getInstance().getJwsRequestSigningAlgorithms(); - - if (!allowedSigningAlgorithms.contains(claims.getAlgorithm().getName())) { - log.error("The " + claims.getAlgorithm().getName() + " algorithm is not supported" + - " by the Solution"); - handleJwsSignatureErrors(obapiRequestContext, "The " + claims.getAlgorithm() - .getName() + " algorithm is not supported by the Solution", - OpenBankingErrorCodes.INVALID_SIGNATURE_CODE); - } - return true; - } - - /** - * Method to validate JWS Signature. - * - * @param clientID client ID of the Application - * @param detachedContentJws JWS in header - * @param obapiRequestContext OB Request context - * @param payload HTTP request payload - * @return boolean - */ - - @Generated(message = "Excluded from code coverage since method includes accessing jwks_uri") - private boolean validateMessageSignature(String clientID, String detachedContentJws, - OBAPIRequestContext obapiRequestContext, String payload) - throws OpenBankingExecutorException, OpenBankingException { - - // Convert Detached JWS into standard JWS. - String reconstructedJws; - try { - reconstructedJws = this.reconstructJws(detachedContentJws, payload); - } catch (OpenBankingExecutorException e) { - log.error("Unable to reconstruct JWS", e); - throw new OpenBankingExecutorException("Malformed JWS Signature", e); - } - - JWSVerifier verifier = null; - JWSObject jwsObject; - RSAKey rsaKey = null; - ECKey ecKey = null; - - // Parse JWSObject to retrieve headers. - try { - jwsObject = JWSObject.parse(reconstructedJws); - } catch (ParseException e) { - log.error("Unable to parse JWS signature" , e); - throw new OpenBankingExecutorException("Unable to parse JWS signature", e); - } - - // Retrieve JWK set - String jwksURI; - String appName = null; - try { - appName = (new CommonServiceProviderRetriever()).getAppPropertyFromSPMetaData(clientID, "software_id"); - } catch (OpenBankingException e) { - log.error("Error while retrieving the app name", e); - throw new OpenBankingExecutorException("Error while retrieving the app name", e); - } - - try { - jwksURI = getJwksUrl(clientID); - - // Get JWKSet from cache or retrieve from onDemand retriever - JWKSet jwkSet = getJwkSet(jwksURI, appName); - - // Get public key from JWK used for signing. - try { - JWK key = retrievePublicKey(jwkSet, jwsObject); - // Public key of the Signing certificate is retrieved - Available 1 key with use:"Sig" - if (key != null) { - X509Certificate x509Certificate = key.getParsedX509CertChain().get(0); - // kty: "RSA" - if (key.getKeyType().getValue().equals("RSA")) { - rsaKey = RSAKey.parse(x509Certificate); - //kty: "EC" - } else if (key.getKeyType().getValue().equals("EC")) { - ecKey = ECKey.parse(x509Certificate); - //log error if the kty is not supported for the allowed signing alg. - } else { - String errorMessage = String.format("The kty %s of the Key is not supported", - key.getKeyType().getValue()); - log.error(errorMessage); - throw new OpenBankingExecutorException(errorMessage); - } - } else { - log.error("Public key of the signing certificate not found in JWK set"); - throw new OpenBankingExecutorException("Public key of the signing certificate not found in JWK " + - "set"); - } - } catch (JOSEException e) { - log.error("Certificate not valid", e); - throw new OpenBankingExecutorException("Certificate not valid", e); - } - - } catch (OpenBankingException e) { - log.error("Unable to validate JWS Signature retrieving public key", e); - throw new OpenBankingExecutorException("Unable to validate JWS Signature retrieving public key", e); - } - - // Validating "iss" , "tan", "alg", "kid" - boolean areClaimsValid = validateClaims(obapiRequestContext, jwsObject.getHeader(), appName, jwksURI); - - try { - - - JWSAlgorithm jwsAlgorithm = JWSAlgorithm.parse(jwsObject.getHeader().getAlgorithm().getName()); - - Set criticalParameters = new HashSet<>(Arrays.asList(differedCriticalClaims())); - - if (JWSAlgorithm.Family.RSA.contains(jwsAlgorithm)) { - // Define JWSVerifier for JWS signed with RSA Signing alg. - verifier = new RSASSAVerifier( - rsaKey != null ? rsaKey.toRSAPublicKey() : null, criticalParameters); - } else if (JWSAlgorithm.Family.EC.contains(jwsAlgorithm)) { - // Define JWSVerifier for JWS signed with EC Signing alg. - verifier = new ECDSAVerifier( - ecKey != null ? ecKey.toECPublicKey() : null, criticalParameters); - } else { - String errorMessage = "The " + jwsObject.getHeader().getAlgorithm().getName() + " algorithm is not " + - "supported by the Solution"; - log.error(errorMessage); - throw new OpenBankingExecutorException(errorMessage); - - } - - } catch (JOSEException e) { - log.error("Invalid Signing Algorithm" , e); - throw new OpenBankingExecutorException("Invalid JWS Signature,signed with invalid " + - "algorithm", e); - } - - // If claims are verified, verify signature. - // Since asymmetric alg is used, the signature can be verified using public key only. - - boolean verified; - - if (areClaimsValid) { - try { - // Check if payload is b64 encoded or un-encoded - if (isB64HeaderVerifiable(jwsObject)) { - // b64=true - verified = jwsObject.verify(verifier); - } else { - // b64=false - // Produces the signature with un-encoded payload. - // which is the encoded header + ".." + the encoded signature - String[] jwsParts = StringUtils.split(detachedContentJws, DOT_SYMBOL); - - JWSHeader header = JWSHeader.parse(new Base64URL(jwsParts[0])); - Base64URL signature = new Base64URL(jwsParts[1]); - verified = verifier.verify(header, getSigningInput(header, payload), signature); - } - if (verified) { - return true; - } - } catch (JOSEException e) { - log.error("Unable to verify JWS signature", e); - throw new OpenBankingExecutorException("Unable to verify JWS signature", e); - } catch (ParseException e) { - log.error("Error occurred while parsing the JWS Header", e); - throw new OpenBankingExecutorException("Error occurred while parsing the JWS Header", e); - } - } - return false; - } - - /** - * Method to reconstruct a detached JWS with encoded payload. - * - * @param jwsSignature Detached JWS - * @param payload HTTP request payload - * @return boolean - */ - private String reconstructJws(String jwsSignature, String payload) throws OpenBankingExecutorException { - - // GET requests and DELETE requests will not need message signing. - if (StringUtils.isEmpty(payload)) { - throw new OpenBankingExecutorException("Payload is required for JWS reconstruction"); - } - - String[] jwsParts = jwsSignature.split("\\."); - - if (log.isDebugEnabled()) { - log.debug(String.format("Found %d parts in JWS for reconstruction", jwsParts.length)); - } - - // Add Base64Url encoded payload. - if (jwsParts.length == 3) { - jwsParts[1] = Base64URL.encode(payload).toString(); - - // Reconstruct JWS with `.` deliminator - return String.join(DOT_SYMBOL, jwsParts); - } else if (jwsParts.length == 5) { - throw new OpenBankingExecutorException("Not supported for signed and encrypted JWTs."); - } - - throw new OpenBankingExecutorException("Required number of parts not found in JWS for reconstruction"); - } - - /** - * If the b64 header is not available or is true, it is verifiable. - * - * @param jwsObject The reconstructed jws object parsed from x-jws-signature - * @return Boolean - */ - private boolean isB64HeaderVerifiable(JWSObject jwsObject) { - - JWSHeader jwsHeader = jwsObject.getHeader(); - Object b64Value = jwsHeader.getCustomParam(B64_CLAIM_KEY); - return b64Value != null ? ((Boolean) b64Value) : true; - } - - /** - * Method to retrieve payload from File Payment Upload requests. - * - * @param header - * @param jwsPayload - * @return signing input - */ - private byte[] getSigningInput(JWSHeader header, String jwsPayload) { - - String combinedInput = header.toBase64URL().toString() + DOT_SYMBOL + jwsPayload; - return combinedInput.getBytes(StandardCharsets.UTF_8); - } - - /** - * Method to handle errors in JWS Signature validation. - * - * @param obapiRequestContext OB request context object - * @param message error message - */ - public void handleJwsSignatureErrors( - OBAPIRequestContext obapiRequestContext, String message, String errorCode) { - - OpenBankingExecutorError error = new OpenBankingExecutorError(errorCode, - OpenBankingErrorCodes.JWS_SIGNATURE_HANDLE_ERROR, message, OpenBankingErrorCodes.BAD_REQUEST_CODE); - setErrorsToRequestContext(obapiRequestContext, error); - } - - public void setErrorsToRequestContext(OBAPIRequestContext obapiRequestContext, OpenBankingExecutorError error) { - ArrayList executorErrors = obapiRequestContext.getErrors(); - executorErrors.add(error); - obapiRequestContext.setError(true); - obapiRequestContext.setErrors(executorErrors); - } - - /** - * Method to retrieve JWKS Url from service Provider properties. - * @param clientID - * @return - * @throws OpenBankingException - */ - @Generated(message = "Excluded from code coverage since method includes service call") - private String getJwksUrl(String clientID) throws OpenBankingException { - - String jwksURI; - try { - jwksURI = (new CommonServiceProviderRetriever()).getAppPropertyFromSPMetaData(clientID, "jwksURI"); - if (jwksURI == null) { - throw new OpenBankingException("The JWK set URL must not be null"); - } - } catch (OpenBankingException e) { - log.error("JWKS URL is not found", e); - throw new OpenBankingException("The JWKS_URI is not found for client ID " + clientID, e); - } - return jwksURI; - } - - /** - * Method to retrieve the JWKSet from JWKS URI. - * - * @param jwksURI - * @param appName - * @return - * @throws OpenBankingException - */ - @Generated(message = "Excluded from code coverage since method includes accessing jwks_uri") - private JWKSet getJwkSet(String jwksURI, String appName) throws OpenBankingException { - JWKSet jwkSet; - try { - jwkSet = new JWKRetriever().getJWKSet(new URL(jwksURI), appName); - } catch (MalformedURLException e) { - log.error("Provided JWKS URL is malformed", e); - throw new OpenBankingException("The provided JWKS_URI is malformed", e); - } - - return jwkSet; - } - - /** - * Method to retrieve the public key from JWKSet. - * - * @param jwkSet - * @param jwsObject - * @return - */ - @Generated(message = "Excluded from code coverage since method includes accessing JWKSet") - private JWK retrievePublicKey(JWKSet jwkSet, JWSObject jwsObject) { - - JWK key = null; - // First get the key with given kid, use as sig and operation as verify from the list. - JWKMatcher keyMatcherWithKidUseAndOperation = - new JWKMatcher.Builder().keyID(jwsObject.getHeader().getKeyID()) - .keyUse(KeyUse.SIGNATURE) - .keyOperation(KeyOperation.VERIFY) - .build(); - List jwkList = new JWKSelector(keyMatcherWithKidUseAndOperation).select(jwkSet); - - if (jwkList.isEmpty()) { - // If empty, then get the key with given kid and use as sig from the list. - JWKMatcher keyMatcherWithKidAndUse = new JWKMatcher.Builder() - .keyID(jwsObject.getHeader().getKeyID()) - .keyUse(KeyUse.SIGNATURE).build(); - jwkList = new JWKSelector(keyMatcherWithKidAndUse).select(jwkSet); - - if (jwkList.isEmpty()) { - // fail over defaults to ->, then get the key with given kid. - JWKMatcher keyMatcherWithKid = new JWKMatcher.Builder().keyID(jwsObject.getHeader().getKeyID()).build(); - jwkList = new JWKSelector(keyMatcherWithKid).select(jwkSet); - } - } - - if (jwkList.isEmpty()) { - log.error("No matching KID found to retrieve public key in JWK set"); - } else { - key = jwkList.get(0); - } - - return key; - } - - /** - * Differed critical parameters for validation. - * - * @return list of critical claims - */ - public String[] differedCriticalClaims() { - // Implemented at toolkit level - - return new String[0]; - } - - public void setXWso2ApiVersion(String xWso2ApiVersion) { - - this.xWso2ApiVersion = xWso2ApiVersion; - } - - public String getXWso2ApiVersion() { - - return this.xWso2ApiVersion; - } - - public String getXWso2ApiType() { - - return xWso2ApiType; - } - - public void setXWso2ApiType(String xWso2ApiType) { - - this.xWso2ApiType = xWso2ApiType; - } - - // If the header name is different at toolkit level, - // this method need to override. - public String getSignatureHeaderName() { - - return "x-jws-signature"; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/jws/JwsResponseSignatureHandlingExecutor.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/jws/JwsResponseSignatureHandlingExecutor.java deleted file mode 100644 index cdbf3f92..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/jws/JwsResponseSignatureHandlingExecutor.java +++ /dev/null @@ -1,256 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.jws; - -import com.nimbusds.jose.JOSEException; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.error.OpenBankingErrorCodes; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.gateway.executor.core.OpenBankingGatewayExecutor; -import com.wso2.openbanking.accelerator.gateway.executor.exception.OpenBankingExecutorException; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.HashMap; -import java.util.Map; -import java.util.Optional; - -/** - * Executor class for Signing Responses. - * @deprecated - * Use {@link com.wso2.openbanking.accelerator.gateway.handler.JwsResponseSignatureHandler} instead. - */ -@Deprecated -public class JwsResponseSignatureHandlingExecutor implements OpenBankingGatewayExecutor { - - private static final Log log = LogFactory.getLog(JwsResponseSignatureHandlingExecutor.class); - - private String xWso2ApiVersion = null; - private String xWso2ApiType = null; - private String signatureHeaderName = getSignatureHeaderName(); - - @Generated(message = "Excluding from unit tests since it is covered by other methods") - public void preProcessRequest(OBAPIRequestContext obapiRequestContext) { - - if (obapiRequestContext.isError()) { - appendJwsSignatureToRequestContext(obapiRequestContext); - } - } - - @Generated(message = "Excluding from unit tests since it is covered by other methods") - public void postProcessRequest(OBAPIRequestContext obapiRequestContext) { - - if (obapiRequestContext.isError()) { - appendJwsSignatureToRequestContext(obapiRequestContext); - } - } - - /** - * Method to handle pre-process response. - * - * @param obapiResponseContext OB response context object - */ - @Generated(message = "Excluding from unit tests since it is covered by other methods") - public void preProcessResponse(OBAPIResponseContext obapiResponseContext) { - - appendJwsSignatureToResponseContext(obapiResponseContext); - } - - @Generated(message = "Excluding from unit tests since it is covered by other methods") - public void postProcessResponse(OBAPIResponseContext obapiResponseContext) { - - appendJwsSignatureToResponseContext(obapiResponseContext); - } - - /** - * Provide the child classes to decide whether the signature generation is required for requestPath. - * - * @param obapiRequestContext OB response Object - * @return boolean returns if request needs to be signed - */ - @Generated(message = "Excluding from unit tests since there is a call to a method in Common Module") - public boolean isApplicableForRequestPath(OBAPIRequestContext obapiRequestContext) { - - return OpenBankingConfigParser.getInstance().isJwsResponseSigningEnabled(); - } - - /** - * Provide the child classes to decide whether the signature generation is required for error scenarios. - * - * @param obapiResponseContext OB response Object - * @return boolean returns if request needs to be signed - */ - @Generated(message = "Excluding from unit tests since there is a call to a method in Common Module") - public boolean isApplicableForResponsePath(OBAPIResponseContext obapiResponseContext) { - - return OpenBankingConfigParser.getInstance().isJwsResponseSigningEnabled(); - } - - /** - * Method to Generate JWS signature. - * - * @param payloadString - * @return - */ - public String generateJWSSignature(Optional payloadString) - throws OpenBankingExecutorException, JOSEException { - - String jwsSignatureHeader = null; - if (payloadString.isPresent() && StringUtils.isNotBlank(payloadString.get())) { - HashMap criticalParameters = getCriticalHeaderParameters(); - jwsSignatureHeader = GatewayUtils.constructJWSSignature(payloadString.get(), criticalParameters); - } else { - log.debug("Signature cannot be generated as the payload is invalid or authentication context is not " + - "available"); - } - return jwsSignatureHeader; - } - - /** - * HashMap to be returned with crit header keys and values. - * can be extended at toolkit level. - * - * @return HashMap crit header parameters - */ - public HashMap getCriticalHeaderParameters() { - - return new HashMap<>(); - } - - @Generated(message = "Excluding from unit test coverage") - public void setXWso2ApiVersion(String xWso2ApiVersion) { - - this.xWso2ApiVersion = xWso2ApiVersion; - } - - @Generated(message = "Excluding from unit test coverage") - public String getXWso2ApiVersion() { - - return this.xWso2ApiVersion; - } - - @Generated(message = "Excluding from unit test coverage") - public String getXWso2ApiType() { - - return xWso2ApiType; - } - - @Generated(message = "Excluding from unit test coverage") - public void setXWso2ApiType(String xWso2ApiType) { - - this.xWso2ApiType = xWso2ApiType; - } - - /** - * Method to change the expected request header name containing the JWS. - */ - public String getSignatureHeaderName() { - - return "x-jws-signature"; - } - - /** - * Method to append Jws Signature To Request Context. - * @param obapiRequestContext - */ - @Generated(message = "Excluding from unit tests since it is covered by other methods") - private void appendJwsSignatureToRequestContext(OBAPIRequestContext obapiRequestContext) { - - setXWso2ApiVersion(obapiRequestContext.getApiRequestInfo().getVersion()); - setXWso2ApiType(obapiRequestContext.getApiRequestInfo().getContext()); - - String messageID = obapiRequestContext.getMsgInfo().getMessageId(); - - if (!isApplicableForRequestPath(obapiRequestContext)) { - if (log.isDebugEnabled()) { - log.debug(String.format( - "Signature generation is not applicable for this response " + - "with message id : %s", messageID)); - } - return; - } else { - if (log.isDebugEnabled()) { - log.debug(String.format( - "Generating signature for the response " + - "with message id : %s", messageID)); - } - // Retrieve headers and payload. - try { - Map responseHeaders = obapiRequestContext.getMsgInfo().getHeaders(); - Optional payload = GatewayUtils.extractRequestPayload(obapiRequestContext, - obapiRequestContext.getMsgInfo().getHeaders()); - responseHeaders.put(signatureHeaderName, generateJWSSignature(payload)); - obapiRequestContext.setAddedHeaders(responseHeaders); - } catch (OpenBankingException | OpenBankingExecutorException | JOSEException e) { - log.error("Unable to sign response", e); - GatewayUtils.handleRequestInternalServerError(obapiRequestContext, - "Internal Server Error, Unable to sign the response", - OpenBankingErrorCodes.SERVER_ERROR_CODE); - } - } - } - - /** - * Method to append Jws Signature To Response Context. - * @param obapiResponseContext - */ - @Generated(message = "Excluding from unit tests since it is covered by other methods") - private void appendJwsSignatureToResponseContext(OBAPIResponseContext obapiResponseContext) { - - setXWso2ApiVersion(obapiResponseContext.getApiRequestInfo().getVersion()); - setXWso2ApiType(obapiResponseContext.getApiRequestInfo().getContext()); - - String messageID = obapiResponseContext.getMsgInfo().getMessageId(); - - if (!isApplicableForResponsePath(obapiResponseContext)) { - if (log.isDebugEnabled()) { - log.debug(String.format( - "Signature generation is not applicable for this response " + - "with message id : %s", messageID)); - } - return; - } else { - if (log.isDebugEnabled()) { - log.debug(String.format( - "Generating signature for the response " + - "with message id : %s", messageID)); - } - // Retrieve headers and payload. - try { - Map responseHeaders = obapiResponseContext.getMsgInfo().getHeaders(); - Optional payload = GatewayUtils.extractResponsePayload(obapiResponseContext, - obapiResponseContext.getMsgInfo().getHeaders()); - responseHeaders.put(signatureHeaderName, generateJWSSignature(payload)); - obapiResponseContext.setAddedHeaders(responseHeaders); - } catch (OpenBankingException | OpenBankingExecutorException | - JOSEException e) { - log.error("Unable to sign response", e); - GatewayUtils.handleResponseInternalServerError(obapiResponseContext, - "Internal Server Error, Unable to sign the response", - OpenBankingErrorCodes.SERVER_ERROR_CODE); - } - } - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/model/OBAPIRequestContext.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/model/OBAPIRequestContext.java deleted file mode 100644 index 0ab7eb3b..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/model/OBAPIRequestContext.java +++ /dev/null @@ -1,263 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.model; - -import com.wso2.openbanking.accelerator.common.error.OpenBankingErrorCodes; -import com.wso2.openbanking.accelerator.gateway.cache.GatewayCacheKey; -import com.wso2.openbanking.accelerator.gateway.internal.GatewayDataHolder; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import io.swagger.parser.OpenAPIParser; -import io.swagger.v3.oas.models.OpenAPI; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.json.JSONException; -import org.json.JSONObject; -import org.wso2.carbon.apimgt.common.gateway.dto.APIRequestInfoDTO; -import org.wso2.carbon.apimgt.common.gateway.dto.MsgInfoDTO; -import org.wso2.carbon.apimgt.common.gateway.dto.RequestContextDTO; - -import java.io.UnsupportedEncodingException; -import java.security.cert.Certificate; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Map; - -import javax.security.cert.X509Certificate; - -/** - * Open Banking executor request context. - */ -public class OBAPIRequestContext extends RequestContextDTO { - - private static final Log log = LogFactory.getLog(OBAPIRequestContext.class); - private final RequestContextDTO requestContextDTO; - private Map contextProps; - private String modifiedPayload; - private String requestPayload; - private Map addedHeaders; - private boolean isError; - private ArrayList errors; - private String consentId; - private Map analyticsData; - private OpenAPI openAPI; - - public OBAPIRequestContext(RequestContextDTO requestContextDTO, - Map contextProps, Map analyticsData) { - - this.requestContextDTO = requestContextDTO; - this.addedHeaders = new HashMap<>(); - this.errors = new ArrayList<>(); - this.contextProps = contextProps; - this.analyticsData = analyticsData; - - Map headers = requestContextDTO.getMsgInfo().getHeaders(); - String authHeader = headers.get(GatewayConstants.AUTH_HEADER); - if (authHeader != null && !authHeader.isEmpty() && - GatewayUtils.isValidJWTToken(authHeader.replace(GatewayConstants.BEARER_TAG, ""))) { - this.consentId = extractConsentID(authHeader); - } - - String apiId = requestContextDTO.getApiRequestInfo().getApiId(); - Object cacheObject = GatewayDataHolder.getGatewayCache() - .getFromCache(GatewayCacheKey.of(apiId)); - if (cacheObject == null) { - String swaggerDefinition = GatewayUtils.getSwaggerDefinition(apiId); - OpenAPIParser parser = new OpenAPIParser(); - this.openAPI = parser.readContents(swaggerDefinition, null, null).getOpenAPI(); - GatewayDataHolder.getGatewayCache().addToCache(GatewayCacheKey.of(apiId), this.openAPI); - } else { - this.openAPI = (OpenAPI) cacheObject; - } - if (requestContextDTO.getMsgInfo().getHeaders().get(GatewayConstants.CONTENT_TYPE_TAG) != null) { - String contentType = requestContextDTO.getMsgInfo().getHeaders().get(GatewayConstants.CONTENT_TYPE_TAG); - String httpMethod = requestContextDTO.getMsgInfo().getHttpMethod(); - String errorMessage = "Request Content-Type header does not match any allowed types"; - if (contentType.startsWith(GatewayConstants.JWT_CONTENT_TYPE) || contentType.startsWith(GatewayConstants - .JOSE_CONTENT_TYPE)) { - try { - this.requestPayload = GatewayUtils.getTextPayload(requestContextDTO.getMsgInfo().getPayloadHandler() - .consumeAsString()); - } catch (Exception e) { - log.error(String.format("Failed to read the text payload from request. %s", e.getMessage())); - handleContentTypeErrors(OpenBankingErrorCodes.INVALID_CONTENT_TYPE, errorMessage); - } - } else if (GatewayUtils.isEligibleRequest(contentType, httpMethod)) { - try { - this.requestPayload = requestContextDTO.getMsgInfo().getPayloadHandler().consumeAsString(); - } catch (Exception e) { - log.error(String.format("Failed to read the payload from request. %s", e.getMessage())); - handleContentTypeErrors(OpenBankingErrorCodes.INVALID_CONTENT_TYPE, errorMessage); - } - } else { - this.requestPayload = null; - } - } - } - - public String getModifiedPayload() { - - return modifiedPayload; - } - - public void setModifiedPayload(String modifiedPayload) { - - this.modifiedPayload = modifiedPayload; - } - - public Map getAddedHeaders() { - - return addedHeaders; - } - - public void setAddedHeaders(Map addedHeaders) { - - this.addedHeaders = addedHeaders; - } - - public Map getContextProps() { - - return contextProps; - } - - public void setContextProps(Map contextProps) { - - this.contextProps = contextProps; - } - - public boolean isError() { - - return isError; - } - - public void setError(boolean error) { - - isError = error; - } - - public ArrayList getErrors() { - - return errors; - } - - public void setErrors( - ArrayList errors) { - - this.errors = errors; - } - - public String getConsentId() { - - return consentId; - } - - public void setConsentId(String consentId) { - - this.consentId = consentId; - } - - public OpenAPI getOpenAPI() { - - return openAPI; - } - - public void setOpenAPI(OpenAPI openAPI) { - - this.openAPI = openAPI; - } - - public Map getAnalyticsData() { - - return analyticsData; - } - - public void setAnalyticsData(Map analyticsData) { - - this.analyticsData = analyticsData; - } - - @Override - public MsgInfoDTO getMsgInfo() { - - return requestContextDTO.getMsgInfo(); - } - - @Override - public APIRequestInfoDTO getApiRequestInfo() { - - return requestContextDTO.getApiRequestInfo(); - } - - @Override - public X509Certificate[] getClientCerts() { - - return requestContextDTO.getClientCerts(); - } - - @Override - public Certificate[] getClientCertsLatest() { - return requestContextDTO.getClientCertsLatest(); - } - - public String getRequestPayload() { - - return requestPayload; - } - - private String extractConsentID(String jwtToken) { - - String consentIdClaim = null; - try { - if (!jwtToken.contains(GatewayConstants.BASIC_TAG)) { - jwtToken = jwtToken.replace(GatewayConstants.BEARER_TAG, ""); - JSONObject jwtClaims = GatewayUtils.decodeBase64(GatewayUtils.getPayloadFromJWT(jwtToken)); - String consentIdClaimName = - GatewayDataHolder.getInstance().getOpenBankingConfigurationService().getConfigurations() - .get(GatewayConstants.CONSENT_ID_CLAIM_NAME).toString(); - if (!jwtClaims.isNull(consentIdClaimName) && - !jwtClaims.getString(consentIdClaimName).isEmpty()) { - consentIdClaim = jwtClaims.getString(consentIdClaimName); - } - } - } catch (UnsupportedEncodingException | JSONException | IllegalArgumentException e) { - log.error(String.format("Failed to retrieve the consent ID from JWT claims. %s", e.getMessage())); - } - return consentIdClaim; - } - - public void addContextProperty(String key, String value) { - - this.contextProps.put(key, value); - } - - public String getContextProperty(String key) { - - return this.contextProps.get(key); - } - - private void handleContentTypeErrors(String errorCode, String errorMessage) { - OpenBankingExecutorError error = new OpenBankingExecutorError(errorCode, - OpenBankingErrorCodes.UNSUPPORTED_MEDIA_TYPE, errorMessage, - OpenBankingErrorCodes.UNSUPPORTED_MEDIA_TYPE_CODE); - - this.isError = true; - this.errors.add(error); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/model/OBAPIResponseContext.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/model/OBAPIResponseContext.java deleted file mode 100644 index 2ca398bf..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/model/OBAPIResponseContext.java +++ /dev/null @@ -1,199 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.model; - -import com.wso2.openbanking.accelerator.common.error.OpenBankingErrorCodes; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.HttpStatus; -import org.json.JSONObject; -import org.json.XML; -import org.wso2.carbon.apimgt.common.gateway.dto.APIRequestInfoDTO; -import org.wso2.carbon.apimgt.common.gateway.dto.MsgInfoDTO; -import org.wso2.carbon.apimgt.common.gateway.dto.ResponseContextDTO; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Map; - -/** - * Open Banking executor response context. - */ -public class OBAPIResponseContext extends ResponseContextDTO { - - private static final Log log = LogFactory.getLog(OBAPIResponseContext.class); - private ResponseContextDTO responseContextDTO; - private Map contextProps; - private String responsePayload; - private String modifiedPayload; - private Map addedHeaders; - private boolean isError; - private ArrayList errors; - private Map analyticsData; - - public OBAPIResponseContext(ResponseContextDTO responseContextDTO, Map contextProps, - Map analyticsData) { - - this.responseContextDTO = responseContextDTO; - this.contextProps = contextProps; - this.errors = new ArrayList<>(); - this.analyticsData = analyticsData; - this.addedHeaders = new HashMap<>(); - if (responseContextDTO.getMsgInfo().getHeaders().get(GatewayConstants.CONTENT_TYPE_TAG) != null) { - String contentType = responseContextDTO.getMsgInfo().getHeaders().get(GatewayConstants.CONTENT_TYPE_TAG); - String httpMethod = responseContextDTO.getMsgInfo().getHttpMethod(); - String errorMessage = "Request Content-Type header does not match any allowed types"; - if (contentType.startsWith(GatewayConstants.JWT_CONTENT_TYPE)) { - try { - this.responsePayload = GatewayUtils.getTextPayload(responseContextDTO.getMsgInfo() - .getPayloadHandler().consumeAsString()); - } catch (Exception e) { - log.error(String.format("Failed to read the text payload from response. %s", e.getMessage())); - handleContentTypeErrors(OpenBankingErrorCodes.INVALID_CONTENT_TYPE, errorMessage); - } - } else if (GatewayUtils.isEligibleResponse(contentType, httpMethod) && - HttpStatus.SC_NO_CONTENT != responseContextDTO.getStatusCode()) { - try { - this.responsePayload = responseContextDTO.getMsgInfo().getPayloadHandler().consumeAsString(); - if (contentType.contains(GatewayConstants.JSON_CONTENT_TYPE) && - this.responsePayload.contains("soapenv:Body")) { - JSONObject soapPayload = XML.toJSONObject(responseContextDTO.getMsgInfo().getPayloadHandler() - .consumeAsString()).getJSONObject("soapenv:Body"); - if (soapPayload.has("jsonObject")) { - this.responsePayload = soapPayload.getJSONObject("jsonObject").toString(); - } else { - this.responsePayload = null; - } - } - } catch (Exception e) { - log.error(String.format("Failed to read the payload from response. %s", e.getMessage())); - handleContentTypeErrors(OpenBankingErrorCodes.INVALID_CONTENT_TYPE, errorMessage); - } - } else { - this.responsePayload = null; - } - } - } - - public String getModifiedPayload() { - - return modifiedPayload; - } - - public void setModifiedPayload(String modifiedPayload) { - - this.modifiedPayload = modifiedPayload; - } - - public Map getAddedHeaders() { - - return addedHeaders; - } - - public void setAddedHeaders(Map addedHeaders) { - - this.addedHeaders = addedHeaders; - } - - public Map getContextProps() { - - return contextProps; - } - - public void setContextProps(Map contextProps) { - - this.contextProps = contextProps; - } - - public boolean isError() { - - return isError; - } - - public void setError(boolean error) { - - isError = error; - } - - public ArrayList getErrors() { - - return errors; - } - - public void setErrors( - ArrayList errors) { - - this.errors = errors; - } - - public Map getAnalyticsData() { - - return analyticsData; - } - - public void setAnalyticsData(Map analyticsData) { - - this.analyticsData = analyticsData; - } - - @Override - public APIRequestInfoDTO getApiRequestInfo() { - - return this.responseContextDTO.getApiRequestInfo(); - } - - @Override - public int getStatusCode() { - - return responseContextDTO.getStatusCode(); - } - - @Override - public MsgInfoDTO getMsgInfo() { - - return responseContextDTO.getMsgInfo(); - } - - public String getResponsePayload() { - - return responsePayload; - } - - public void addContextProperty(String key, String value) { - - this.contextProps.put(key, value); - } - - public String getContextProperty(String key) { - - return this.contextProps.get(key); - } - - private void handleContentTypeErrors(String errorCode, String errorMessage) { - OpenBankingExecutorError error = new OpenBankingExecutorError(errorCode, - OpenBankingErrorCodes.UNSUPPORTED_MEDIA_TYPE, errorMessage, - OpenBankingErrorCodes.UNSUPPORTED_MEDIA_TYPE_CODE); - - this.isError = true; - this.errors.add(error); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/model/OpenBankingExecutorError.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/model/OpenBankingExecutorError.java deleted file mode 100644 index efcf22a3..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/model/OpenBankingExecutorError.java +++ /dev/null @@ -1,106 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.model; - -import java.util.Map; - -/** - * Error model for Open Banking executors. - */ -public class OpenBankingExecutorError { - - private String code; - private String title; - private String message; - private String httpStatusCode; - private Map links; - - public OpenBankingExecutorError() {} - - public OpenBankingExecutorError(String errorCode) { - this.code = errorCode; - } - - - public OpenBankingExecutorError(String code, String title, String message, String httpStatusCode) { - this.code = code; - this.title = title; - this.message = message; - this.httpStatusCode = httpStatusCode; - } - - public OpenBankingExecutorError(String code, String title, String message, String httpStatusCode, - Map links) { - this.code = code; - this.title = title; - this.message = message; - this.httpStatusCode = httpStatusCode; - this.links = links; - } - - public String getCode() { - - return code; - } - - public void setCode(String code) { - - this.code = code; - } - - public String getTitle() { - - return title; - } - - public void setTitle(String title) { - - this.title = title; - } - - public String getMessage() { - - return message; - } - - public void setMessage(String message) { - - this.message = message; - } - - public String getHttpStatusCode() { - - return httpStatusCode; - } - - public void setHttpStatusCode(String httpStatusCode) { - - this.httpStatusCode = httpStatusCode; - } - - public Map getLinks() { - - return links; - } - - public void setLinks(Map links) { - - this.links = links; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/model/RevocationStatus.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/model/RevocationStatus.java deleted file mode 100644 index cb1a27c4..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/model/RevocationStatus.java +++ /dev/null @@ -1,43 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.model; - -/** - * This is used to get Revocation Status message. - */ -public enum RevocationStatus { - - GOOD("Good"), UNKNOWN("Unknown"), REVOKED("Revoked"); - private String message; - - RevocationStatus(String message) { - - this.message = message; - } - - /** - * Get revocation status message. - * - * @return status message - */ - public String getMessage() { - - return message; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/CRLValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/CRLValidator.java deleted file mode 100644 index 47cf1a26..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/CRLValidator.java +++ /dev/null @@ -1,383 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.revocation; - -import com.wso2.openbanking.accelerator.common.exception.CertificateValidationException; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.HTTPClientUtils; -import com.wso2.openbanking.accelerator.gateway.executor.model.RevocationStatus; -import com.wso2.openbanking.accelerator.gateway.executor.util.CertificateValidationUtils; -import com.wso2.openbanking.accelerator.gateway.internal.TPPCertValidatorDataHolder; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.HttpHost; -import org.apache.http.HttpResponse; -import org.apache.http.client.config.RequestConfig; -import org.apache.http.client.methods.HttpGet; -import org.apache.http.impl.client.CloseableHttpClient; -import org.bouncycastle.asn1.ASN1InputStream; -import org.bouncycastle.asn1.ASN1Primitive; -import org.bouncycastle.asn1.DERIA5String; -import org.bouncycastle.asn1.DEROctetString; -import org.bouncycastle.asn1.x509.CRLDistPoint; -import org.bouncycastle.asn1.x509.DistributionPoint; -import org.bouncycastle.asn1.x509.DistributionPointName; -import org.bouncycastle.asn1.x509.Extension; -import org.bouncycastle.asn1.x509.GeneralName; -import org.bouncycastle.asn1.x509.GeneralNames; - -import java.io.IOException; -import java.io.InputStream; -import java.net.MalformedURLException; -import java.security.InvalidKeyException; -import java.security.NoSuchAlgorithmException; -import java.security.NoSuchProviderException; -import java.security.SignatureException; -import java.security.cert.CRLException; -import java.security.cert.CertificateException; -import java.security.cert.CertificateFactory; -import java.security.cert.X509CRL; -import java.security.cert.X509Certificate; -import java.util.ArrayList; -import java.util.Date; -import java.util.List; - -/** - * This is used to verify whether a certificate is revoked or not by using the Certificate Revocation List published - * by the CA. - */ -public class CRLValidator implements RevocationValidator { - - private static final Log log = LogFactory.getLog(CRLValidator.class); - - private final int retryCount; - private static int httpConnectTimeout; - private static int httpConnectionRequestTimeout; - private static int httpSocketTimeout; - - public CRLValidator(int retryCount) { - - this.retryCount = retryCount; - } - - - public CRLValidator(int retryCount, int httpConnectTimeout, int httpConnectionRequestTimeout, - int httpSocketTimeout) { - - this.retryCount = retryCount; - CRLValidator.httpConnectTimeout = httpConnectTimeout; - CRLValidator.httpConnectionRequestTimeout = httpConnectionRequestTimeout; - CRLValidator.httpSocketTimeout = httpSocketTimeout; - } - - /** - * Extracts all CRL distribution point URLs from the "CRL Distribution Point" extension in a X.509 certificate. - * If CRL distribution point extension or CRL Urls are unavailable, throw an exception. - * - * @param cert X509 certificate - * @return List of CRL Urls in the certificate - * @throws CertificateValidationException certificateValidationException - */ - public static List getCRLUrls(X509Certificate cert) throws CertificateValidationException { - - List crlUrls; - byte[] crlDPExtensionValue = getCRLDPExtensionValue(cert); - if (crlDPExtensionValue == null) { - throw new CertificateValidationException("Certificate with serial num:" + cert.getSerialNumber() - + " doesn't have CRL Distribution points"); - } - CRLDistPoint distPoint = getCrlDistPoint(crlDPExtensionValue); - crlUrls = getCrlUrlsFromDistPoint(distPoint); - - if (crlUrls.isEmpty()) { - throw new CertificateValidationException("Cannot get CRL urls from certificate with serial num:" + - cert.getSerialNumber()); - } - return crlUrls; - } - - /** - * Get revocation status of a certificate using CRL Url. - * - * @param peerCert peer certificate - * @param retryCount retry count to connect to CRL Url and get the CRL - * @param crlUrls List of CRL Urls - * @param certificateRevocationProxyEnabled whether certificate revocation proxy enabled in the config - * @param certificateRevocationProxyHost certificate revocation proxy host - * @param certificateRevocationProxyPort certificate revocation proxy port - * @return Revocation status of the certificate - * @throws CertificateValidationException certificateValidationException - */ - public static RevocationStatus getCRLRevocationStatus(X509Certificate peerCert, X509Certificate issuerCert, - int retryCount, List crlUrls, - boolean certificateRevocationProxyEnabled, - String certificateRevocationProxyHost, - int certificateRevocationProxyPort) - throws CertificateValidationException { - - // Check with distributions points in the list one by one. if one fails go to the other. - for (String crlUrl : crlUrls) { - if (log.isDebugEnabled()) { - log.debug("Trying to get CRL for URL: " + crlUrl); - } - X509CRL x509CRL = downloadCRLFromWeb(crlUrl, retryCount, peerCert, issuerCert, - certificateRevocationProxyEnabled, certificateRevocationProxyHost, certificateRevocationProxyPort); - if (x509CRL != null) { - return getRevocationStatusFromCRL(x509CRL, peerCert); - } - } - throw new CertificateValidationException("Cannot check revocation status with the certificate"); - } - - /** - * **************************************** - * Util methods for CRL Validation. - * **************************************** - */ - - private static boolean isValidX509Crl(X509CRL x509CRL, X509Certificate peerCert, X509Certificate issuerCert) - throws CertificateValidationException { - - Date currentDate = CertificateValidationUtils.getNewDate(); - Date nextUpdate = x509CRL.getNextUpdate(); - boolean isValid = false; - - if (isValidX509CRLFromIssuer(x509CRL, peerCert, issuerCert)) { - isValid = isValidX509CRLFromNextUpdate(x509CRL, currentDate, nextUpdate); - } - return isValid; - } - - private static boolean isValidX509CRLFromIssuer(X509CRL x509CRL, X509Certificate peerCert, - X509Certificate issuerCert) - throws CertificateValidationException { - - if (!peerCert.getIssuerDN().equals(x509CRL.getIssuerDN())) { - throw new CertificateValidationException("X509 CRL is not valid. Issuer DN in the peer " + - "certificate: " + peerCert.getIssuerDN() + " does not match with the Issuer DN in the X509 CRL: " + - x509CRL.getIssuerDN()); - } - - // Verify the signature of the CRL. - try { - x509CRL.verify(issuerCert.getPublicKey()); - return true; - } catch (CRLException | NoSuchAlgorithmException | InvalidKeyException | NoSuchProviderException | - SignatureException e) { - throw new CertificateValidationException("CRL signature cannot be verified", e); - } - } - - private static boolean isValidX509CRLFromNextUpdate(X509CRL x509CRL, Date currentDate, Date nextUpdate) - throws CertificateValidationException { - - if (nextUpdate != null) { - if (log.isDebugEnabled()) { - log.debug("Validating the next update date: " + nextUpdate.toString() + " with the current date: " + - currentDate.toString()); - } - if (currentDate.before(x509CRL.getNextUpdate())) { - return true; - } else { - throw new CertificateValidationException("X509 CRL is not valid. Next update date: " + - nextUpdate.toString() + " is before the current date: " + currentDate.toString()); - } - } else { - log.debug("Couldn't validate the X509 CRL, next update date is not available."); - } - return false; - } - - private static X509CRL downloadCRLFromWeb(String crlURL, int retryCount, X509Certificate peerCert, - X509Certificate issuerCert, boolean certificateRevocationProxyEnabled, - String certificateRevocationProxyHost, int certificateRevocationProxyPort) - throws CertificateValidationException { - - X509CRL x509CRL = null; - if (log.isDebugEnabled()) { - log.debug("Certificate revocation check proxy enabled: " + certificateRevocationProxyEnabled); - } - try (CloseableHttpClient client = HTTPClientUtils.getHttpsClient()) { - - HttpGet httpGet = new HttpGet(crlURL); - if (certificateRevocationProxyEnabled) { - log.debug("Setting certificate revocation proxy started."); - if (certificateRevocationProxyHost == null || certificateRevocationProxyHost.trim().isEmpty()) { - String message = "Certificate revocation proxy server host is not configured. Please do set the " + - "'CertificateManagement -> CertificateRevocationProxy -> ProxyHost' file"; - log.error(message); - throw new CertificateValidationException(message); - } - if (log.isDebugEnabled()) { - log.debug("Certificate revocation proxy: " + certificateRevocationProxyHost + ":" + - certificateRevocationProxyPort); - } - HttpHost proxy = new HttpHost(certificateRevocationProxyHost, certificateRevocationProxyPort); - RequestConfig config = RequestConfig.custom().setProxy(proxy).build(); - httpGet.setConfig(config); - log.debug("Setting certificate revocation proxy finished."); - } - - // adding request timeout configurations - RequestConfig timeoutRequestConfig; - if (httpGet.getConfig() == null) { - httpGet.setConfig(RequestConfig.custom().build()); - } - timeoutRequestConfig = RequestConfig.copy(httpGet.getConfig()) - .setConnectTimeout(httpConnectTimeout) - .setConnectionRequestTimeout(httpConnectionRequestTimeout) - .setSocketTimeout(httpSocketTimeout) - .build(); - httpGet.setConfig(timeoutRequestConfig); - // add debug logs - if (log.isDebugEnabled()) { - log.debug("CRL request timeout configurations: " + "httpConnectTimeout: " + httpConnectTimeout + - ", httpConnectionRequestTimeout: " + httpConnectionRequestTimeout + ", httpSocketTimeout: " + - httpSocketTimeout); - } - - HttpResponse httpResponse = client.execute(httpGet); - //Check errors in response: - if (httpResponse.getStatusLine().getStatusCode() / 100 != 2) { - throw new CertificateValidationException("Error getting crl response." + - "Response code is " + httpResponse.getStatusLine().getStatusCode()); - } - InputStream in = httpResponse.getEntity().getContent(); - - CertificateFactory cf = CertificateFactory.getInstance("X.509"); - X509CRL x509CRLDownloaded = (X509CRL) cf.generateCRL(in); - if (log.isDebugEnabled()) { - log.debug("CRL is downloaded from CRL Url: " + crlURL); - } - - if (isValidX509Crl(x509CRLDownloaded, peerCert, issuerCert)) { - x509CRL = x509CRLDownloaded; - } - } catch (MalformedURLException e) { - throw new CertificateValidationException("CRL Url is malformed", e); - } catch (IOException e) { - if (retryCount == 0) { - throw new CertificateValidationException("Cant reach the CRL Url: " + crlURL, e); - } else { - if (log.isDebugEnabled()) { - log.debug("Cant reach CRL Url: " + crlURL + ". Retrying to connect - attempt " + retryCount); - } - return downloadCRLFromWeb(crlURL, --retryCount, peerCert, issuerCert, - certificateRevocationProxyEnabled, certificateRevocationProxyHost, - certificateRevocationProxyPort); - } - } catch (CertificateException e) { - throw new CertificateValidationException("Error when generating certificate factory.", e); - } catch (CRLException e) { - throw new CertificateValidationException("Cannot generate X509CRL from the stream data", e); - } catch (OpenBankingException e) { - throw new CertificateValidationException("Error when creating http client.", e); - } - return x509CRL; - } - - private static RevocationStatus getRevocationStatusFromCRL(X509CRL x509CRL, X509Certificate peerCert) { - - if (x509CRL.isRevoked(peerCert)) { - return RevocationStatus.REVOKED; - } else { - return RevocationStatus.GOOD; - } - } - - private static byte[] getCRLDPExtensionValue(X509Certificate cert) { - - //DER-encoded octet string of the extension value for CRLDistributionPoints identified by the passed-in oid - return cert.getExtensionValue(Extension.cRLDistributionPoints.getId()); - } - - private static CRLDistPoint getCrlDistPoint(byte[] crlDPExtensionValue) throws CertificateValidationException { - - //crlDPExtensionValue is encoded in ASN.1 format - //DER (Distinguished Encoding Rules) is one of ASN.1 encoding rules defined in ITU-T X.690, 2002, specification. - //ASN.1 encoding rules can be used to encode any data object into a binary file. Read the object in octets. - CRLDistPoint distPoint; - try (ASN1InputStream crlDPEx = new ASN1InputStream(crlDPExtensionValue); - ASN1InputStream asn1InOctets = - new ASN1InputStream(((DEROctetString) (crlDPEx).readObject()).getOctets())) { - //Get Input stream in octets - ASN1Primitive crlDERObject = asn1InOctets.readObject(); - distPoint = CRLDistPoint.getInstance(crlDERObject); - } catch (IOException e) { - throw new CertificateValidationException("Cannot read certificate to get CRL urls", e); - } - return distPoint; - } - - private static List getCrlUrlsFromDistPoint(CRLDistPoint distPoint) { - - List crlUrls = new ArrayList<>(); - //Loop through ASN1Encodable DistributionPoints - for (DistributionPoint dp : distPoint.getDistributionPoints()) { - //get ASN1Encodable DistributionPointName - DistributionPointName dpn = dp.getDistributionPoint(); - if (dpn != null && dpn.getType() == DistributionPointName.FULL_NAME) { - //Create ASN1Encodable General Names - GeneralName[] genNames = GeneralNames.getInstance(dpn.getName()).getNames(); - // Look for a URI - for (GeneralName genName : genNames) { - if (genName.getTagNo() == GeneralName.uniformResourceIdentifier) { - //DERIA5String contains an ascii string. - //A IA5String is a restricted character string type in the ASN.1 notation - String url = DERIA5String.getInstance(genName.getName()).getString().trim(); - crlUrls.add(url); - } - } - } - } - return crlUrls; - } - - /** - * Checks revocation status (Good, Revoked) of the peer certificate. - * - * @param peerCert peer certificate - * @param issuerCert issuer certificate of the peer - * @return revocation status of the peer certificate - * @throws CertificateValidationException certificateValidationException - */ - @Override - public RevocationStatus checkRevocationStatus(X509Certificate peerCert, X509Certificate issuerCert) - throws CertificateValidationException { - - TPPCertValidatorDataHolder tppCertValidatorDataHolder = TPPCertValidatorDataHolder.getInstance(); - - final boolean isCertificateRevocationProxyEnabled = tppCertValidatorDataHolder - .isCertificateRevocationProxyEnabled(); - final int certificateRevocationProxyPort = tppCertValidatorDataHolder - .getCertificateRevocationProxyPort(); - final String certificateRevocationProxyHost = tppCertValidatorDataHolder - .getCertificateRevocationProxyHost(); - - List crlUrls = getCRLUrls(peerCert); - return getCRLRevocationStatus(peerCert, issuerCert, retryCount, crlUrls, isCertificateRevocationProxyEnabled, - certificateRevocationProxyHost, certificateRevocationProxyPort); - } - - @Override - public int getRetryCount() { - - return retryCount; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/OCSPValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/OCSPValidator.java deleted file mode 100644 index 8f0269b1..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/OCSPValidator.java +++ /dev/null @@ -1,416 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.revocation; - -import com.wso2.openbanking.accelerator.common.exception.CertificateValidationException; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.HTTPClientUtils; -import com.wso2.openbanking.accelerator.gateway.executor.model.RevocationStatus; -import com.wso2.openbanking.accelerator.gateway.executor.util.CertificateValidationUtils; -import com.wso2.openbanking.accelerator.gateway.internal.TPPCertValidatorDataHolder; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.HttpHost; -import org.apache.http.HttpResponse; -import org.apache.http.client.config.RequestConfig; -import org.apache.http.client.methods.HttpPost; -import org.apache.http.entity.ByteArrayEntity; -import org.apache.http.entity.ContentType; -import org.apache.http.impl.client.CloseableHttpClient; -import org.bouncycastle.asn1.ASN1IA5String; -import org.bouncycastle.asn1.ASN1InputStream; -import org.bouncycastle.asn1.DEROctetString; -import org.bouncycastle.asn1.ocsp.OCSPObjectIdentifiers; -import org.bouncycastle.asn1.ocsp.OCSPResponseStatus; -import org.bouncycastle.asn1.x509.AccessDescription; -import org.bouncycastle.asn1.x509.AuthorityInformationAccess; -import org.bouncycastle.asn1.x509.Extension; -import org.bouncycastle.asn1.x509.Extensions; -import org.bouncycastle.asn1.x509.GeneralName; -import org.bouncycastle.cert.X509CertificateHolder; -import org.bouncycastle.cert.ocsp.BasicOCSPResp; -import org.bouncycastle.cert.ocsp.CertificateID; -import org.bouncycastle.cert.ocsp.CertificateStatus; -import org.bouncycastle.cert.ocsp.OCSPException; -import org.bouncycastle.cert.ocsp.OCSPReq; -import org.bouncycastle.cert.ocsp.OCSPReqBuilder; -import org.bouncycastle.cert.ocsp.OCSPResp; -import org.bouncycastle.cert.ocsp.SingleResp; -import org.bouncycastle.jce.provider.BouncyCastleProvider; -import org.bouncycastle.operator.DigestCalculatorProvider; -import org.bouncycastle.operator.OperatorCreationException; -import org.bouncycastle.operator.jcajce.JcaDigestCalculatorProviderBuilder; - -import java.io.ByteArrayInputStream; -import java.io.IOException; -import java.io.InputStream; -import java.math.BigInteger; -import java.security.Security; -import java.security.cert.CertificateEncodingException; -import java.security.cert.X509Certificate; -import java.util.ArrayList; -import java.util.List; - -/** - * This is used to verify a certificate is revoked or not by using the Online Certificate Status Protocol published - * by the CA. - */ -public class OCSPValidator implements RevocationValidator { - - private static final Log log = LogFactory.getLog(OCSPValidator.class); - - private static final String BC = "BC"; - private final int retryCount; - private static int httpConnectTimeout; - private static int httpConnectionRequestTimeout; - private static int httpSocketTimeout; - - public OCSPValidator(int retryCount) { - - this.retryCount = retryCount; - } - - public OCSPValidator(int retryCount, int httpConnectTimeout, int httpConnectionRequestTimeout, - int httpSocketTimeout) { - - this.retryCount = retryCount; - OCSPValidator.httpConnectTimeout = httpConnectTimeout; - OCSPValidator.httpConnectionRequestTimeout = httpConnectionRequestTimeout; - OCSPValidator.httpSocketTimeout = httpSocketTimeout; - } - - /** - * Authority Information Access (AIA) is a non-critical extension in an X509 Certificate. This contains the - * URL of the OCSP endpoint if one is available. - * - * @param cert is the certificate - * @return a list of URLs in AIA extension of the certificate which will hopefully contain an OCSP endpoint - * @throws CertificateValidationException certificateValidationException - */ - public static List getAIALocations(X509Certificate cert) throws CertificateValidationException { - - List ocspUrlList; - byte[] aiaExtensionValue = getAiaExtensionValue(cert); - if (aiaExtensionValue == null) { - throw new CertificateValidationException("Certificate with serial num: " + - cert.getSerialNumber() + " doesn't have Authority Information Access points"); - } - AuthorityInformationAccess authorityInformationAccess = getAuthorityInformationAccess(aiaExtensionValue); - ocspUrlList = getOcspUrlsFromAuthorityInfoAccess(authorityInformationAccess); - - if (ocspUrlList.isEmpty()) { - throw new CertificateValidationException("Cant get OCSP urls from certificate with serial num: " + - cert.getSerialNumber()); - } - - return ocspUrlList; - } - - /** - * This method generates an OCSP Request to be sent to an OCSP endpoint. - * - * @param issuerCert is the Certificate of the Issuer of the peer certificate we are interested in - * @param serialNumber of the peer certificate - * @return generated OCSP request - * @throws CertificateValidationException certificateRevocationValidationException - */ - private static OCSPReq generateOCSPRequest(X509Certificate issuerCert, BigInteger serialNumber) - throws CertificateValidationException { - - // Add provider BC - Security.addProvider(new BouncyCastleProvider()); - try { - - byte[] issuerCertEnc = issuerCert.getEncoded(); - X509CertificateHolder certificateHolder = new X509CertificateHolder(issuerCertEnc); - DigestCalculatorProvider digCalcProv = new JcaDigestCalculatorProviderBuilder().setProvider(BC).build(); - - // CertID structure is used to uniquely identify certificates that are the subject of - // an OCSP request or response and has an ASN.1 definition. CertID structure is defined in RFC 2560 - CertificateID id = new CertificateID(digCalcProv.get(CertificateID.HASH_SHA1), certificateHolder, - serialNumber); - - // basic request generation with nonce - OCSPReqBuilder builder = new OCSPReqBuilder(); - builder.addRequest(id); - - // create details for nonce extension. The nonce extension is used to bind a request to a response to - // prevent replay attacks. As the name implies, the nonce value is something that the client should only - // use once within a reasonably small period. - BigInteger nonce = BigInteger.valueOf(System.currentTimeMillis()); - - // create the request Extension - builder.setRequestExtensions(new Extensions(new Extension(OCSPObjectIdentifiers.id_pkix_ocsp_nonce, false, - new DEROctetString(nonce.toByteArray())))); - - return builder.build(); - } catch (CertificateEncodingException | IOException | OCSPException | OperatorCreationException e) { - throw new CertificateValidationException("Cannot generate OSCP Request with the given certificate with " + - "serial num: " + serialNumber, e); - } - } - - /** - * Get revocation status of a certificate using OCSP Url. - * - * @param peerCert peer certificate - * @param issuerCert issuer certificate of peer - * @param retryCount retry count to connect to OCSP Url and get the OCSP response - * @param locations AIA locations - * @param certificateRevocationProxyEnabled whether certificate revocation proxy enabled in the config - * @param certificateRevocationProxyHost certificate revocation proxy host - * @param certificateRevocationProxyPort certificate revocation proxy port - * @return Revocation status of the certificate - * @throws CertificateValidationException certificateValidationException - */ - public static RevocationStatus getOCSPRevocationStatus(X509Certificate peerCert, X509Certificate issuerCert, - int retryCount, List locations, - boolean certificateRevocationProxyEnabled, - String certificateRevocationProxyHost, - int certificateRevocationProxyPort) - throws CertificateValidationException { - - OCSPReq request = generateOCSPRequest(issuerCert, peerCert.getSerialNumber()); - for (String serviceUrl : locations) { - SingleResp[] responses; - try { - if (log.isDebugEnabled()) { - log.debug("Trying to get OCSP Response from : " + serviceUrl); - } - OCSPResp ocspResponse = getOCSPResponse(serviceUrl, request, retryCount, - certificateRevocationProxyEnabled, certificateRevocationProxyHost, - certificateRevocationProxyPort); - if (OCSPResponseStatus.SUCCESSFUL != ocspResponse.getStatus()) { - log.debug("OCSP Response is not successfully received."); - continue; - } - - BasicOCSPResp basicResponse = (BasicOCSPResp) ocspResponse.getResponseObject(); - responses = (basicResponse == null) ? null : basicResponse.getResponses(); - } catch (OCSPException | CertificateValidationException e) { - // On any error, consider the other AIA locations as well. - log.debug("Certificate revocation check failed due to an exception", e); - continue; - } - - if (responses != null && responses.length == 1) { - return getRevocationStatusFromOCSP(responses[0]); - } - } - throw new CertificateValidationException("Cant get Revocation Status from OCSP using any of the OCSP Urls " + - "for certificate with serial num:" + peerCert.getSerialNumber()); - } - - private static List getOcspUrlsFromAuthorityInfoAccess(AuthorityInformationAccess - authorityInformationAccess) { - - List ocspUrlList = new ArrayList<>(); - AccessDescription[] accessDescriptions; - if (authorityInformationAccess != null) { - accessDescriptions = authorityInformationAccess.getAccessDescriptions(); - for (AccessDescription accessDescription : accessDescriptions) { - - GeneralName gn = accessDescription.getAccessLocation(); - if (gn.getTagNo() == GeneralName.uniformResourceIdentifier) { - ASN1IA5String str = ASN1IA5String.getInstance(gn.getName()); - String accessLocation = str.getString(); - ocspUrlList.add(accessLocation); - } - } - } - return ocspUrlList; - } - - private static AuthorityInformationAccess getAuthorityInformationAccess(byte[] aiaExtensionValue) - throws CertificateValidationException { - - AuthorityInformationAccess authorityInformationAccess; - try (ASN1InputStream asn1InputStream = - new ASN1InputStream(((DEROctetString) - (new ASN1InputStream(new ByteArrayInputStream(aiaExtensionValue)).readObject())) - .getOctets())) { - authorityInformationAccess = AuthorityInformationAccess.getInstance(asn1InputStream.readObject()); - } catch (IOException e) { - throw new CertificateValidationException("Cannot read certificate to get OSCP urls", e); - } - return authorityInformationAccess; - } - - private static byte[] getAiaExtensionValue(X509Certificate cert) { - - //Gets the DER-encoded OCTET string for the extension value for Authority information access Points - return cert.getExtensionValue(Extension.authorityInfoAccess.getId()); - } - - /** - * Gets an ASN.1 encoded OCSP response (as defined in RFC 2560) from the given service URL. Currently supports - * only HTTP. - * - * @param serviceUrl URL of the OCSP endpoint. - * @param request an OCSP request object. - * @param certificateRevocationProxyEnabled whether certificate revocation proxy enabled in the config - * @param certificateRevocationProxyHost certificate revocation proxy host - * @param certificateRevocationProxyPort certificate revocation proxy port - * @return OCSP response encoded in ASN.1 structure. - * @throws CertificateValidationException certificateValidationException - */ - private static OCSPResp getOCSPResponse(String serviceUrl, OCSPReq request, int retryCount, - boolean certificateRevocationProxyEnabled, - String certificateRevocationProxyHost, - int certificateRevocationProxyPort) - throws CertificateValidationException { - - OCSPResp ocspResp = null; - if (log.isDebugEnabled()) { - log.debug("Certificate revocation check proxy enabled: " + certificateRevocationProxyEnabled); - } - try (CloseableHttpClient client = HTTPClientUtils.getHttpsClient()) { - HttpPost httpPost = new HttpPost(serviceUrl); - - if (certificateRevocationProxyEnabled) { - log.debug("Setting certificate revocation proxy started."); - if (certificateRevocationProxyHost == null || certificateRevocationProxyHost.trim().isEmpty()) { - String message = "Certificate revocation proxy server host is not configured. Please do set the " + - "'CertificateManagement -> CertificateRevocationProxy -> ProxyHost' file"; - log.error(message); - throw new CertificateValidationException(message); - } - - if (log.isDebugEnabled()) { - log.debug("Certificate revocation proxy: " + certificateRevocationProxyHost + ":" + - certificateRevocationProxyPort); - } - HttpHost proxy = new HttpHost(certificateRevocationProxyHost, certificateRevocationProxyPort); - RequestConfig config = RequestConfig.custom().setProxy(proxy).build(); - httpPost.setConfig(config); - log.debug("Setting certificate revocation proxy finished."); - } - - // adding request timeout configurations - RequestConfig timeoutRequestConfig; - if (httpPost.getConfig() == null) { - httpPost.setConfig(RequestConfig.custom().build()); - } - timeoutRequestConfig = RequestConfig.copy(httpPost.getConfig()) - .setConnectTimeout(httpConnectTimeout) - .setConnectionRequestTimeout(httpConnectionRequestTimeout) - .setSocketTimeout(httpSocketTimeout) - .build(); - httpPost.setConfig(timeoutRequestConfig); - // add debug logs - if (log.isDebugEnabled()) { - log.debug("OCSP request timeout configurations: " + "httpConnectTimeout: " + httpConnectTimeout + - ", httpConnectionRequestTimeout: " + httpConnectionRequestTimeout + ", httpSocketTimeout: " + - httpSocketTimeout); - } - - setRequestProperties(request.getEncoded(), httpPost); - HttpResponse httpResponse = client.execute(httpPost); - - //Check errors in response, if response status code is not 200 (success) range, throws exception - // eg: if response code is 200 (success) or 201 (accepted) return true, - // if response code is 404 (not found) or 500 throw exception - if (httpResponse.getStatusLine().getStatusCode() / 100 != 2) { - throw new CertificateValidationException("Error getting ocsp response." + - "Response code is " + httpResponse.getStatusLine().getStatusCode()); - } - InputStream in = httpResponse.getEntity().getContent(); - ocspResp = new OCSPResp(in); - } catch (IOException e) { - if (log.isDebugEnabled()) { - log.debug("Certificate revocation check failed due to an exception", e); - } - if (retryCount == 0) { - throw new CertificateValidationException("Cannot get ocspResponse from url: " - + serviceUrl, e); - } else { - log.info("Cant reach URI: " + serviceUrl + ". Retrying to connect - attempt " + retryCount); - return getOCSPResponse(serviceUrl, request, --retryCount, certificateRevocationProxyEnabled, - certificateRevocationProxyHost, certificateRevocationProxyPort); - } - } catch (OpenBankingException e) { - throw new CertificateValidationException("Error when creating http client.", e); - } - return ocspResp; - } - - private static void setRequestProperties(byte[] message, HttpPost httpPost) { - - httpPost.addHeader(CertificateValidationUtils.HTTP_CONTENT_TYPE, - CertificateValidationUtils.HTTP_CONTENT_TYPE_OCSP); - httpPost.addHeader(CertificateValidationUtils.HTTP_ACCEPT, - CertificateValidationUtils.HTTP_ACCEPT_OCSP); - - httpPost.setEntity(new ByteArrayEntity(message, - ContentType.create(CertificateValidationUtils.CONTENT_TYPE))); - } - - private static RevocationStatus getRevocationStatusFromOCSP(SingleResp resp) - throws CertificateValidationException { - - Object status = resp.getCertStatus(); - if (status == CertificateStatus.GOOD) { - return RevocationStatus.GOOD; - } else if (status instanceof org.bouncycastle.cert.ocsp.RevokedStatus) { - return RevocationStatus.REVOKED; - } else if (status instanceof org.bouncycastle.cert.ocsp.UnknownStatus) { - return RevocationStatus.UNKNOWN; - } - throw new CertificateValidationException("Cant recognize Certificate Status"); - } - - /** - * Check revocation status (Good, Revoked, Unknown) of the peer certificate. - * - * @param peerCert peer certificate - * @param issuerCert issuer certificate of the peer - * @return revocation status of the peer certificate - * @throws CertificateValidationException certificateValidationException - */ - @Override - public RevocationStatus checkRevocationStatus(X509Certificate peerCert, X509Certificate issuerCert) - throws CertificateValidationException { - - if (issuerCert == null) { - throw new CertificateValidationException("Issuer Certificate is not available for " + - "OCSP validation"); - } - List locations = getAIALocations(peerCert); - if (log.isDebugEnabled()) { - log.debug("Peer certificate AIA locations: " + locations); - } - TPPCertValidatorDataHolder tppCertValidatorDataHolder = TPPCertValidatorDataHolder.getInstance(); - - final boolean isCertificateRevocationProxyEnabled = tppCertValidatorDataHolder - .isCertificateRevocationProxyEnabled(); - final int certificateRevocationProxyPort = tppCertValidatorDataHolder - .getCertificateRevocationProxyPort(); - final String certificateRevocationProxyHost = tppCertValidatorDataHolder - .getCertificateRevocationProxyHost(); - - return getOCSPRevocationStatus(peerCert, issuerCert, retryCount, locations, isCertificateRevocationProxyEnabled - , certificateRevocationProxyHost, certificateRevocationProxyPort); - } - - @Override - public int getRetryCount() { - - return retryCount; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/RevocationValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/RevocationValidator.java deleted file mode 100644 index fa8f707f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/RevocationValidator.java +++ /dev/null @@ -1,48 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.revocation; - - -import com.wso2.openbanking.accelerator.common.exception.CertificateValidationException; -import com.wso2.openbanking.accelerator.gateway.executor.model.RevocationStatus; - -import java.security.cert.X509Certificate; - -/** - * This interface needs to be implemented by any certificate revocation validator. - */ -public interface RevocationValidator { - - /** - * Checks revocation status of the peer certificate. - * - * @param peerCert peer certificate - * @param issuerCert issuer certificate - * @return revocation status - * @throws CertificateValidationException when an error occurs while checking the revocation status - */ - RevocationStatus checkRevocationStatus(X509Certificate peerCert, X509Certificate issuerCert) - throws CertificateValidationException; - - /** - * Get revocation validator retry count. - * - * @return validator retry count - */ - int getRetryCount(); -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/service/CertValidationService.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/service/CertValidationService.java deleted file mode 100644 index 84897229..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/service/CertValidationService.java +++ /dev/null @@ -1,245 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.service; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.CertificateValidationException; -import com.wso2.openbanking.accelerator.common.exception.TPPValidationException; -import com.wso2.openbanking.accelerator.common.model.PSD2RoleEnum; -import com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor.CertificateContent; -import com.wso2.openbanking.accelerator.common.util.eidas.certificate.extractor.CertificateContentExtractor; -import com.wso2.openbanking.accelerator.gateway.cache.GatewayCacheKey; -import com.wso2.openbanking.accelerator.gateway.cache.TppValidationCache; -import com.wso2.openbanking.accelerator.gateway.executor.model.RevocationStatus; -import com.wso2.openbanking.accelerator.gateway.executor.revocation.RevocationValidator; -import com.wso2.openbanking.accelerator.gateway.internal.TPPCertValidatorDataHolder; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.security.cert.X509Certificate; -import java.util.Collections; -import java.util.List; -import java.util.Map; -import java.util.Objects; - -/** - * CertRevocationValidation Service class is responsible for validating client certificates. - */ -public class CertValidationService { - - private static final Log log = LogFactory.getLog(CertValidationService.class); - private static CertValidationService certValidationService; - - private CertValidationService() { - - } - - /** - * Singleton getInstance method to create only one object. - * - * @return CertValidationService object - */ - public static synchronized CertValidationService getInstance() { - if (certValidationService == null) { - certValidationService = new CertValidationService(); - } - - return certValidationService; - } - - - /** - * Validate the certificate revocation status. - * - * @param peerCertificate X509Certificate - * @param issuerCertificate X509Certificate - * @return true if the certificate is not revoked - * @deprecated use {@link #verify(X509Certificate, X509Certificate, int, int, int, int)} instead - */ - @Deprecated - public boolean verify(X509Certificate peerCertificate, X509Certificate issuerCertificate, int retryCount) { - - OpenBankingConfigParser openBankingConfigParser = OpenBankingConfigParser.getInstance(); - RevocationValidatorFactory revocationValidatorFactory = new RevocationValidatorFactory(); - Map revocationValidators = openBankingConfigParser.getCertificateRevocationValidators(); - - // OCSP validation is checked first as it is faster than the CRL validation. Moving to CRL validation - // only if an error occurs during the OCSP validation. - RevocationValidator[] validators = revocationValidators - .entrySet() - .stream() - .sorted(Map.Entry.comparingByKey()) - .map(Map.Entry::getValue) - .map(type -> revocationValidatorFactory.getValidator(type, retryCount)) - .filter(Objects::nonNull) - .toArray(RevocationValidator[]::new); - - for (RevocationValidator validator : validators) { - RevocationStatus revocationStatus = isRevoked(validator, peerCertificate, issuerCertificate); - if (RevocationStatus.GOOD == revocationStatus) { - return true; - } else if (RevocationStatus.REVOKED == revocationStatus) { - return false; - } - } - log.error("Unable to verify certificate revocation information"); - return false; - } - - /** - * Validate the certificate revocation status. - * - * @param peerCertificate X509Certificate - * @param issuerCertificate X509Certificate - * @param retryCount retry count - * @param connectTimeout connect timeout - * @param connectionRequestTimeout connection request timeout - * @param socketTimeout socket timeout - * @return true if the certificate is not revoked - */ - public boolean verify(X509Certificate peerCertificate, X509Certificate issuerCertificate, int retryCount, - int connectTimeout, int connectionRequestTimeout, int socketTimeout) { - - OpenBankingConfigParser openBankingConfigParser = OpenBankingConfigParser.getInstance(); - RevocationValidatorFactory revocationValidatorFactory = new RevocationValidatorFactory(); - Map revocationValidators = openBankingConfigParser.getCertificateRevocationValidators(); - - // OCSP validation is checked first as it is faster than the CRL validation. Moving to CRL validation - // only if an error occurs during the OCSP validation. - RevocationValidator[] validators = revocationValidators - .entrySet() - .stream() - .sorted(Map.Entry.comparingByKey()) - .map(Map.Entry::getValue) - .map(type -> revocationValidatorFactory.getValidator(type, retryCount, connectTimeout, - connectionRequestTimeout, socketTimeout)) - .filter(Objects::nonNull) - .toArray(RevocationValidator[]::new); - - for (RevocationValidator validator : validators) { - RevocationStatus revocationStatus = isRevoked(validator, peerCertificate, issuerCertificate); - if (RevocationStatus.GOOD == revocationStatus) { - return true; - } else if (RevocationStatus.REVOKED == revocationStatus) { - return false; - } - } - log.error("Unable to verify certificate revocation information"); - return false; - } - - private RevocationStatus isRevoked(RevocationValidator validator, X509Certificate peerCertificate, - X509Certificate issuerCertificate) { - - if (log.isDebugEnabled()) { - log.debug("X509 Certificate validation with " + validator.getClass().getSimpleName()); - } - try { - return validator.checkRevocationStatus(peerCertificate, issuerCertificate); - } catch (CertificateValidationException e) { - log.warn("Unable to validate certificate revocation with " + - validator.getClass().getSimpleName(), e); - return RevocationStatus.UNKNOWN; - } - } - - public boolean validateTppRoles(X509Certificate tppCertificate, List requiredPSD2Roles) - throws TPPValidationException, CertificateValidationException { - - if (TPPCertValidatorDataHolder.getInstance().isTppValidationEnabled()) { - - String tppValidationServiceImplClassPath = - TPPCertValidatorDataHolder.getInstance().getTPPValidationServiceImpl(); - if (StringUtils.isNotBlank(tppValidationServiceImplClassPath)) { - TPPValidationService tppValidationService = - TPPCertValidatorDataHolder.getInstance().getTppValidationService(); - - if (tppValidationService != null) { - - // Initializing certificate cache and cache key - TppValidationCache tppValidationServiceCache = TppValidationCache.getInstance(); - String tppValidationCacheKeyStr = tppValidationService.getCacheKey(tppCertificate, - requiredPSD2Roles, Collections.emptyMap()); - GatewayCacheKey tppValidationCacheKey = GatewayCacheKey.of(tppValidationCacheKeyStr); - - // Executing TPP role validation process or retrieve last status from cache - if (tppValidationServiceCache.getFromCache(tppValidationCacheKey) != null) { - // previous result is present in cache, return result - return tppValidationServiceCache.getFromCache(tppValidationCacheKey); - } else { - final boolean result = tppValidationService - .validate(tppCertificate, requiredPSD2Roles, Collections.emptyMap()); - if (result) { - // Adding result to cache - tppValidationServiceCache.addToCache(tppValidationCacheKey, true); - return true; - } - } - } else { - throw new TPPValidationException( - "Unable to find the implementation class for TPP validation service"); - } - } else { - throw new TPPValidationException("TPP validation service " + - "class implementation is empty"); - } - } else if (TPPCertValidatorDataHolder.getInstance().isPsd2RoleValidationEnabled()) { - return isRequiredRolesMatchWithScopes(tppCertificate, requiredPSD2Roles); - } else { - throw new TPPValidationException("Both TPP validation and PSD2 role validation services are disabled"); - } - return false; - } - - /** - * Validate whether the psd2 roles match with the scopes. - * - * @param tppCertificate eidas certificate with roles - * @param requiredPSD2Roles client requested roles - * @return true if all required values are present in the certificate - */ - private boolean isRequiredRolesMatchWithScopes(X509Certificate tppCertificate - , List requiredPSD2Roles) throws CertificateValidationException, TPPValidationException { - - - // Extract the certContent from the eidas certificate (i.e. roles, authorization number, etc) - CertificateContent certContent = CertificateContentExtractor.extract(tppCertificate); - - if (log.isDebugEnabled()) { - log.debug("The TPP is requesting roles: " + requiredPSD2Roles); - log.debug("Provided PSD2 eIDAS certificate" + - " contains the role: " + certContent.getPspRoles()); - } - - // Validate whether the eIDAS certificate contains the required roles that matches with the token scopes. - for (PSD2RoleEnum requiredRole : requiredPSD2Roles) { - if (!certContent.getPspRoles().contains(requiredRole.name())) { - // Return false if any one of the roles that are bound to the scope is not present in the PSD2 - // role list of the client eIDAS certificate. - final String errorMsg = "The PSD2 eIDAS certificate does not contain the required role " - + requiredRole.toString(); - - log.error(errorMsg); - throw new TPPValidationException(errorMsg); - } - } - return true; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/service/RevocationValidatorFactory.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/service/RevocationValidatorFactory.java deleted file mode 100644 index ee6d5c1c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/service/RevocationValidatorFactory.java +++ /dev/null @@ -1,73 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.service; - -import com.wso2.openbanking.accelerator.gateway.executor.revocation.CRLValidator; -import com.wso2.openbanking.accelerator.gateway.executor.revocation.OCSPValidator; -import com.wso2.openbanking.accelerator.gateway.executor.revocation.RevocationValidator; - -/** - * RevocationValidatorFactory is used to get a object of type RevocationValidator. - */ -public class RevocationValidatorFactory { - - private static final String OCSP_VALIDATOR = "OCSP"; - private static final String CRL_VALIDATOR = "CRL"; - - - /** - * Get a object of type RevocationValidator. - * - * @param validatorType name of the required RevocationValidator type - * @param retryCount retry count to connect to revocation validator endpoint and get the response - * @return RevocationValidator type object (OCSP/CRL) - * @deprecated use {@link #getValidator(String, int, int, int, int)} instead - */ - @Deprecated - public RevocationValidator getValidator(String validatorType, int retryCount) { - if (OCSP_VALIDATOR.equalsIgnoreCase(validatorType)) { - return new OCSPValidator(retryCount); - } else if (CRL_VALIDATOR.equalsIgnoreCase(validatorType)) { - return new CRLValidator(retryCount); - } else { - return null; - } - } - - /** - * Get a object of type RevocationValidator. - * - * @param validatorType name of the required RevocationValidator type. - * @param retryCount retry count to connect to revocation validator endpoint and get the response. - * @param connectTimeout timeout for connecting to revocation validator endpoint. - * @param connectionRequestTimeout timeout for getting a connection from the connection manager. - * @param socketTimeout timeout for getting the response from the revocation validator endpoint. - * @return - */ - public RevocationValidator getValidator(String validatorType, int retryCount, int connectTimeout, - int connectionRequestTimeout, int socketTimeout) { - if (OCSP_VALIDATOR.equalsIgnoreCase(validatorType)) { - return new OCSPValidator(retryCount, connectTimeout, connectionRequestTimeout, socketTimeout); - } else if (CRL_VALIDATOR.equalsIgnoreCase(validatorType)) { - return new CRLValidator(retryCount, connectTimeout, connectionRequestTimeout, socketTimeout); - } else { - return null; - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/service/TPPValidationService.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/service/TPPValidationService.java deleted file mode 100644 index 866cc711..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/service/TPPValidationService.java +++ /dev/null @@ -1,60 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.service; - -import com.wso2.openbanking.accelerator.common.exception.TPPValidationException; -import com.wso2.openbanking.accelerator.common.model.PSD2RoleEnum; - -import java.security.cert.X509Certificate; -import java.util.List; -import java.util.Map; - -/** - * Manager interface to be used for TPP validation using external services. - */ -public interface TPPValidationService { - - /** - * Validate the status of a TPP. - * - * @param peerCertificate Certificate of the TPP - * @param requiredPSD2Roles Roles that are required to be validated with the TPP validation service according to - * the current flow - * @param metadata Metadata information - * @return - * @throws TPPValidationException - */ - boolean validate(X509Certificate peerCertificate, - List requiredPSD2Roles, Map metadata) - throws TPPValidationException; - - /** - * Get the cache key used for the caching the response. Implementation should return an appropriate ID that is - * unique to the API flow. - * - * @param peerCertificate Certificate of the TPP - * @param requiredPSD2Roles Roles that are required to be validated with the TPP validation service according to - * the current flow - * @param metadata Metadata information - * @return - * @throws TPPValidationException - */ - String getCacheKey(X509Certificate peerCertificate, - List requiredPSD2Roles, Map metadata) - throws TPPValidationException; -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/util/CertificateValidationUtils.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/util/CertificateValidationUtils.java deleted file mode 100644 index f1238d1e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/executor/util/CertificateValidationUtils.java +++ /dev/null @@ -1,300 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.util; - -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.error.OpenBankingErrorCodes; -import com.wso2.openbanking.accelerator.common.exception.CertificateValidationException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OpenBankingExecutorError; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.base.ServerConfiguration; - -import java.io.ByteArrayInputStream; -import java.io.IOException; -import java.io.InputStream; -import java.nio.file.Files; -import java.nio.file.Paths; -import java.security.InvalidKeyException; -import java.security.KeyStore; -import java.security.KeyStoreException; -import java.security.NoSuchAlgorithmException; -import java.security.NoSuchProviderException; -import java.security.SignatureException; -import java.security.cert.Certificate; -import java.security.cert.CertificateException; -import java.security.cert.CertificateFactory; -import java.security.cert.X509Certificate; -import java.util.ArrayList; -import java.util.Date; -import java.util.Enumeration; -import java.util.Optional; - -/** - * Utility class containing certificate util methods. - */ -public class CertificateValidationUtils { - - public static final String BEGIN_CERT = "-----BEGIN CERTIFICATE-----"; - public static final String END_CERT = "-----END CERTIFICATE-----"; - public static final String X509_CERT_INSTANCE_NAME = "X.509"; - public static final String HTTP_CONTENT_TYPE = "Content-Type"; - public static final String HTTP_CONTENT_TYPE_OCSP = "application/ocsp-request"; - public static final String HTTP_ACCEPT = "Accept"; - public static final String HTTP_ACCEPT_OCSP = "application/ocsp-response"; - public static final String CONTENT_TYPE = "application/json"; - public static final String TRUSTSTORE_LOCATION_CONF_KEY = "Security.TrustStore.Location"; - public static final String TRUSTSTORE_PASS_CONF_KEY = "Security.TrustStore.Password"; - private static final Log LOG = LogFactory.getLog(CertificateValidationUtils.class); - private static KeyStore trustStore = null; - - private CertificateValidationUtils() { - // Adding a private constructor to hide the implicit public one. - } - - /** - * @deprecated use com.wso2.openbanking.accelerator.common.util.CertificateUtils.isExpired() instead - */ - @Deprecated - public static boolean isExpired(X509Certificate peerCertificate) { - try { - peerCertificate.checkValidity(); - } catch (CertificateException e) { - LOG.error("Certificate with the serial number " + - peerCertificate.getSerialNumber() + " issued by the CA " + - peerCertificate.getIssuerDN().toString() + " is expired. Caused by, " + e.getMessage()); - return true; - } - return false; - } - - /** - * Get issuer certificate from the truststore. - * - * @param peerCertificate peer certificate - * @return certificate issuer of the peer certificate - * @throws CertificateValidationException when unable to validate the certificate - */ - public static X509Certificate getIssuerCertificateFromTruststore(X509Certificate peerCertificate) - throws CertificateValidationException { - - KeyStore loadedTrustStore = getTrustStore(); - if (loadedTrustStore == null) { - throw new CertificateValidationException("Client truststore has not been initialized"); - } - - return retrieveCertificateFromTruststore(peerCertificate, loadedTrustStore); - - } - - /** - * Get the truststore. This methods needs to be synchronized with the loadTrustStore() method - * - * @return instance of the truststore - */ - public static synchronized KeyStore getTrustStore() { - return trustStore; - } - - /** - * Get certificate from truststore. - * - * @param peerCertificate peer certificate - * @param loadedTrustStore truststore - * @return certificate retrieved from truststore - * @throws CertificateValidationException when unable to validate the certificate - */ - public static X509Certificate retrieveCertificateFromTruststore( - X509Certificate peerCertificate, KeyStore loadedTrustStore) throws CertificateValidationException { - - Enumeration enumeration; - java.security.cert.X509Certificate certificate; - try { - // Get aliases of all the certificates in the truststore. - enumeration = loadedTrustStore.aliases(); - } catch (KeyStoreException e) { - throw new CertificateValidationException("Error while retrieving aliases from keystore", e); - } - - // As there is no any specific way to query the issuer certificate from the truststore, public keys of all the - // certificates in the truststore are validated against the signature of the peer certificate to identify the - // issuer. - if (enumeration != null) { - while (enumeration.hasMoreElements()) { - String alias = null; - try { - alias = (String) enumeration.nextElement(); - certificate = (java.security.cert.X509Certificate) loadedTrustStore.getCertificate(alias); - } catch (KeyStoreException e) { - throw new CertificateValidationException("Unable to read the certificate from truststore with " + - "the alias: " + alias, e); - } - try { - peerCertificate.verify(certificate.getPublicKey()); - LOG.debug("Valid issuer certificate found in the client truststore"); - return certificate; - } catch (CertificateException | NoSuchAlgorithmException | InvalidKeyException | - NoSuchProviderException | SignatureException e) { - // Unable to verify the signature. Check with the next certificate. - } - } - } else { - throw new CertificateValidationException("Unable to read the certificate aliases from the truststore"); - } - throw new CertificateValidationException("Unable to find the immediate issuer from the truststore of the " + - "certificate with the serial number " + peerCertificate.getSerialNumber() + " issued by the CA " + - peerCertificate.getIssuerDN().toString()); - } - - /** - * Loads the Truststore. - * - * @param trustStorePassword truststore password - */ - @SuppressFBWarnings("PATH_TRAVERSAL_IN") - // Suppressed content - Files.newInputStream(Paths.get(trustStorePath))) - // Suppression reason - False Positive : trustStorePath is obtained from deployment.toml. So it can be marked - // as a trusted filepath - // Suppressed warning count - 1 - @Generated(message = "Ignoring because ServerConfiguration cannot be mocked") - public static synchronized void loadTrustStore(char[] trustStorePassword) - throws KeyStoreException, IOException, CertificateException, NoSuchAlgorithmException { - - String trustStorePath = ServerConfiguration.getInstance() - .getFirstProperty(CertificateValidationUtils.TRUSTSTORE_LOCATION_CONF_KEY); - try (InputStream inputStream = Files.newInputStream(Paths.get(trustStorePath))) { - trustStore = KeyStore.getInstance(OpenBankingConstants.TRUSTSTORE_CONF_TYPE_DEFAULT); - trustStore.load(inputStream, trustStorePassword); - } - } - - - /** - * Loads the Truststore. - * - * This method is deprecated as it allows custom absolute file paths which could result in - * path traversal attacks. Do not use this method unless the custom path is trusted. - * @param trustStorePath truststore path - * @param trustStorePassword truststore password - */ - @SuppressFBWarnings("PATH_TRAVERSAL_IN") - // Suppressed content - dataHolder.getKeyStoreLocation() - // Suppression reason - False Positive : Keystore location is obtained from deployment.toml. So it can be marked - // as a trusted filepath - // Suppressed warning count - 1 - @Deprecated - public static synchronized void loadTrustStore(String trustStorePath, char[] trustStorePassword) - throws KeyStoreException, IOException, CertificateException, NoSuchAlgorithmException { - - try (InputStream inputStream = Files.newInputStream(Paths.get(trustStorePath))) { - trustStore = KeyStore.getInstance(OpenBankingConstants.TRUSTSTORE_CONF_TYPE_DEFAULT); - trustStore.load(inputStream, trustStorePassword); - } - } - - public static void handleExecutorErrors(CertificateValidationException e - , OBAPIRequestContext obapiRequestContext) { - OpenBankingExecutorError error = new OpenBankingExecutorError(e.getErrorCode(), e.getMessage(), - e.getErrorPayload(), OpenBankingErrorCodes.UNAUTHORIZED_CODE); - handleExecutorErrors(error, obapiRequestContext); - } - - public static void handleExecutorErrors(OpenBankingExecutorError error - , OBAPIRequestContext obapiRequestContext) { - ArrayList executorErrors = obapiRequestContext.getErrors(); - executorErrors.add(error); - - obapiRequestContext.setError(true); - obapiRequestContext.setErrors(executorErrors); - } - - /** - * Convert javax.security.cert.X509Certificate to java.security.cert.X509Certificate. - * - * @param cert the certificate to be converted - * @return java.security.cert.X509Certificate type certificate - * @deprecated use convertCert(javax.security.cert.X509Certificate cert) method instead. - */ - @Deprecated - public static Optional convert(javax.security.cert.X509Certificate cert) { - - try { - return convertCert(cert); - } catch (CertificateException e) { - // Not logging the errors again as it is done in the convertCert method - return Optional.empty(); - } - } - - /** - * Convert javax.security.cert.X509Certificate to java.security.cert.X509Certificate - * This method will also handle the exceptions that could occur in the process of converting the certificate. - * Will be having the above convert method as well as it is a public method and will deprecate it gradually. - * - * @param cert the certificate to be converted - * @return java.security.cert.X509Certificate type certificate - * @throws CertificateException - */ - public static Optional convertCert(javax.security.cert.X509Certificate cert) - throws CertificateException { - - if (cert != null) { - try { - byte[] encoded = cert.getEncoded(); - ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(encoded); - java.security.cert.CertificateFactory certificateFactory - = java.security.cert.CertificateFactory.getInstance(X509_CERT_INSTANCE_NAME); - return Optional.of((java.security.cert.X509Certificate) certificateFactory.generateCertificate( - byteArrayInputStream)); - } catch (javax.security.cert.CertificateEncodingException e) { - String errorMsg = "Error while decoding the certificate "; - LOG.error(errorMsg, e); - throw new CertificateException(errorMsg, e); - } catch (java.security.cert.CertificateException e) { - String errorMsg = "Error while generating the certificate "; - LOG.error(errorMsg, e); - throw new CertificateException(errorMsg, e); - } - } else { - return Optional.empty(); - } - } - - /** - * Converts java.security.cert.Certificate to java.security.cert.X509Certificate. - * - * @param certificate java.security.cert.Certificate which needs conversion - * @return java.security.cert.X509Certificate - * @throws CertificateException thrown if an error occurs while converting - */ - public static Optional convertCertToX509Cert(Certificate certificate) - throws CertificateException { - - CertificateFactory cf = CertificateFactory.getInstance(X509_CERT_INSTANCE_NAME); - ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(certificate.getEncoded()); - return Optional.of((X509Certificate) cf.generateCertificate(byteArrayInputStream)); - } - - public static Date getNewDate() { - return new Date(); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/handler/GatewayClientAuthenticationHandler.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/handler/GatewayClientAuthenticationHandler.java deleted file mode 100644 index 0ffe6650..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/handler/GatewayClientAuthenticationHandler.java +++ /dev/null @@ -1,79 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). All Rights Reserved. - *

- * This software is the property of WSO2 LLC. and its suppliers, if any. - * Dissemination of any information or reproduction of any material contained - * herein in any form is strictly forbidden, unless permitted by WSO2 expressly. - * You may not alter or remove any copyright or other notice from copies of this content. - */ - -package com.wso2.openbanking.accelerator.gateway.handler; - -import com.wso2.openbanking.accelerator.common.error.OpenBankingErrorCodes; -import com.wso2.openbanking.accelerator.gateway.internal.GatewayDataHolder; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import org.apache.axis2.context.MessageContext; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.synapse.core.axis2.Axis2MessageContext; -import org.apache.synapse.rest.AbstractHandler; - -import java.io.UnsupportedEncodingException; -import java.net.URLEncoder; -import java.security.cert.CertificateEncodingException; -import java.security.cert.X509Certificate; -import java.util.Map; - -/** - * Handler to send transport certificate as a header to identity server. - * Responds with an error if the transport certificate is not found or malformed. - */ -public class GatewayClientAuthenticationHandler extends AbstractHandler { - - private static final Log log = LogFactory.getLog(GatewayClientAuthenticationHandler.class); - - @Override - public boolean handleRequest(org.apache.synapse.MessageContext messageContext) { - - log.debug("Gateway Client Authentication Handler engaged"); - MessageContext ctx = ((Axis2MessageContext) messageContext).getAxis2MessageContext(); - X509Certificate x509Certificate = GatewayUtils.extractAuthCertificateFromMessageContext(ctx); - Map headers = (Map) ctx.getProperty(MessageContext.TRANSPORT_HEADERS); - - if (x509Certificate != null) { - log.debug("Valid certificate found in request"); - try { - String certificateHeader = GatewayDataHolder.getInstance().getClientTransportCertHeaderName(); - String encodedCert = GatewayUtils.getPEMEncodedCertificateString(x509Certificate); - if (GatewayDataHolder.getInstance().isUrlEncodeClientTransportCertHeaderEnabled()) { - log.debug("URL encoding pem encoded transport certificate"); - encodedCert = URLEncoder.encode(encodedCert, "UTF-8"); - } - headers.put(certificateHeader, encodedCert); - ctx.setProperty(MessageContext.TRANSPORT_HEADERS, headers); - if (log.isDebugEnabled()) { - log.debug(String.format("Added encoded transport certificate in header %s", certificateHeader)); - } - } catch (CertificateEncodingException | UnsupportedEncodingException e) { - log.error("Unable to encode client transport certificate", e); - GatewayUtils.returnSynapseHandlerJSONError(messageContext, OpenBankingErrorCodes.BAD_REQUEST_CODE, - GatewayUtils.getOAuth2JsonErrorBody(GatewayConstants.INVALID_REQUEST, - GatewayConstants.TRANSPORT_CERT_MALFORMED)); - } - } else { - log.debug(GatewayConstants.TRANSPORT_CERT_NOT_FOUND); - GatewayUtils.returnSynapseHandlerJSONError(messageContext, OpenBankingErrorCodes.BAD_REQUEST_CODE, - GatewayUtils.getOAuth2JsonErrorBody(GatewayConstants.INVALID_REQUEST, - GatewayConstants.TRANSPORT_CERT_NOT_FOUND)); - } - return true; - } - - @Override - public boolean handleResponse(org.apache.synapse.MessageContext messageContext) { - - return true; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/handler/JwsResponseSignatureHandler.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/handler/JwsResponseSignatureHandler.java deleted file mode 100644 index 2cbcb347..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/handler/JwsResponseSignatureHandler.java +++ /dev/null @@ -1,293 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.handler; - -import com.nimbusds.jose.JOSEException; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.error.OpenBankingErrorCodes; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.gateway.executor.exception.OpenBankingExecutorException; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.synapse.AbstractSynapseHandler; -import org.apache.synapse.MessageContext; -import org.apache.synapse.core.axis2.Axis2MessageContext; -import org.apache.synapse.rest.RESTConstants; -import org.json.JSONArray; -import org.json.JSONObject; - -import java.util.HashMap; -import java.util.Map; -import java.util.Optional; - -/** - * Handler class for Signing Responses. - */ -public class JwsResponseSignatureHandler extends AbstractSynapseHandler { - - private static final Log log = LogFactory.getLog(JwsResponseSignatureHandler.class); - - private String xWso2ApiVersion = null; - private String xWso2ApiType = null; - private final String signatureHeaderName = getSignatureHeaderName(); - public static final String ERRORS_TAG = "errors"; - public static final String INTERNAL_SERVER_ERROR = "Internal server error"; - - /** - * Constructor for JwsResponseSignatureHandler. - */ - @Generated(message = "Ignoring since method contains no logics") - public JwsResponseSignatureHandler() { - - log.debug("Initializing JwsResponseSignatureHandler to append jws response signature."); - } - - /** - * Handle request message coming into the engine. - * - * @param messageContext incoming request message context - * @return whether mediation flow should continue - */ - @Override - @Generated(message = "Ignoring since method contains no logics") - public boolean handleRequestInFlow(MessageContext messageContext) { - - return true; - - } - - /** - * Handle request message going out from the engine. - * - * @param messageContext outgoing request message context - * @return whether mediation flow should continue - */ - @Override - @Generated(message = "Ignoring since method contains no logics") - public boolean handleRequestOutFlow(MessageContext messageContext) { - - return true; - } - - /** - * Handle response message coming into the engine. - * - * @param messageContext incoming response message context - * @return whether mediation flow should continue - */ - @Override - @Generated(message = "Ignoring since all cases are covered from other unit tests") - public boolean handleResponseInFlow(MessageContext messageContext) { - - return appendJwsSignatureToResponse(messageContext); - } - - /** - * Handle response message going out from the engine. - * - * @param messageContext outgoing response message context - * @return whether mediation flow should continue - */ - @Override - public boolean handleResponseOutFlow(MessageContext messageContext) { - - org.apache.axis2.context.MessageContext axis2MC = - ((Axis2MessageContext) messageContext).getAxis2MessageContext(); - Map headers = (Map) axis2MC.getProperty(org.apache.axis2.context.MessageContext.TRANSPORT_HEADERS); - if (messageContext.getEnvelope() != null && messageContext.getEnvelope().getBody() != null && - StringUtils.contains(messageContext.getEnvelope().getBody().toString(), - "Schema validation failed")) { - // Add jws header for schema errors, This is due to schema validation happens after responseInFlow. - // So we need to regenerate the jws for schema validation error responses. - return appendJwsSignatureToResponse(messageContext); - } else if (headers.containsKey(signatureHeaderName) && headers.get(signatureHeaderName) != null) { - return true; - } else { - // Add jws header, if it's not added yet. - return appendJwsSignatureToResponse(messageContext); - } - } - - /** - * Method to append Jws Signature to the response. - * - * @param messageContext response/request message context. - * @return jws signature response is successfully appended. - */ - private boolean appendJwsSignatureToResponse(MessageContext messageContext) { - - setXWso2ApiVersion((String) messageContext.getProperty(RESTConstants.SYNAPSE_REST_API_VERSION)); - setXWso2ApiType((String) messageContext.getProperty(RESTConstants.REST_API_CONTEXT)); - - try { - boolean applicable = isApplicable(messageContext); - if (!applicable) { - log.debug("Signature generation is not applicable for this response"); - return true; - } else { - log.debug("Generating signature for the response"); - } - } catch (RuntimeException e) { - log.debug("Internal Server Error, Unable to append jws signature", e); - GatewayUtils.returnSynapseHandlerJSONError(messageContext, OpenBankingErrorCodes.SERVER_ERROR_CODE, - getFormattedSignatureHandlingErrorResponse(messageContext, OpenBankingErrorCodes.SERVER_ERROR_CODE, - INTERNAL_SERVER_ERROR, "Internal Server Error, Unable to append jws signature")); - } - - // Build the payload from messageContext. - org.apache.axis2.context.MessageContext axis2MC = - ((Axis2MessageContext) messageContext).getAxis2MessageContext(); - Map headers = (Map) axis2MC.getProperty(org.apache.axis2.context.MessageContext.TRANSPORT_HEADERS); - Optional payloadString; - try { - payloadString = GatewayUtils.buildMessagePayloadFromMessageContext(axis2MC, headers); - } catch (OpenBankingException e) { - log.error("Unable to build response payload", e); - GatewayUtils.returnSynapseHandlerJSONError(messageContext, OpenBankingErrorCodes.SERVER_ERROR_CODE, - getFormattedSignatureHandlingErrorResponse(messageContext, OpenBankingErrorCodes.SERVER_ERROR_CODE, - INTERNAL_SERVER_ERROR, "Internal Server Error, Unable to build response payload")); - return true; - } - - if (payloadString.isPresent()) { - try { - headers.put(signatureHeaderName, generateJWSSignature(payloadString)); - } catch (JOSEException | OpenBankingException e) { - log.error("Unable to sign response", e); - GatewayUtils.returnSynapseHandlerJSONError(messageContext, OpenBankingErrorCodes.SERVER_ERROR_CODE, - getFormattedSignatureHandlingErrorResponse(messageContext, - OpenBankingErrorCodes.SERVER_ERROR_CODE, INTERNAL_SERVER_ERROR, - "Internal Server Error, Unable to sign the response")); - return true; - } - } else { - log.debug("Signature cannot be generated as the payload is invalid or not present."); - } - axis2MC.setProperty(org.apache.axis2.context.MessageContext.TRANSPORT_HEADERS, headers); - return true; - } - - /** - * Method to change the expected request header name containing the JWS. - * - * @return String signature header name. - */ - @Generated(message = "Excluding from unit tests since there is no logics to test") - public String getSignatureHeaderName() { - - return "x-jws-signature"; - } - - /** - * Provide the child classes to decide whether the signature generation is required for requestPath. - * - * @param messageContext OB response Object - * @return boolean returns if request needs to be signed - */ - @Generated(message = "Excluding from unit tests since there is a call to a method in Common Module") - public boolean isApplicable(MessageContext messageContext) { - - return OpenBankingConfigParser.getInstance().isJwsResponseSigningEnabled(); - } - - /** - * Method to Generate JWS signature. - * - * @param payloadString payload. - * @return String jws signature. - */ - public String generateJWSSignature(Optional payloadString) - throws OpenBankingException, JOSEException { - - String jwsSignatureHeader = null; - if (payloadString.isPresent() && StringUtils.isNotBlank(payloadString.get())) { - HashMap criticalParameters = getCriticalHeaderParameters(); - try { - jwsSignatureHeader = GatewayUtils.constructJWSSignature(payloadString.get(), criticalParameters); - } catch (OpenBankingExecutorException e) { - throw new OpenBankingException(e.getMessage()); - } - } else { - log.debug("Signature cannot be generated as the payload is invalid."); - } - return jwsSignatureHeader; - } - - /** - * HashMap to be returned with crit header keys and values. - * can be extended at toolkit level. - * - * @return HashMap crit header parameters - */ - @Generated(message = "Excluding from unit test coverage") - public HashMap getCriticalHeaderParameters() { - - return new HashMap<>(); - } - - /** - * Method to get the formatted error response for jws signature response. - * - * @param messageContext messageContext - * @param code error code - * @param title error title - * @param errorMessage error message - * @return String error response - */ - @Generated(message = "Excluding from unit test coverage") - public String getFormattedSignatureHandlingErrorResponse(MessageContext messageContext, String code, String title, - String errorMessage) { - - JSONObject payload = new JSONObject(); - JSONArray errorList = new JSONArray(); - JSONObject errorObj = new JSONObject(); - errorObj.put("Code", code); - errorObj.put("Title", title); - errorObj.put("Message", errorMessage); - errorList.put(errorObj); - return payload.put(ERRORS_TAG, errorList).toString(); - } - - @Generated(message = "Excluding from unit test coverage") - public void setXWso2ApiVersion(String xWso2ApiVersion) { - - this.xWso2ApiVersion = xWso2ApiVersion; - } - - @Generated(message = "Excluding from unit test coverage") - public String getXWso2ApiVersion() { - - return this.xWso2ApiVersion; - } - - @Generated(message = "Excluding from unit test coverage") - public String getXWso2ApiType() { - - return xWso2ApiType; - } - - @Generated(message = "Excluding from unit test coverage") - public void setXWso2ApiType(String xWso2ApiType) { - - this.xWso2ApiType = xWso2ApiType; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/internal/GatewayDataHolder.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/internal/GatewayDataHolder.java deleted file mode 100644 index d67230ee..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/internal/GatewayDataHolder.java +++ /dev/null @@ -1,372 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.internal; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.HTTPClientUtils; -import com.wso2.openbanking.accelerator.common.util.OpenBankingUtils; -import com.wso2.openbanking.accelerator.data.publisher.common.constants.DataPublishingConstants; -import com.wso2.openbanking.accelerator.gateway.cache.GatewayCache; -import com.wso2.openbanking.accelerator.gateway.executor.core.AbstractRequestRouter; -import com.wso2.openbanking.accelerator.gateway.throttling.ThrottleDataPublisher; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import org.apache.http.impl.client.CloseableHttpClient; -import org.wso2.carbon.apimgt.impl.APIConstants; -import org.wso2.carbon.apimgt.impl.APIManagerConfiguration; -import org.wso2.carbon.apimgt.impl.APIManagerConfigurationService; -import org.wso2.carbon.base.ServerConfiguration; -import org.wso2.carbon.identity.core.util.IdentityUtil; - -import java.util.Arrays; -import java.util.HashMap; -import java.util.Map; - -/** - * Data holder for executor core. - */ -public class GatewayDataHolder { - - private static volatile GatewayDataHolder instance; - private static volatile CloseableHttpClient httpClient; - private static volatile GatewayCache gatewayCache; - private OpenBankingConfigurationService openBankingConfigurationService; - private Map configurations; - private APIManagerConfigurationService apiManagerConfigurationService; - private AbstractRequestRouter requestRouter; - private Map urlMap; - private ThrottleDataPublisher throttleDataPublisher; - private int gatewayCacheAccessExpiry; - private int gatewayCacheModifiedExpiry; - private String keyStoreLocation; - private char[] keyStorePassword; - private String keyAlias; - private String keyPassword; - private boolean isAPIMAnalyticsEnabled; - private boolean isOBDataPublishingEnabled; - private String workerThreadCount; - private String clientTransportCertHeaderName; - private boolean isUrlEncodeClientTransportCertHeaderEnabled; - - private GatewayDataHolder() { - - } - - public static GatewayDataHolder getInstance() { - - if (instance == null) { - synchronized (GatewayDataHolder.class) { - if (instance == null) { - instance = new GatewayDataHolder(); - } - } - } - return instance; - } - - public static CloseableHttpClient getHttpClient() throws OpenBankingException { - - if (httpClient == null) { - synchronized (GatewayDataHolder.class) { - if (httpClient == null) { - httpClient = HTTPClientUtils.getHttpsClient(); - } - } - } - return httpClient; - } - - public static GatewayCache getGatewayCache() { - - if (gatewayCache == null) { - synchronized (GatewayDataHolder.class) { - if (gatewayCache == null) { - gatewayCache = new GatewayCache(); - } - } - } - return gatewayCache; - } - - private static void setGatewayCache(GatewayCache cache) { - gatewayCache = cache; - } - - public OpenBankingConfigurationService getOpenBankingConfigurationService() { - - return openBankingConfigurationService; - } - - public void setOpenBankingConfigurationService( - OpenBankingConfigurationService openBankingConfigurationService) { - - this.openBankingConfigurationService = openBankingConfigurationService; - if (openBankingConfigurationService != null) { - this.configurations = openBankingConfigurationService.getConfigurations(); - AbstractRequestRouter configuredRequestRouter = (AbstractRequestRouter) - OpenBankingUtils.getClassInstanceFromFQN(configurations.get(GatewayConstants.REQUEST_ROUTER) - .toString()); - setGatewayCacheAccessExpiry((String) configurations.get(GatewayConstants.GATEWAY_CACHE_EXPIRY)); - setGatewayCacheModifiedExpiry((String) configurations - .get(GatewayConstants.GATEWAY_CACHE_MODIFIEDEXPIRY)); - this.urlMap = constructURLMap(); - configuredRequestRouter.build(); - this.setRequestRouter(configuredRequestRouter); - if (configurations.get(GatewayConstants.GATEWAY_THROTTLE_DATAPUBLISHER) != null) { - this.setThrottleDataPublisher((ThrottleDataPublisher) OpenBankingUtils - .getClassInstanceFromFQN(configurations.get(GatewayConstants.GATEWAY_THROTTLE_DATAPUBLISHER) - .toString())); - } - - setAPIMAnalyticsEnabled((String) configurations.get(DataPublishingConstants.APIM_ANALYTICS_ENABLED)); - setOBDataPublishingEnabled((String) configurations.get(DataPublishingConstants.DATA_PUBLISHING_ENABLED)); - setWorkerThreadCount((String) configurations.get(DataPublishingConstants.WORKER_THREAD_COUNT)); - setClientTransportCertHeaderName((String) configurations.get(OpenBankingConstants. - CLIENT_TRANSPORT_CERT_HEADER_NAME)); - setUrlEncodeClientTransportCertHeaderEnabled((String) configurations.get(OpenBankingConstants. - URL_ENCODE_CLIENT_TRANSPORT_CERT_HEADER_ENABLED)); - } - } - - public AbstractRequestRouter getRequestRouter() { - - return requestRouter; - } - - public void setRequestRouter( - AbstractRequestRouter requestRouter) { - - this.requestRouter = requestRouter; - } - - public int getGatewayCacheAccessExpiry() { - - return gatewayCacheAccessExpiry; - } - - public void setGatewayCacheAccessExpiry(String expTime) { - - this.gatewayCacheAccessExpiry = expTime == null ? 60 : Integer.parseInt(expTime); - } - - public int getGatewayCacheModifiedExpiry() { - - return gatewayCacheModifiedExpiry; - } - - public void setGatewayCacheModifiedExpiry(String expTime) { - - this.gatewayCacheModifiedExpiry = expTime == null ? 60 : Integer.parseInt(expTime); - } - - public String getKeyStoreLocation() { - - return keyStoreLocation == null ? ServerConfiguration.getInstance() - .getFirstProperty(GatewayConstants.KEYSTORE_LOCATION_TAG) : keyStoreLocation; - - } - - public void setKeyStoreLocation(String keyStoreLocation) { - - this.keyStoreLocation = keyStoreLocation; - } - - public char[] getKeyStorePassword() { - - if (this.keyStorePassword == null) { - char[] password = ServerConfiguration.getInstance() - .getFirstProperty(GatewayConstants.KEYSTORE_PASSWORD_TAG).toCharArray(); - this.keyStorePassword = password; - return Arrays.copyOf(this.keyStorePassword, this.keyStorePassword.length); - } else { - return Arrays.copyOf(this.keyStorePassword, this.keyStorePassword.length); - } - } - - public void setKeyStorePassword(char[] keyStorePassword) { - - if (keyStorePassword != null) { - this.keyStorePassword = Arrays.copyOf(keyStorePassword, keyStorePassword.length); - } - } - - public String getKeyAlias() { - - return keyAlias == null ? ServerConfiguration.getInstance() - .getFirstProperty(GatewayConstants.SIGNING_ALIAS_TAG) : keyAlias; - } - - public void setKeyAlias(String keyAlias) { - - this.keyAlias = keyAlias; - } - - public String getKeyPassword() { - - return keyPassword == null ? ServerConfiguration.getInstance() - .getFirstProperty(GatewayConstants.SIGNING_KEY_PASSWORD) : keyPassword; - } - - public void setKeyPassword(String keyPassword) { - - this.keyPassword = keyPassword; - } - - public void setApiManagerConfiguration(APIManagerConfigurationService apiManagerConfigurationService) { - - this.apiManagerConfigurationService = apiManagerConfigurationService; - } - - public APIManagerConfigurationService getApiManagerConfigurationService() { - - return apiManagerConfigurationService; - } - - public Map getUrlMap() { - - return urlMap; - } - - public void setUrlMap(Map configurations) { - - this.urlMap = configurations; - } - - private Map constructURLMap() { - - Map urlMap = new HashMap<>(); - //get admin credentials - APIManagerConfiguration config = apiManagerConfigurationService.getAPIManagerConfiguration(); - - String adminUsername = config.getFirstProperty(APIConstants.API_KEY_VALIDATOR_USERNAME); - urlMap.put(GatewayConstants.USERNAME, adminUsername); - - char[] adminPassword = config.getFirstProperty(APIConstants.API_KEY_VALIDATOR_PASSWORD).toCharArray(); - urlMap.put(GatewayConstants.PASSWORD, adminPassword); - - //read APIM store hostname - String apimStoreHostName = configurations.get(OpenBankingConstants.STORE_HOSTNAME).toString(); - if (!apimStoreHostName.endsWith("/")) { - apimStoreHostName = apimStoreHostName.concat("/"); - } - - //set the url for obtaining a token - String tokenURL = configurations.get(OpenBankingConstants.TOKEN_ENDPOINT).toString(); - urlMap.put(GatewayConstants.TOKEN_URL, tokenURL); - - //set the url for apim application creation - String applicationCreationURL = - apimStoreHostName.concat(configurations.get(OpenBankingConstants.APIM_APPCREATION).toString()); - urlMap.put(GatewayConstants.APP_CREATE_URL, applicationCreationURL); - - // set the url for mapping apim app keys to IAM service provider keys - String mapApplicationKeysURL = - apimStoreHostName.concat(configurations.get(OpenBankingConstants.APIM_KEYGENERATION).toString()); - urlMap.put(GatewayConstants.KEY_MAP_URL, mapApplicationKeysURL); - - //set the url to retrieve all published APIs - String retrieveAPIsURL = - apimStoreHostName.concat(configurations.get(OpenBankingConstants.APIM_GETAPIS).toString()); - urlMap.put(GatewayConstants.API_RETRIEVE_URL, retrieveAPIsURL); - - //set the url to subscribe to APIS - String subscribeAPIsURL = - apimStoreHostName.concat(configurations.get(OpenBankingConstants.APIM_SUBSCRIBEAPIS).toString()); - urlMap.put(GatewayConstants.API_SUBSCRIBE_URL, subscribeAPIsURL); - - //set the url to retrieve the subscriptions for an application - //set the url to get subscribed APIS - String retrieveSubscriptionURL = - apimStoreHostName.concat(configurations.get(OpenBankingConstants.APIM_GETSUBSCRIPTIONS).toString()); - urlMap.put(GatewayConstants.API_GET_SUBSCRIBED, retrieveSubscriptionURL); - - String iamHostName = GatewayDataHolder.getInstance() - .getApiManagerConfigurationService().getAPIManagerConfiguration() - .getFirstProperty("APIKeyValidator.ServerURL").split("/services")[0]; - - urlMap.put(GatewayConstants.IAM_HOSTNAME, iamHostName); - - String iamDCREndpoint = IdentityUtil.getProperty("OAuth.OAuth2DCREPUrl").split("/api/")[1]; - iamDCREndpoint = iamHostName.concat("/api/").concat(iamDCREndpoint); - urlMap.put(GatewayConstants.IAM_DCR_URL, iamDCREndpoint); - return urlMap; - } - - public ThrottleDataPublisher getThrottleDataPublisher() { - - return throttleDataPublisher; - } - - public void setThrottleDataPublisher( - ThrottleDataPublisher throttleDataPublisher) { - - this.throttleDataPublisher = throttleDataPublisher; - } - - public boolean isAPIMAnalyticsEnabled() { - - return isAPIMAnalyticsEnabled; - } - - public void setAPIMAnalyticsEnabled(String apimAnalyticsEnabled) { - - isAPIMAnalyticsEnabled = Boolean.parseBoolean(apimAnalyticsEnabled); - } - - public boolean isOBDataPublishingEnabled() { - - return isOBDataPublishingEnabled; - } - - public void setOBDataPublishingEnabled(String obDataPublishingEnabled) { - - isOBDataPublishingEnabled = Boolean.parseBoolean(obDataPublishingEnabled); - } - - public void setWorkerThreadCount(String workerThreadCount) { - this.workerThreadCount = workerThreadCount; - } - - public String getWorkerThreadCount() { - - return workerThreadCount; - } - - public String getClientTransportCertHeaderName() { - - return clientTransportCertHeaderName; - } - - public void setClientTransportCertHeaderName(String clientTransportCertHeaderName) { - - this.clientTransportCertHeaderName = clientTransportCertHeaderName; - } - - public boolean isUrlEncodeClientTransportCertHeaderEnabled() { - - return isUrlEncodeClientTransportCertHeaderEnabled; - } - - public void setUrlEncodeClientTransportCertHeaderEnabled(String isUrlEncodeClientTransportCertHeaderEnabled) { - - this.isUrlEncodeClientTransportCertHeaderEnabled = - Boolean.parseBoolean(isUrlEncodeClientTransportCertHeaderEnabled); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/internal/GatewayServiceComponent.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/internal/GatewayServiceComponent.java deleted file mode 100644 index e5e9677f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/internal/GatewayServiceComponent.java +++ /dev/null @@ -1,87 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.internal; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.osgi.service.component.ComponentContext; -import org.osgi.service.component.annotations.Activate; -import org.osgi.service.component.annotations.Component; -import org.osgi.service.component.annotations.Deactivate; -import org.osgi.service.component.annotations.Reference; -import org.osgi.service.component.annotations.ReferenceCardinality; -import org.osgi.service.component.annotations.ReferencePolicy; -import org.wso2.carbon.apimgt.impl.APIManagerConfigurationService; - -/** - * Service class for executor core. - */ -@Component( - name = "com.wso2.open.banking.common", - immediate = true -) -public class GatewayServiceComponent { - - private static final Log log = LogFactory.getLog(GatewayServiceComponent.class); - - @Activate - protected void activate(ComponentContext context) { - - log.debug("Open banking gateway component is activated "); - } - - @Deactivate - protected void deactivate(ComponentContext context) { - - log.debug("Open banking gateway component is deactivated "); - } - - @Reference( - service = OpenBankingConfigurationService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetConfigService" - ) - public void setConfigService(OpenBankingConfigurationService openBankingConfigurationService) { - - GatewayDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - } - - public void unsetConfigService(OpenBankingConfigurationService openBankingConfigurationService) { - - GatewayDataHolder.getInstance().setOpenBankingConfigurationService(null); - } - - @Reference( - service = APIManagerConfigurationService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unSetAPIMConfigs" - ) - public void setAPIMConfig(APIManagerConfigurationService apManagerConfigurationService) { - - GatewayDataHolder.getInstance().setApiManagerConfiguration(apManagerConfigurationService); - } - - public void unSetAPIMConfigs(APIManagerConfigurationService apManagerConfigurationService) { - - GatewayDataHolder.getInstance().setApiManagerConfiguration(apManagerConfigurationService); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/internal/TPPCertValidatorComponent.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/internal/TPPCertValidatorComponent.java deleted file mode 100644 index f22e9966..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/internal/TPPCertValidatorComponent.java +++ /dev/null @@ -1,129 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.internal; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.gateway.executor.util.CertificateValidationUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.osgi.service.component.ComponentContext; -import org.osgi.service.component.annotations.Activate; -import org.osgi.service.component.annotations.Component; -import org.osgi.service.component.annotations.Deactivate; -import org.osgi.service.component.annotations.Reference; -import org.osgi.service.component.annotations.ReferenceCardinality; -import org.osgi.service.component.annotations.ReferencePolicy; -import org.wso2.carbon.apimgt.impl.APIManagerConfigurationService; -import org.wso2.carbon.base.ServerConfiguration; - -import java.io.IOException; -import java.security.KeyStoreException; -import java.security.NoSuchAlgorithmException; -import java.security.cert.CertificateException; -import java.util.concurrent.Executors; -import java.util.concurrent.ScheduledExecutorService; -import java.util.concurrent.ScheduledFuture; -import java.util.concurrent.TimeUnit; - -/** - * Service Component For Gateway Component. - **/ -@Component(name = "com.wso2.openbanking.accelerator.gateway.internal.TPPCertValidatorComponent", - immediate = true) -public class TPPCertValidatorComponent { - - private static final Log log = LogFactory.getLog(TPPCertValidatorComponent.class); - private static final Integer SCHEDULED_INITIAL_DELAY_IN_SECONDS = 1; - - @Activate - protected void activate(ComponentContext context) { - - Object certificateRevocationEnabled = OpenBankingConfigParser.getInstance(). - getConfiguration().get(OpenBankingConstants.CERTIFICATE_REVOCATION_VALIDATION_ENABLED); - final boolean isCertificateRevocationEnabled = - certificateRevocationEnabled != null && Boolean.parseBoolean((String) certificateRevocationEnabled); - - Object transportCertIssuerValidationEnabled = OpenBankingConfigParser.getInstance(). - getConfiguration().get(OpenBankingConstants.TRANSPORT_CERT_ISSUER_VALIDATION_ENABLED); - final boolean isTransportCertIssuerValidationEnabled = transportCertIssuerValidationEnabled != null - && Boolean.parseBoolean((String) transportCertIssuerValidationEnabled); - - // Loading truststore - if (isCertificateRevocationEnabled || isTransportCertIssuerValidationEnabled) { - ScheduledExecutorService scheduledExecutor = Executors.newScheduledThreadPool(1); - Runnable readTruststore = () -> { - try { - CertificateValidationUtils.loadTrustStore( - ServerConfiguration.getInstance().getFirstProperty(CertificateValidationUtils - .TRUSTSTORE_PASS_CONF_KEY).toCharArray()); - log.info("client truststore successfully loaded into certificate validator"); - } catch (KeyStoreException | IOException | CertificateException | NoSuchAlgorithmException e) { - log.error("Unable to load the client truststore", e); - } - }; - - // Initiate the scheduled truststore loading with an interval value configured as the truststore - // dynamic loading interval in open-banking.xml. - ScheduledFuture scheduledFuture = scheduledExecutor.scheduleAtFixedRate(readTruststore, - SCHEDULED_INITIAL_DELAY_IN_SECONDS, OpenBankingConfigParser.getInstance() - .getTruststoreDynamicLoadingInterval(), TimeUnit.SECONDS); - if (scheduledFuture.isCancelled()) { - log.error("Error occurred while loading the client truststore into certificate validator"); - } - } - TPPCertValidatorDataHolder.getInstance().initializeTPPValidationDataHolder(); - log.debug("OB Gateway component is activated "); - } - - @Deactivate - protected void deactivate(ComponentContext ctxt) { - log.debug("Client registration validation handler is deactivated"); - } - - @Reference( - service = OpenBankingConfigurationService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetConfigService" - ) - public void setConfigService(OpenBankingConfigurationService openBankingConfigurationService) { - TPPCertValidatorDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - } - - public void unsetConfigService(OpenBankingConfigurationService openBankingConfigurationService) { - TPPCertValidatorDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - } - - @Reference(name = "api.manager.config.service", - service = APIManagerConfigurationService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetAPIManagerConfigurationService" - ) - protected void setAPIConfigurationService(APIManagerConfigurationService confService) { - log.debug("API manager configuration service bound to the OB Gateway component"); - TPPCertValidatorDataHolder.getInstance().setApiManagerConfiguration(confService); - } - - protected void unsetAPIManagerConfigurationService(APIManagerConfigurationService amcService) { - log.debug("API manager configuration service unbound from the OB Gateway component"); - TPPCertValidatorDataHolder.getInstance().setApiManagerConfiguration(null); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/internal/TPPCertValidatorDataHolder.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/internal/TPPCertValidatorDataHolder.java deleted file mode 100644 index 1ac87fa2..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/internal/TPPCertValidatorDataHolder.java +++ /dev/null @@ -1,476 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.internal; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.gateway.executor.service.TPPValidationService; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.apimgt.impl.APIManagerConfigurationService; - -import java.util.ArrayList; -import java.util.List; - -/** - * Data Holder For Gateway Component. - **/ -public class TPPCertValidatorDataHolder { - - private static final Log log = LogFactory.getLog(TPPCertValidatorDataHolder.class); - - private static TPPCertValidatorDataHolder instance = null; - - private int tppValidationCacheExpiry; - private int tppCertRevocationCacheExpiry; - private int certificateRevocationProxyPort; - private int certificateRevocationValidationRetryCount; - private int connectTimeout; - private int connectionRequestTimeout; - private int socketTimeout; - - private boolean psd2RoleValidationEnabled; - private boolean certificateRevocationProxyEnabled; - private boolean transportCertIssuerValidationEnabled; - private boolean certificateRevocationValidationEnabled; - - private String tppValidationServiceImpl; - private String certificateRevocationProxyHost; - - private List revocationValidationExcludedIssuersList; - - private TPPValidationService tppValidationService; - private OpenBankingConfigurationService openBankingConfigurationService; - private APIManagerConfigurationService apiManagerConfigurationService; - - private TPPCertValidatorDataHolder() { - // Disable direct object creation - } - - public static synchronized TPPCertValidatorDataHolder getInstance() { - if (instance == null) { - instance = new TPPCertValidatorDataHolder(); - } - return instance; - } - - public TPPValidationService getTppValidationService() { - return tppValidationService; - } - - public void setTppValidationService() { - if (isTppValidationEnabled()) { - - String tppValidationServiceImplClass = getTPPValidationServiceImpl(); - if (StringUtils.isNotBlank(tppValidationServiceImplClass)) { - try { - this.tppValidationService = (TPPValidationService) Class.forName(tppValidationServiceImplClass) - .newInstance(); - } catch (ClassNotFoundException e) { - log.error("Unable to find the TPP validation service class " + - "implementation", e); - } catch (InstantiationException | IllegalAccessException e) { - log.error("Error occurred while loading the TPP validation " + - "service class implementation", e); - } - } else { - log.error("TPP validation service class implementation cannot be empty"); - } - } - } - - /** - * Certificate revocation cache expiry time has been configured in the open-banking.xml. - * Default value is 3600 seconds. - */ - public int getTppCertRevocationCacheExpiry() { - return this.tppCertRevocationCacheExpiry; - } - - public void setTppCertRevocationCacheExpiry() { - try { - Object clientCertificateCacheExpiry = this.openBankingConfigurationService. - getConfigurations().get(OpenBankingConstants.CLIENT_CERTIFICATE_CACHE_EXPIRY); - if (clientCertificateCacheExpiry != null) { - this.tppCertRevocationCacheExpiry = Integer.parseInt((String) clientCertificateCacheExpiry); - } else { - this.tppCertRevocationCacheExpiry = 3600; - } - } catch (NumberFormatException e) { - throw new NumberFormatException("Error occurred while reading the client certificate cache expiry value " + - "in open-banking.xml. caused by, " + e.getMessage()); - } - } - - /** - * Tpp validation cache expiry time has been configured in the open-banking.xml. - * Default value is 3600 seconds. - */ - public int getTppValidationCacheExpiry() { - return this.tppValidationCacheExpiry; - } - - public void setTppValidationCacheExpiry() { - try { - Object clientCertificateCacheExpiry = this.openBankingConfigurationService. - getConfigurations().get(OpenBankingConstants.TPP_VALIDATION_CACHE_EXPIRY); - if (clientCertificateCacheExpiry != null) { - this.tppValidationCacheExpiry = Integer.parseInt((String) clientCertificateCacheExpiry); - } else { - this.tppValidationCacheExpiry = 3600; - } - } catch (NumberFormatException e) { - throw new NumberFormatException("Error occurred while reading the tpp validation cache expiry value " + - "in open-banking.xml. caused by, " + e.getMessage()); - } - } - - /** - * Check if the certificate revocation validation is enabled. - * - * @return true if the certificate validation is enabled. Default value has been sent to true. - */ - public boolean isCertificateRevocationValidationEnabled() { - return this.certificateRevocationValidationEnabled; - } - - public void setCertificateRevocationValidationEnabled() { - - Object isCertificateRevocationEnabled = this.openBankingConfigurationService. - getConfigurations().get(OpenBankingConstants.CERTIFICATE_REVOCATION_VALIDATION_ENABLED); - if (isCertificateRevocationEnabled != null) { - this.certificateRevocationValidationEnabled = Boolean.parseBoolean((String) isCertificateRevocationEnabled); - } else { - this.certificateRevocationValidationEnabled = true; - } - } - - public void setCertificateRevocationValidationExcludedIssuers() { - Object revocationValidationExcludedIssuers = this.openBankingConfigurationService. - getConfigurations().get(OpenBankingConstants.CERTIFICATE_REVOCATION_VALIDATION_EXCLUDED_ISSUERS); - - this.revocationValidationExcludedIssuersList = new ArrayList<>(); - if (revocationValidationExcludedIssuers instanceof ArrayList) { - revocationValidationExcludedIssuersList.addAll((ArrayList) revocationValidationExcludedIssuers); - } else if (revocationValidationExcludedIssuers instanceof String) { - revocationValidationExcludedIssuersList.add((String) revocationValidationExcludedIssuers); - } - } - - /** - * Get the certificate issuers whose issued certificates are excluded from revocation validation. - * - * @return Returns a list of certificate issuers whose issued certificates are excluded from revocation validation - */ - public List getCertificateRevocationValidationExcludedIssuers() { - return this.revocationValidationExcludedIssuersList; - - } - - /** - * Validate the TPP using external service implementation. - * - * @return Default value has been set to false - */ - public boolean isTppValidationEnabled() { - Object isTppValidationEnabled = this.openBankingConfigurationService. - getConfigurations().get(OpenBankingConstants.TPP_VALIDATION_ENABLED); - if (isTppValidationEnabled != null) { - return Boolean.parseBoolean((String) isTppValidationEnabled); - } else { - return false; - } - } - - public void setCertificateRevocationValidationRetryCount() { - try { - Object certificateRevocationValidationRetryCountObj = this.openBankingConfigurationService. - getConfigurations().get(OpenBankingConstants.CERTIFICATE_REVOCATION_VALIDATION_RETRY_COUNT); - if (certificateRevocationValidationRetryCountObj != null) { - this.certificateRevocationValidationRetryCount = - Integer.parseInt((String) certificateRevocationValidationRetryCountObj); - } else { - this.certificateRevocationValidationRetryCount = 3; - } - } catch (NumberFormatException e) { - throw new NumberFormatException("Error occurred while reading the certificate revocation validation" + - " retry count in open-banking.xml. " + e.getMessage()); - } - } - - /** - * Get the certificate revocation validation retry count for CRL and OCSP validations. - * - * @return returns the certificate revocation validation retry count for CRL and OCSP validations. - * Default value has been set to 3. - */ - public int getCertificateRevocationValidationRetryCount() { - return this.certificateRevocationValidationRetryCount; - } - - /** - * Get the certificate revocation validation connectTimeout for CRL and OCSP validations. - * - * @return returns the certificate revocation validation connectTimeout for CRL and OCSP validations. - */ - public int getConnectTimeout() { - - return connectTimeout; - } - - /** - * Set the certificate revocation validation connectTimeout for CRL and OCSP validations. - * - */ - public void setConnectTimeout() { - try { - Object certValidationConnectTimeout = this.openBankingConfigurationService. - getConfigurations().get(OpenBankingConstants.CERTIFICATE_REVOCATION_VALIDATION_CONNECT_TIMEOUT); - if (certValidationConnectTimeout != null) { - this.connectTimeout = Integer.parseInt((String) certValidationConnectTimeout); - } else { - // Default value has been set to 10000 milliseconds. - this.connectTimeout = 10000; - } - } catch (NumberFormatException e) { - throw new NumberFormatException("Error occurred while reading the connectTimeout value " + - "in open-banking.xml. caused by, " + e.getMessage()); - } - } - - /** - * Get the certificate revocation validation connectionRequestTimeout for CRL and OCSP validations. - * - * @return returns the certificate revocation validation connectionRequestTimeout for CRL and OCSP validations. - */ - public int getConnectionRequestTimeout() { - - return connectionRequestTimeout; - } - - /** - * Set the certificate revocation validation connectionRequestTimeout for CRL and OCSP validations. - * - */ - public void setConnectionRequestTimeout() { - - try { - Object certValidationConnectionRequestTimeout = this.openBankingConfigurationService.getConfigurations() - .get(OpenBankingConstants.CERTIFICATE_REVOCATION_VALIDATION_CONNECTION_REQUEST_TIMEOUT); - if (certValidationConnectionRequestTimeout != null) { - this.connectionRequestTimeout = Integer.parseInt((String) certValidationConnectionRequestTimeout); - } else { - // Default value has been set to 10000 milliseconds. - this.connectionRequestTimeout = 10000; - } - } catch (NumberFormatException e) { - throw new NumberFormatException("Error occurred while reading the connection request timeout value " + - "in open-banking.xml. caused by, " + e.getMessage()); - } - } - - /** - * Get the certificate revocation validation socketTimeout for CRL and OCSP validations. - * - * @return returns the certificate revocation validation socketTimeout for CRL and OCSP validations. - */ - public int getSocketTimeout() { - - return socketTimeout; - } - - /** - * Set the certificate revocation validation socketTimeout for CRL and OCSP validations. - * - */ - public void setSocketTimeout() { - - try { - Object certValidationSocketTimeout = this.openBankingConfigurationService.getConfigurations() - .get(OpenBankingConstants.CERTIFICATE_REVOCATION_VALIDATION_SOCKET_TIMEOUT); - if (certValidationSocketTimeout != null) { - this.socketTimeout = Integer.parseInt((String) certValidationSocketTimeout); - } else { - // Default value has been set to 10000 milliseconds. - this.socketTimeout = 10000; - } - } catch (NumberFormatException e) { - throw new NumberFormatException("Error occurred while reading the socket timeout value " + - "in open-banking.xml. caused by, " + e.getMessage()); - } - } - - /** - * Get the certificate revocation validation manager implementation class to validate the revocation status - * of a certificate. - * - * @return class name of the certificate revocation validator implementation class - */ - public String getTPPValidationServiceImpl() { - return this.tppValidationServiceImpl; - } - - public void setTPPValidationServiceImpl() { - Object revocationValidationManagerImpl = this.openBankingConfigurationService. - getConfigurations().get(OpenBankingConstants.TPP_VALIDATION_SERVICE_IMPL_CLASS); - if (revocationValidationManagerImpl != null) { - this.tppValidationServiceImpl = String.valueOf(revocationValidationManagerImpl) - .replaceAll("[\\n\\t ]", ""); - } - } - - /** - * Returns whether the certification revocation proxy is enabled. - *

- * If enabled, the certificate revocation checks will be done via the configured proxy - * - * @return {@code true} if certificate revocation proxy is enabled, {@code false} otherwise. The default value is - * {@code false} - */ - public boolean isCertificateRevocationProxyEnabled() { - return certificateRevocationProxyEnabled; - } - - public void setCertificateRevocationProxyEnabled() { - Object isCertificateRevocationProxyEnabled = this.openBankingConfigurationService. - getConfigurations().get(OpenBankingConstants.CERTIFICATE_REVOCATION_PROXY_ENABLED); - if (isCertificateRevocationProxyEnabled != null) { - this.certificateRevocationProxyEnabled = Boolean.parseBoolean((String) isCertificateRevocationProxyEnabled); - } else { - this.certificateRevocationProxyEnabled = false; - } - } - - /** - * Returns the certificate revocation proxy port. - *

- * The certificate revocation checks will be done via this proxy port, if the - * {@code CertificateRevocationProxyEnabled} value is set to {@code true} - * - * @return certificate revocation proxy port - */ - public int getCertificateRevocationProxyPort() { - return this.certificateRevocationProxyPort; - } - - public void setCertificateRevocationProxyPort() { - Object certificateRevocationProxyPortObj = this.openBankingConfigurationService. - getConfigurations().get(OpenBankingConstants.CERTIFICATE_REVOCATION_PROXY_PORT); - this.certificateRevocationProxyPort = certificateRevocationProxyPortObj != null ? - Integer.parseInt((String) certificateRevocationProxyPortObj) : 8080; - } - - /** - * Returns the certificate revocation proxy host. - *

- * The certificate revocation checks will be done via this proxy host, if the - * {@code CertificateRevocationProxyEnabled} value is set to {@code true} - * - * @return certificate revocation proxy host - */ - public String getCertificateRevocationProxyHost() { - return this.certificateRevocationProxyHost; - } - - public void setCertificateRevocationProxyHost() { - Object certificateRevocationProxyHostObj = this.openBankingConfigurationService. - getConfigurations().get(OpenBankingConstants.CERTIFICATE_REVOCATION_PROXY_HOST); - this.certificateRevocationProxyHost = - certificateRevocationProxyHostObj != null ? ((String) certificateRevocationProxyHostObj).trim() : ""; - } - - /** - * Validate the issuer of the client certificate when the client certificate is sent as a header when the TLS - * session is terminated before gateway. - * - * @return Default value has been set to true - */ - public boolean isTransportCertIssuerValidationEnabled() { - return this.transportCertIssuerValidationEnabled; - } - - public void setTransportCertIssuerValidationEnabled() { - Object transportCertIssuerValidationEnabledObj = this.openBankingConfigurationService. - getConfigurations().get(OpenBankingConstants.TRANSPORT_CERT_ISSUER_VALIDATION_ENABLED); - if (transportCertIssuerValidationEnabledObj != null) { - this.transportCertIssuerValidationEnabled = - Boolean.parseBoolean((String) transportCertIssuerValidationEnabledObj); - } else { - this.transportCertIssuerValidationEnabled = true; - } - } - - /** - * Validate the TPP PSD2 roles. - * - * @return Default value has been set to true - */ - public boolean isPsd2RoleValidationEnabled() { - return this.psd2RoleValidationEnabled; - } - - public void setPsd2RoleValidationEnabled() { - Object psd2RoleValidationEnabledObj = this.openBankingConfigurationService. - getConfigurations().get(OpenBankingConstants.PSD2_ROLE_VALIDATION_ENABLED); - if (psd2RoleValidationEnabledObj != null) { - this.psd2RoleValidationEnabled = Boolean.parseBoolean((String) psd2RoleValidationEnabledObj); - } else { - this.psd2RoleValidationEnabled = true; - } - } - - public OpenBankingConfigurationService getOpenBankingConfigurationService() { - return openBankingConfigurationService; - } - - public void setOpenBankingConfigurationService(OpenBankingConfigurationService openBankingConfigurationService) { - this.openBankingConfigurationService = openBankingConfigurationService; - } - - public void setApiManagerConfiguration(APIManagerConfigurationService apiManagerConfigurationService) { - - this.apiManagerConfigurationService = apiManagerConfigurationService; - } - - public APIManagerConfigurationService getApiManagerConfigurationService() { - - return apiManagerConfigurationService; - } - - public void initializeTPPValidationDataHolder() { - setTPPValidationServiceImpl(); - setTppValidationService(); - - setTppValidationCacheExpiry(); - setPsd2RoleValidationEnabled(); - setTppCertRevocationCacheExpiry(); - setCertificateRevocationProxyHost(); - setCertificateRevocationProxyPort(); - setCertificateRevocationProxyEnabled(); - setTransportCertIssuerValidationEnabled(); - setCertificateRevocationValidationEnabled(); - setCertificateRevocationValidationRetryCount(); - setCertificateRevocationValidationExcludedIssuers(); - setConnectTimeout(); - setConnectionRequestTimeout(); - setSocketTimeout(); - - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/mediator/BasicAuthMediator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/mediator/BasicAuthMediator.java deleted file mode 100644 index cbac8b26..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/mediator/BasicAuthMediator.java +++ /dev/null @@ -1,61 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.mediator; - -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.synapse.MessageContext; -import org.apache.synapse.mediators.AbstractMediator; - -import java.nio.charset.StandardCharsets; -import java.util.Base64; - -/** - * Append Basic Authorisation header to the API calls from gateway to be passed on to the - * key management server. Invoked by the synapse in-sequence.xml files. - */ -public class BasicAuthMediator extends AbstractMediator { - - private static final Log log = LogFactory.getLog(BasicAuthMediator.class); - - public BasicAuthMediator() { - - log.info("Initializing Basic Authentication Mediator to append basic auth credentials in gateway " + - "in-sequence."); - } - - @Override - public boolean mediate(MessageContext messageContext) { - - messageContext.setProperty("basicAuthentication", ("Basic " + Base64.getEncoder().encodeToString( - (getAPIMConfigFromKey(GatewayConstants.API_KEY_VALIDATOR_USERNAME) + ":" - + getAPIMConfigFromKey(GatewayConstants.API_KEY_VALIDATOR_PASSWORD)) - .getBytes(StandardCharsets.UTF_8)))); - return true; - } - - @Generated(message = "Excluded since this method is used for testing purposes") - String getAPIMConfigFromKey(String key) { - - return GatewayUtils.getAPIMgtConfig(key); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/reporter/OBAnalyticsMetricReporter.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/reporter/OBAnalyticsMetricReporter.java deleted file mode 100644 index c0c90f60..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/reporter/OBAnalyticsMetricReporter.java +++ /dev/null @@ -1,67 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.reporter; - -import com.wso2.openbanking.accelerator.gateway.internal.GatewayDataHolder; -import org.wso2.am.analytics.publisher.exception.MetricCreationException; -import org.wso2.am.analytics.publisher.reporter.AbstractMetricReporter; -import org.wso2.am.analytics.publisher.reporter.CounterMetric; -import org.wso2.am.analytics.publisher.reporter.MetricSchema; -import org.wso2.am.analytics.publisher.reporter.TimerMetric; -import org.wso2.am.analytics.publisher.reporter.cloud.DefaultAnalyticsMetricReporter; - -import java.util.Map; - -/** - * OB Analytics Metric Reporter class. - */ -public class OBAnalyticsMetricReporter extends AbstractMetricReporter { - - private DefaultAnalyticsMetricReporter defaultAnalyticsMetricReporter; - - public OBAnalyticsMetricReporter(Map properties) throws MetricCreationException { - - super(properties); - if (GatewayDataHolder.getInstance().isAPIMAnalyticsEnabled()) { - defaultAnalyticsMetricReporter = new DefaultAnalyticsMetricReporter(properties); - } - } - - @Override - protected void validateConfigProperties(Map map) throws MetricCreationException { - - } - - @Override - protected CounterMetric createCounter(String name, MetricSchema schema) throws MetricCreationException { - - CounterMetric counterMetric = null; - if (GatewayDataHolder.getInstance().isAPIMAnalyticsEnabled()) { - counterMetric = defaultAnalyticsMetricReporter.createCounterMetric(name, schema); - } - return new OBCounterMetric(name, counterMetric, schema); - } - - @Override - protected TimerMetric createTimer(String s) { - - return null; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/reporter/OBCounterMetric.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/reporter/OBCounterMetric.java deleted file mode 100644 index b507c31a..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/reporter/OBCounterMetric.java +++ /dev/null @@ -1,111 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.reporter; - -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.gateway.internal.GatewayDataHolder; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.am.analytics.publisher.exception.MetricReportingException; -import org.wso2.am.analytics.publisher.reporter.CounterMetric; -import org.wso2.am.analytics.publisher.reporter.MetricEventBuilder; -import org.wso2.am.analytics.publisher.reporter.MetricSchema; -import org.wso2.am.analytics.publisher.reporter.cloud.DefaultChoreoFaultMetricEventBuilder; -import org.wso2.am.analytics.publisher.reporter.cloud.DefaultChoreoResponseMetricEventBuilder; -import org.wso2.am.analytics.publisher.reporter.cloud.DefaultFaultMetricEventBuilder; -import org.wso2.am.analytics.publisher.reporter.cloud.DefaultResponseMetricEventBuilder; - -import java.util.concurrent.ExecutorService; -import java.util.concurrent.Executors; - -/** - * OB Counter metric class to publish am analytics and ob analytics data. - */ -public class OBCounterMetric implements CounterMetric { - - private static final Log log = LogFactory.getLog(OBCounterMetric.class); - private final ExecutorService obExecutorService; - private final CounterMetric counterMetric; - private String name; - private MetricSchema schema; - - public OBCounterMetric(String name, CounterMetric counterMetric, MetricSchema schema) { - - this.counterMetric = counterMetric; - this.name = name; - this.schema = schema; - int workerThreadCount = Integer.parseInt(GatewayDataHolder.getInstance().getWorkerThreadCount()); - this.obExecutorService = Executors.newFixedThreadPool(workerThreadCount); - } - - @Override - public int incrementCount(MetricEventBuilder builder) { - - int status; - - if (GatewayDataHolder.getInstance().isAPIMAnalyticsEnabled()) { - try { - status = counterMetric.incrementCount(builder); - } catch (MetricReportingException e) { - log.error("Error while publishing APIM analytics", e); - status = 0; - } - } else { - log.debug("APIM analytics disabled."); - status = 0; - } - - // OB data publishing - if (GatewayDataHolder.getInstance().isOBDataPublishingEnabled()) { - obExecutorService.submit(new OBTimestampPublisher(builder)); - } - return status; - } - - @Override - public String getName() { - - return this.name; - } - - @Override - public MetricSchema getSchema() { - - return this.schema; - } - - @Override - @Generated(message = "This is skipped because schema cannot be mocked") - public MetricEventBuilder getEventBuilder() { - - switch (schema) { - case RESPONSE: - return new DefaultResponseMetricEventBuilder(); - case ERROR: - return new DefaultFaultMetricEventBuilder(); - case CHOREO_RESPONSE: - return new DefaultChoreoResponseMetricEventBuilder(); - case CHOREO_ERROR: - return new DefaultChoreoFaultMetricEventBuilder(); - default: - // will not happen - return null; - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/reporter/OBTimestampPublisher.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/reporter/OBTimestampPublisher.java deleted file mode 100644 index 4ac020ab..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/reporter/OBTimestampPublisher.java +++ /dev/null @@ -1,82 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.reporter; - -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.data.publisher.common.util.OBDataPublisherUtil; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.am.analytics.publisher.exception.MetricReportingException; -import org.wso2.am.analytics.publisher.reporter.MetricEventBuilder; - -import java.time.Duration; -import java.time.Instant; -import java.util.HashMap; -import java.util.Map; - -/** - * OB timestamp publisher worker class. - */ -public class OBTimestampPublisher implements Runnable { - - private MetricEventBuilder builder; - private static final String CORRELATION_ID = "correlationId"; - private static final String REQUEST_TIMESTAMP = "requestTimestamp"; - private static final String BACKEND_LATENCY = "backendLatency"; - private static final String REQUEST_MEDIATION_LATENCY = "requestMediationLatency"; - private static final String RESPONSE_LATENCY = "responseLatency"; - private static final String RESPONSE_MEDIATION_LATENCY = "responseMediationLatency"; - private static final String API_LATENCY_INPUT_STREAM = "APILatencyInputStream"; - private static final String API_LATENCY_STREAM_VERSION = "1.0.0"; - private static final Log log = LogFactory.getLog(OBTimestampPublisher.class); - - public OBTimestampPublisher(MetricEventBuilder builder) { - - this.builder = builder; - } - - public void run() { - - try { - Map eventMap = builder.build(); - Map analyticsData = new HashMap<>(); - Object requestTimestamp = eventMap.get(REQUEST_TIMESTAMP); - analyticsData.put(CORRELATION_ID, eventMap.get(CORRELATION_ID)); - analyticsData.put(REQUEST_TIMESTAMP, requestTimestamp); - analyticsData.put(BACKEND_LATENCY, - eventMap.get(BACKEND_LATENCY) != null ? eventMap.get(BACKEND_LATENCY) : 0L); - analyticsData.put(REQUEST_MEDIATION_LATENCY, - eventMap.get(REQUEST_MEDIATION_LATENCY) != null ? eventMap.get(REQUEST_MEDIATION_LATENCY) : 0L); - analyticsData.put(RESPONSE_LATENCY, - eventMap.get(RESPONSE_LATENCY) != null ? eventMap.get(RESPONSE_LATENCY) - : Duration.between(Instant.parse(requestTimestamp.toString()), Instant.now()).toMillis()); - analyticsData.put(RESPONSE_MEDIATION_LATENCY, eventMap.get(RESPONSE_MEDIATION_LATENCY) != null ? - eventMap.get(RESPONSE_MEDIATION_LATENCY) : 0L); - publishLatencyData(analyticsData); - } catch (MetricReportingException e) { - log.error("Error while collecting latency stats", e); - } - } - - @Generated(message = "This method is already covered") - protected void publishLatencyData(Map analyticsData) { - OBDataPublisherUtil.publishData(API_LATENCY_INPUT_STREAM, API_LATENCY_STREAM_VERSION, analyticsData); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/synapse/handler/DisputeResolutionSynapseHandler.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/synapse/handler/DisputeResolutionSynapseHandler.java deleted file mode 100644 index 5bd95d8a..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/synapse/handler/DisputeResolutionSynapseHandler.java +++ /dev/null @@ -1,187 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.synapse.handler; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.common.util.OpenBankingUtils; -import com.wso2.openbanking.accelerator.data.publisher.common.util.OBDataPublisherUtil; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.synapse.AbstractSynapseHandler; -import org.apache.synapse.MessageContext; -import org.apache.synapse.core.axis2.Axis2MessageContext; -import org.wso2.carbon.apimgt.impl.APIConstants; - -import java.time.Instant; -import java.util.HashMap; -import java.util.Map; -import java.util.Optional; - -/** - * Dispute Resolution Synapse Handler. - */ -public class DisputeResolutionSynapseHandler extends AbstractSynapseHandler { - private static final Log log = LogFactory.getLog(DisputeResolutionSynapseHandler.class); - - /** - * Handle request message coming into the engine. - * - * @param messageContext incoming request message context - * @return whether mediation flow should continue - */ - @Override - public boolean handleRequestInFlow(MessageContext messageContext) { - - //Checking Dispute Resolution Feature is Enabled - if (!OpenBankingConfigParser.getInstance().isDisputeResolutionEnabled()) { - return true; - } - - org.apache.axis2.context.MessageContext axis2MC = - ((Axis2MessageContext) messageContext).getAxis2MessageContext(); - Map headers = (Map) axis2MC.getProperty(org.apache.axis2.context.MessageContext.TRANSPORT_HEADERS); - - Map contextEntries = messageContext.getContextEntries(); - - //Extracting Request Body - Optional requestBody = Optional.empty(); - try { - requestBody = GatewayUtils.buildMessagePayloadFromMessageContext(axis2MC, headers); - } catch (OpenBankingException e) { - log.error("Unable to build request payload", e); - } - - contextEntries.put(OpenBankingConstants.REQUEST_BODY, - requestBody.isPresent() ? requestBody.get() : null); - return true; - } - - /** - * Handle request message going out from the engine. - * - * @param messageContext outgoing request message context - * @return whether mediation flow should continue - */ - @Override - @Generated(message = "Ignoring since method contains no logics") - public boolean handleRequestOutFlow(MessageContext messageContext) { - return true; - } - - /** - * Handle response message coming into the engine. - * - * @param messageContext incoming response message context - * @return whether mediation flow should continue - */ - @Override - @Generated(message = "Ignoring since method contains no logics") - public boolean handleResponseInFlow(MessageContext messageContext) { - return true; - } - - /** - * Handle response message going out from the engine. - * - * @param messageContext outgoing response message context - * @return whether mediation flow should continue - */ - @Override - public boolean handleResponseOutFlow(MessageContext messageContext) { - - //Checking Dispute Resolution Feature is Enabled - if (!OpenBankingConfigParser.getInstance().isDisputeResolutionEnabled()) { - return true; - } - - org.apache.axis2.context.MessageContext axis2MessageContext - = ((Axis2MessageContext) messageContext).getAxis2MessageContext(); - - Map disputeResolutionData = new HashMap<>(); - int statusCode = 0; - String httpMethod = null; - String headers = null; - long unixTimestamp = Instant.now().getEpochSecond(); - String electedResource = (String) messageContext.getProperty(APIConstants.API_ELECTED_RESOURCE); - String responseBody = null; - String requestBody = null; - - //Extracting Status Code - String stringStatusCode = axis2MessageContext.getProperty(GatewayConstants.HTTP_SC).toString(); - statusCode = Integer.parseInt(stringStatusCode); - - //Extracting Headers - Map headerMap = (Map) axis2MessageContext - .getProperty(org.apache.axis2.context.MessageContext.TRANSPORT_HEADERS); - headers = headerMap.toString(); - - //Extracting HTTP Method - if (messageContext.getProperty(GatewayConstants.HTTP_METHOD) != null) { - httpMethod = (String) messageContext.getProperty(GatewayConstants.HTTP_METHOD); - } else { - httpMethod = GatewayConstants.UNKNOWN; - } - - //Extracting Response Body - Optional response = Optional.empty(); - try { - response = GatewayUtils.buildMessagePayloadFromMessageContext(axis2MessageContext, headerMap); - } catch (OpenBankingException e) { - log.error("Unable to build response payload", e); - - } - responseBody = response.get(); - - //Extracting Request Body - Map contextEntries = messageContext.getContextEntries(); - requestBody = (String) contextEntries.get(OpenBankingConstants.REQUEST_BODY); - - //reduced Headers, Request and Response Body Lengths - requestBody = OpenBankingUtils.reduceStringLength(requestBody, - OpenBankingConfigParser.getInstance().getMaxRequestBodyLength()); - responseBody = OpenBankingUtils.reduceStringLength(responseBody, - OpenBankingConfigParser.getInstance().getMaxResponseBodyLength()); - headers = OpenBankingUtils.reduceStringLength(headers, - OpenBankingConfigParser.getInstance().getMaxHeaderLength()); - - // Add the captured data put into the disputeResolutionData Map - disputeResolutionData.put(OpenBankingConstants.STATUS_CODE, statusCode); - disputeResolutionData.put(OpenBankingConstants.HTTP_METHOD, httpMethod); - disputeResolutionData.put(OpenBankingConstants.ELECTED_RESOURCE, electedResource); - disputeResolutionData.put(OpenBankingConstants.TIMESTAMP, unixTimestamp); - disputeResolutionData.put(OpenBankingConstants.HEADERS, headers); - disputeResolutionData.put(OpenBankingConstants.RESPONSE_BODY, responseBody); - disputeResolutionData.put(OpenBankingConstants.REQUEST_BODY, requestBody); - - //Checking configurations to publish Dispute Data - if (OpenBankingUtils.isPublishableDisputeData(statusCode)) { - OBDataPublisherUtil.publishData(OpenBankingConstants.DISPUTE_RESOLUTION_STREAM_NAME, - OpenBankingConstants.DISPUTE_RESOLUTION_STREAM_VERSION, disputeResolutionData); - } - - return true; - } - -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/throttling/OBThrottlingExtensionImpl.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/throttling/OBThrottlingExtensionImpl.java deleted file mode 100644 index 3ad97057..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/throttling/OBThrottlingExtensionImpl.java +++ /dev/null @@ -1,70 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.throttling; - -import com.wso2.openbanking.accelerator.gateway.internal.GatewayDataHolder; -import org.wso2.carbon.apimgt.common.gateway.dto.ExtensionResponseDTO; -import org.wso2.carbon.apimgt.common.gateway.dto.ExtensionResponseStatus; -import org.wso2.carbon.apimgt.common.gateway.dto.RequestContextDTO; -import org.wso2.carbon.apimgt.common.gateway.dto.ResponseContextDTO; -import org.wso2.carbon.apimgt.common.gateway.extensionlistener.ExtensionListener; - -/** - * Throttling Extension listener implementation for OB. - */ -public class OBThrottlingExtensionImpl implements ExtensionListener { - - @Override - public ExtensionResponseDTO preProcessRequest(RequestContextDTO requestContextDTO) { - - ExtensionResponseDTO responseDTO = null; - ThrottleDataPublisher throttleDataPublisher = GatewayDataHolder.getInstance().getThrottleDataPublisher(); - if (throttleDataPublisher != null) { - responseDTO = new ExtensionResponseDTO(); - responseDTO.setCustomProperty(throttleDataPublisher.getCustomProperties(requestContextDTO)); - responseDTO.setResponseStatus(ExtensionResponseStatus.CONTINUE.toString()); - } - return responseDTO; - - } - - @Override - public ExtensionResponseDTO postProcessRequest(RequestContextDTO requestContextDTO) { - - return null; - } - - @Override - public ExtensionResponseDTO preProcessResponse(ResponseContextDTO responseContextDTO) { - - return null; - } - - @Override - public ExtensionResponseDTO postProcessResponse(ResponseContextDTO responseContextDTO) { - - return null; - } - - @Override - public String getType() { - - return null; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/throttling/ThrottleDataPublisher.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/throttling/ThrottleDataPublisher.java deleted file mode 100644 index 4da2df7f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/throttling/ThrottleDataPublisher.java +++ /dev/null @@ -1,32 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.throttling; - -import org.wso2.carbon.apimgt.common.gateway.dto.RequestContextDTO; - -import java.util.Map; - -/** - * Throttling data publisher interface. - */ -public interface ThrottleDataPublisher { - - public Map getCustomProperties(RequestContextDTO requestContextDTO); - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/util/GatewayConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/util/GatewayConstants.java deleted file mode 100644 index 9691cfed..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/util/GatewayConstants.java +++ /dev/null @@ -1,145 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.util; - -/** - * Gateway common constants class. - */ -public class GatewayConstants { - - public static final String CONTENT_TYPE_TAG = "Content-Type"; - public static final String CONTENT_LENGTH = "Content-Length"; - public static final String JWT_CONTENT_TYPE = "application/jwt"; - public static final String JSON_CONTENT_TYPE = "application/json"; - public static final String JOSE_CONTENT_TYPE = "application/jose"; - public static final String APPLICATION_XML_CONTENT_TYPE = "application/xml"; - public static final String TEXT_XML_CONTENT_TYPE = "text/xml"; - public static final String GET_HTTP_METHOD = "GET"; - public static final String POST_HTTP_METHOD = "POST"; - public static final String PUT_HTTP_METHOD = "PUT"; - public static final String PATCH_HTTP_METHOD = "PATCH"; - public static final String DELETE_HTTP_METHOD = "DELETE"; - public static final String ACCEPT = "Accept"; - public static final String AUTH_HEADER = "Authorization"; - public static final String BASIC_TAG = "Basic "; - public static final String BEARER_TAG = "Bearer "; - public static final String PUBLISHER_API_PATH = "api/am/publisher/apis/"; - public static final String SWAGGER_ENDPOINT = "/swagger"; - public static final String REGULATORY_CUSTOM_PROP = "x-wso2-regulatory-api"; - public static final String API_TYPE_CUSTOM_PROP = "x-wso2-api-type"; - public static final String IS_RETURN_RESPONSE = "isReturnResponse"; - public static final String MODIFIED_STATUS = "ModifiedStatus"; - public static final String APPLICATION = "application"; - public static final String APPLICATION_USER = "application_user"; - public static final String AXIS2_MTLS_CERT_PROPERTY = "ssl.client.auth.cert.X509"; - public static final String BEGIN_CERT = "-----BEGIN CERTIFICATE-----"; - public static final String END_CERT = "-----END CERTIFICATE-----"; - - //dcr related configs - public static final String AM_APP_NAME_CACHEKEY = "APP_NAME"; - public static final String APP_CREATE_URL = "APP_CREATION_URL"; - public static final String KEY_MAP_URL = "KEY_MAPPING_URL"; - public static final String API_RETRIEVE_URL = "API_RETRIEVAL_URL"; - public static final String API_SUBSCRIBE_URL = "API_SUBSCRIBE_URL"; - public static final String API_GET_SUBSCRIBED = "API_GET_SUBSCRIPTIONS"; - public static final String TOKEN_URL = "TOKEN_URL"; - public static final String USERNAME = "userName"; - public static final String IAM_DCR_URL = "DCR_Endpoint"; - public static final String PASSWORD = "password"; - public static final String IAM_HOSTNAME = "IAM_Hostname"; - public static final String VALIDATE_JWT = "DCR.RequestJWTValidation"; - - - //Config elements - public static final String CONSENT_VALIDATION_ENDPOINT_TAG = "Gateway.ConsentValidationEndpoint"; - public static final String KEYSTORE_LOCATION_TAG = "Security.InternalKeyStore.Location"; - public static final String KEYSTORE_PASSWORD_TAG = "Security.InternalKeyStore.Password"; - public static final String SIGNING_ALIAS_TAG = "Security.InternalKeyStore.KeyAlias"; - public static final String SIGNING_KEY_PASSWORD = "Security.InternalKeyStore.KeyPassword"; - public static final String API_KEY_VALIDATOR_USERNAME = "APIKeyValidator.Username"; - public static final String API_KEY_VALIDATOR_PASSWORD = "APIKeyValidator.Password"; - public static final String PUBLISHER_HOSTNAME = "PublisherURL"; - public static final String CONTEXT_PROP_CACHE_KEY = "_contextProp"; - public static final String ANALYTICS_PROP_CACHE_KEY = "_analyticsData"; - public static final String API_DATA_STREAM = "APIInputStream"; - public static final String API_DATA_VERSION = "1.0.0"; - public static final String ERROR_STATUS_PROP = "errorStatusCode"; - public static final String CONSENT_ID_CLAIM_NAME = "Identity.ConsentIDClaimName"; - public static final String REQUEST_ROUTER = "Gateway.RequestRouter"; - public static final String GATEWAY_CACHE_EXPIRY = "Gateway.Cache.GatewayCache.CacheAccessExpiry"; - public static final String GATEWAY_CACHE_MODIFIEDEXPIRY = "Gateway.Cache.GatewayCache.CacheModifiedExpiry"; - public static final String GATEWAY_THROTTLE_DATAPUBLISHER = "Gateway.CustomThrottleDataPublisher"; - - public static final String CUSTOMER_CARE_OFFICER_SCOPE = "consents:read_all"; - - public static final String API_TYPE_CONSENT = "consent"; - public static final String API_TYPE_NON_REGULATORY = "non-regulatory"; - - // Idempotency - public static final String REQUEST_CACHE_KEY = "Request"; - public static final String CREATED_TIME_CACHE_KEY = "Created_Time"; - public static final String RESPONSE_CACHE_KEY = "Response"; - public static final String TRUE = "true"; - public static final String IDEMPOTENCY_KEY_CACHE_KEY = "Idempotency_Key"; - - // Error constants - public static final String INVALID_CLIENT = "invalid_client"; - public static final String INVALID_REQUEST = "invalid_request"; - public static final String CLIENT_CERTIFICATE_MISSING = "Invalid mutual TLS request. Client certificate is missing"; - public static final String CLIENT_CERTIFICATE_INVALID = "Invalid mutual TLS request. Client certificate is invalid"; - public static final String INVALID_GRANT_TYPE = "Access failure for API: grant type validation failed."; - public static final String INVALID_CREDENTIALS = "Invalid Credentials. Make sure you have provided the " + - "correct security credentials "; - public static final String MISSING_CREDENTIALS = "Invalid Credentials. Make sure your API invocation call " + - "has a header - 'Authorization'"; - public static final String TRANSPORT_CERT_NOT_FOUND = "Valid transport certificate not found in the request"; - public static final String TRANSPORT_CERT_MALFORMED = "Provided transport certificate is malformed"; - - // Error codes - public static final int API_AUTH_INVALID_CREDENTIALS = 900901; - public static final int API_AUTH_MISSING_CREDENTIALS = 900902; - - // Oauth2 constants - public static final String AUTHORIZED_USER_TYPE_CLAIM_NAME = "aut"; - public static final String DEFAULT = "default"; - public static final String OPENID = "openid"; - - // HTTP methods - public static final String GET = "GET"; - public static final String POST = "POST"; - public static final String PUT = "PUT"; - public static final String PATCH = "PATCH"; - public static final String DELETE = "DELETE"; - - // Grant types - public static final String CLIENT_CREDENTIALS = "client_credentials"; - public static final String AUTHORIZATION_CODE = "authorization_code"; - public static final String IMPLICIT = "Implicit"; - public static final String PASSWORD_GRANT = "Password"; - - //Dispute resolution constants - public static final String UNKNOWN = "Unknown"; - public static final String ERROR_CODE = "ERROR_CODE"; - public static final String HTTP_RESPONSE_STATUS_CODE = "HTTP_RESPONSE_STATUS_CODE"; - public static final String CUSTOM_HTTP_SC = "CUSTOM_HTTP_SC"; - public static final String HTTP_SC = "HTTP_SC"; - public static final String HTTP_METHOD = "api.ut.HTTP_METHOD"; - public static final String API_BODY = "API"; - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/util/GatewaySignatureHandlingUtils.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/util/GatewaySignatureHandlingUtils.java deleted file mode 100644 index 25571307..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/util/GatewaySignatureHandlingUtils.java +++ /dev/null @@ -1,161 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.util; - -import com.nimbusds.jose.JOSEObjectType; -import com.nimbusds.jose.JWSAlgorithm; -import com.nimbusds.jose.JWSHeader; -import com.nimbusds.jose.JWSObject; -import com.nimbusds.jose.Payload; -import com.nimbusds.jose.util.Base64URL; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.identity.IdentityConstants; -import com.wso2.openbanking.accelerator.common.identity.retriever.ServerIdentityRetriever; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.gateway.executor.exception.OpenBankingExecutorException; -import org.apache.commons.lang.StringUtils; - -import java.io.UnsupportedEncodingException; -import java.nio.charset.StandardCharsets; -import java.security.Key; -import java.util.HashMap; -import java.util.Optional; - -/** - * Utility class for Signature Handling. - */ -public class GatewaySignatureHandlingUtils { - - private static final String B64_CLAIM_KEY = "b64"; - - /** - * Returns the JWS Header. - * @param kid Key id of the signing certificate. - * @param criticalParameters Hashmap of critical paramters - * @param algorithm Signing algorithm - * @return JWSHeader returns Jws Header - */ - public static JWSHeader constructJWSHeader(String kid, HashMap criticalParameters, - JWSAlgorithm algorithm) { - return new JWSHeader.Builder(algorithm) - .keyID(kid) - .type(JOSEObjectType.JOSE) - .criticalParams(criticalParameters.keySet()) - .customParams(criticalParameters) - .build(); - } - - /** - * Creates a JWS Object. - * @param header JWS header - * @param responsePayload response payload as a string - * @return JWSObject jws object created - */ - public static JWSObject constructJWSObject(JWSHeader header, String responsePayload) { - return new JWSObject(header, new Payload(responsePayload)); - } - - /** - * Returns the signing input with encoded jws header and un-encoded payload. - * @param jwsHeader JWS Header - * @param payloadString Response payload - * @return signing input - * @throws UnsupportedEncodingException throws UnsupportedEncodingException Exception - */ - public static byte[] getSigningInput(JWSHeader jwsHeader, String payloadString) - throws UnsupportedEncodingException { - - String combinedInput = jwsHeader.toBase64URL().toString() + "." + payloadString; - return combinedInput.getBytes(StandardCharsets.UTF_8); - } - - /** - * Method to create a detached jws. - * @param jwsHeader header part of the JWS - * @param signature signature part of the JWS - * @return String Detached JWS - */ - public static String createDetachedJws(JWSHeader jwsHeader, Base64URL signature) { - - return jwsHeader.toBase64URL().toString() + ".." + signature.toString(); - } - - /** - * Loads the KID of the signing certificate. - * @return String Key ID of the public key - */ - @Generated(message = "Excluding from unit tests since there is a call to a method " + - "in Common Module") - public static String getSigningKeyId() { - - return OpenBankingConfigParser.getInstance().getOBIdnRetrieverSigningCertificateKid(); - } - - /** - * Returns the signing key. - * - * @return Key Signing key - * @throws OpenBankingExecutorException throws OpenBanking Exception - */ - @Generated(message = "Excluding from unit tests since there is a call to a method " + - "in Common Module") - public static Optional getSigningKey() throws OpenBankingExecutorException { - - try { - return ServerIdentityRetriever.getPrimaryCertificate(IdentityConstants.CertificateType.SIGNING); - } catch (OpenBankingException e) { - throw new OpenBankingExecutorException("Unable to load primary signing certificate", e); - } - } - - @Generated(message = "Excluding from unit tests since a signer is required to create a valid JWSObject") - public static String createDetachedJws(String serializedJws) { - - String[] jwsParts = StringUtils.split(serializedJws, "."); - return jwsParts[0] + ".." + jwsParts[2]; - } - - /** - * JWSAlgorithm to be returned in JWS header when signing. - * can be extended at toolkit level. - * - * @return JWSAlgorithm the signing algorithm defined to use in configs - */ - @Generated(message = "Excluding from unit tests since there is a call to a method " + - "in Common Module") - public static JWSAlgorithm getSigningAlgorithm() { - - String alg = OpenBankingConfigParser.getInstance().getJwsResponseSigningAlgorithm(); - return JWSAlgorithm.parse(alg); - } - - /** - * If the b64 header is not available or is true, it is verifiable. - * - * @param jwsObject The reconstructed jws object parsed from x-jws-signature - * @return Boolean - */ - public static boolean isB64HeaderVerifiable(JWSObject jwsObject) { - - JWSHeader jwsHeader = jwsObject.getHeader(); - Object b64Value = jwsHeader.getCustomParam(B64_CLAIM_KEY); - return b64Value != null ? ((Boolean) b64Value) : true; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/util/GatewayUtils.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/util/GatewayUtils.java deleted file mode 100644 index 15a10fdc..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/util/GatewayUtils.java +++ /dev/null @@ -1,829 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.util; - -import com.nimbusds.jose.JOSEException; -import com.nimbusds.jose.JWSAlgorithm; -import com.nimbusds.jose.JWSHeader; -import com.nimbusds.jose.JWSObject; -import com.nimbusds.jose.JWSSigner; -import com.nimbusds.jose.crypto.ECDSASigner; -import com.nimbusds.jose.crypto.RSASSASigner; -import com.nimbusds.jose.util.Base64URL; -import com.wso2.openbanking.accelerator.common.error.OpenBankingErrorCodes; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingRuntimeException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.gateway.cache.GatewayCacheKey; -import com.wso2.openbanking.accelerator.gateway.executor.exception.OpenBankingExecutorException; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OpenBankingExecutorError; -import com.wso2.openbanking.accelerator.gateway.internal.GatewayDataHolder; -import io.swagger.v3.oas.models.OpenAPI; -import io.swagger.v3.oas.models.PathItem; -import io.swagger.v3.oas.models.security.OAuthFlows; -import io.swagger.v3.oas.models.security.SecurityRequirement; -import io.swagger.v3.oas.models.security.SecurityScheme; -import org.apache.axiom.om.OMElement; -import org.apache.axiom.om.util.AXIOMUtil; -import org.apache.axis2.AxisFault; -import org.apache.axis2.Constants; -import org.apache.commons.io.IOUtils; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.HttpResponse; -import org.apache.http.client.methods.HttpGet; -import org.apache.synapse.MessageContext; -import org.apache.synapse.commons.json.JsonUtil; -import org.apache.synapse.core.axis2.Axis2MessageContext; -import org.apache.synapse.core.axis2.Axis2Sender; -import org.apache.synapse.transport.nhttp.NhttpConstants; -import org.apache.synapse.transport.passthru.PassThroughConstants; -import org.apache.synapse.transport.passthru.util.RelayUtils; -import org.json.JSONException; -import org.json.JSONObject; -import org.json.XML; -import org.wso2.carbon.context.PrivilegedCarbonContext; - -import java.io.IOException; -import java.io.InputStream; -import java.io.UnsupportedEncodingException; -import java.nio.charset.StandardCharsets; -import java.security.Key; -import java.security.PrivateKey; -import java.security.cert.CertificateEncodingException; -import java.security.cert.X509Certificate; -import java.security.interfaces.ECPrivateKey; -import java.util.ArrayList; -import java.util.Base64; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.Optional; - -import javax.ws.rs.core.MediaType; -import javax.xml.stream.XMLStreamException; - -/** - * Utility methods used in gateway. - */ -public class GatewayUtils { - private static final Log log = LogFactory.getLog(GatewayUtils.class); - - private static final String SOAP_ENV_START_TAG = ""; - private static final String SOAP_ENV_END_TAG = ""; - - /** - * Method to decode the base64 encoded JSON payload. - * - * @param payload base64 encoded payload - * @return Decoded JSON Object - * @throws UnsupportedEncodingException When encoding is not UTF-8 - */ - public static JSONObject decodeBase64(String payload) throws UnsupportedEncodingException { - - return new JSONObject(new String(Base64.getDecoder().decode(payload), - String.valueOf(StandardCharsets.UTF_8))); - } - - /** - * Method to extract JWT payload section as a string. - * - * @param jwtString full JWT - * @return Payload section of JWT - */ - public static String getPayloadFromJWT(String jwtString) { - - return jwtString.split("\\.")[1]; - } - - /** - * Method to retrieve payload as a String from XML. - * This method expects 'xml-multiple' instruction in xmlPayload to transform single element json arrays. - * synapse.commons.json.output.xmloutMultiplePI=true should be set in synapse.properties file. - * - * This method is deprecated. Use getXMLPayloadToSign method instead. - * - * @param xmlPayload Payload in XML format - * @return String version of JSON object - */ - @Deprecated - public static String getPayloadFromXML(String xmlPayload) throws OpenBankingException { - - String jsonString = null; - try { - OMElement omElement = AXIOMUtil.stringToOM(xmlPayload); - jsonString = JsonUtil.toJsonString(omElement).toString(); - } catch (AxisFault e) { - log.error("Error occurred while reading the xml payload"); - throw new OpenBankingException("Error occurred while reading the xml payload", e); - } catch (XMLStreamException e) { - log.error("Error occurred while transforming the xml payload to json"); - throw new OpenBankingException("Error occurred while transforming the xml payload to json", e); - } - JSONObject jsonObject = new JSONObject(jsonString); - return jsonObject.has("Body") ? jsonObject.get("Body").toString() : null; - } - - /** - * Method to retrieve payload from response with xml Payload. - * - * @param xmlPayload Payload in XML format - * @return String payload - */ - public static String getXMLPayloadToSign(String xmlPayload) throws OpenBankingException { - - try { - OMElement omElement = AXIOMUtil.stringToOM(xmlPayload); - OMElement firstElement = (OMElement) omElement.getFirstOMChild(); - if (firstElement != null) { - return firstElement.toString(); - } else { - return ""; - } - } catch (XMLStreamException e) { - log.error("Error occurred while transforming the xml payload."); - throw new OpenBankingException("Error occurred while transforming the xml payload", e); - } - } - - public static String getTextPayload(String payload) { - - return XML.toJSONObject(payload).getJSONObject("soapenv:Body").getJSONObject("text").getString("content"); - - } - - /** - * Method to obatain basic auth header. - * - * @param username Username of Auth header - * @param password Password of Auth header - * @return basic auth header - */ - public static String getBasicAuthHeader(String username, String password) { - - byte[] authHeader = Base64.getEncoder().encode((username + ":" + password).getBytes(StandardCharsets.UTF_8)); - return GatewayConstants.BASIC_TAG + new String(authHeader, StandardCharsets.UTF_8); - } - - /** - * Method to obtain swagger definition from publisher API. - * - * @param apiId ID of the API - * @return String of swagger definition - */ - @Generated(message = "Cannot test without running APIM. Integration test will be written for this") - public static String getSwaggerDefinition(String apiId) { - - String publisherHostName = - GatewayDataHolder.getInstance().getOpenBankingConfigurationService() - .getConfigurations() - .get(GatewayConstants.PUBLISHER_HOSTNAME).toString(); - - String publisherAPIURL = publisherHostName.endsWith("/") ? - publisherHostName + GatewayConstants.PUBLISHER_API_PATH + apiId + GatewayConstants.SWAGGER_ENDPOINT : - publisherHostName + "/" + GatewayConstants.PUBLISHER_API_PATH + apiId + - GatewayConstants.SWAGGER_ENDPOINT; - - HttpGet httpGet = new HttpGet(publisherAPIURL); - String userName = getAPIMgtConfig(GatewayConstants.API_KEY_VALIDATOR_USERNAME); - String password = getAPIMgtConfig(GatewayConstants.API_KEY_VALIDATOR_PASSWORD); - - httpGet.setHeader(GatewayConstants.AUTH_HEADER, GatewayUtils.getBasicAuthHeader(userName, password)); - HttpResponse response = null; - try { - response = GatewayDataHolder.getHttpClient().execute(httpGet); - InputStream in = response.getEntity().getContent(); - return IOUtils.toString(in, String.valueOf(StandardCharsets.UTF_8)); - } catch (IOException | OpenBankingException e) { - throw new OpenBankingRuntimeException("Failed to retrieve swagger definition from API", e); - } - } - - /** - * Method to read API mgt configs when key is given. - * - * @param key config key - * @return config value - */ - public static String getAPIMgtConfig(String key) { - - return GatewayDataHolder.getInstance() - .getApiManagerConfigurationService().getAPIManagerConfiguration().getFirstProperty(key); - } - - public static boolean isValidJWTToken(String jwtString) { - - String[] jwtPart = jwtString.split("\\."); - if (jwtPart.length != 3) { - return false; - } - try { - decodeBase64(jwtPart[0]); - decodeBase64(jwtPart[1]); - } catch (UnsupportedEncodingException | JSONException | IllegalArgumentException e) { - return false; - } - return true; - } - - /** - * Check the content type and http method of the request. - * - * @param contentType - contentType - * @param httpMethod - httpMethod - * @return - */ - public static boolean isEligibleRequest(String contentType, String httpMethod) { - - return (contentType.startsWith(GatewayConstants.JSON_CONTENT_TYPE) || - contentType.startsWith(GatewayConstants.APPLICATION_XML_CONTENT_TYPE) || - contentType.startsWith(GatewayConstants.TEXT_XML_CONTENT_TYPE)) && - (GatewayConstants.POST_HTTP_METHOD.equals(httpMethod) || GatewayConstants.PUT_HTTP_METHOD - .equals(httpMethod)); - } - - /** - * Check the content type and http method of the response. - * - * @param contentType - contentType - * @param httpMethod - httpMethod - * @return - */ - public static boolean isEligibleResponse(String contentType, String httpMethod) { - - return (contentType.startsWith(GatewayConstants.JSON_CONTENT_TYPE) || - contentType.startsWith(GatewayConstants.APPLICATION_XML_CONTENT_TYPE) || - contentType.startsWith(GatewayConstants.TEXT_XML_CONTENT_TYPE)) && - (GatewayConstants.GET_HTTP_METHOD.equals(httpMethod) || GatewayConstants. - POST_HTTP_METHOD.equals(httpMethod) || GatewayConstants.PUT_HTTP_METHOD.equals(httpMethod) - || GatewayConstants.PATCH_HTTP_METHOD.equals(httpMethod) || GatewayConstants. - DELETE_HTTP_METHOD.equals(httpMethod)); - } - - /** - * Method to extract request payload from OBAPIRequestContext. - * - * @param obapiRequestContext - * @param requestHeaders - * @return - */ - public static Optional extractRequestPayload(OBAPIRequestContext obapiRequestContext, - Map requestHeaders) - throws OpenBankingException { - - Optional payloadString = Optional.empty(); - - if (requestHeaders.containsKey(GatewayConstants.CONTENT_TYPE_TAG)) { - if (requestHeaders.get(GatewayConstants.CONTENT_TYPE_TAG).contains( - GatewayConstants.TEXT_XML_CONTENT_TYPE) - || requestHeaders.get(GatewayConstants.CONTENT_TYPE_TAG).contains( - GatewayConstants.APPLICATION_XML_CONTENT_TYPE)) { - try { - payloadString = Optional.of(GatewayUtils.getXMLPayloadToSign( - obapiRequestContext.getMsgInfo().getPayloadHandler().consumeAsString())); - } catch (Exception e) { - throw new OpenBankingException("Internal Server Error, Unable to process Payload"); - } - } else { - payloadString = Optional.ofNullable(obapiRequestContext.getRequestPayload()); - } - } else { - payloadString = Optional.ofNullable(obapiRequestContext.getRequestPayload()); - } - - return payloadString; - } - - /** - * Method to extract response payload from OBAPIResponseContext. - * - * @param obapiResponseContext - * @param responseHeaders - * @return - */ - public static Optional extractResponsePayload(OBAPIResponseContext obapiResponseContext, - Map responseHeaders) - throws OpenBankingException { - - Optional payloadString = Optional.empty(); - - if (responseHeaders.containsKey(GatewayConstants.CONTENT_TYPE_TAG)) { - if (responseHeaders.get(GatewayConstants.CONTENT_TYPE_TAG).contains( - GatewayConstants.TEXT_XML_CONTENT_TYPE) - || responseHeaders.get(GatewayConstants.CONTENT_TYPE_TAG).contains( - GatewayConstants.APPLICATION_XML_CONTENT_TYPE)) { - try { - payloadString = Optional.of(GatewayUtils.getXMLPayloadToSign( - obapiResponseContext.getMsgInfo().getPayloadHandler().consumeAsString())); - } catch (Exception e) { - throw new OpenBankingException("Internal Server Error, Unable to process Payload"); - } - } else { - payloadString = Optional.ofNullable(obapiResponseContext.getResponsePayload()); - } - } else { - payloadString = Optional.ofNullable(obapiResponseContext.getResponsePayload()); - } - - return payloadString; - } - - /** - * Returns the JWS with a detached payload. - * @param payloadString response payload - * @param criticalParameters Critical parameters - * @return String Detached Jws Signature - * @throws JOSEException throws JOSEException Exception - * @throws OpenBankingExecutorException throws OpenBanking Executor Exception - */ - @Generated(message = "Excluding from unit tests since it is covered by other methods") - public static String constructJWSSignature(String payloadString, HashMap criticalParameters) - throws OpenBankingExecutorException, JOSEException { - - String detachedJWS; - - Optional signingKey; - - // Get from config parser - JWSAlgorithm algorithm = GatewaySignatureHandlingUtils.getSigningAlgorithm(); - - // Get signing certificate of ASPSP from keystore - signingKey = GatewaySignatureHandlingUtils.getSigningKey(); - - if (signingKey.isPresent()) { - // Create a new JWSSigner - JWSSigner signer; - Key privateKey = signingKey.get(); - - // Retrieve kid or empty string for signingKeyId - String signingKeyId = GatewaySignatureHandlingUtils.getSigningKeyId(); - - if (StringUtils.isBlank(signingKeyId)) { - throw new OpenBankingExecutorException("The kid is not present to sign."); - } - - JWSHeader jwsHeader = GatewaySignatureHandlingUtils.constructJWSHeader(signingKeyId, - criticalParameters, algorithm); - JWSObject jwsObject = GatewaySignatureHandlingUtils.constructJWSObject(jwsHeader, - payloadString); - - if (privateKey.getAlgorithm().equals("RSA")) { - // If the signing key is an RSA Key - signer = new RSASSASigner((PrivateKey) privateKey); - } else if (privateKey.getAlgorithm().equals("EC")) { - // If the signing key is an EC Key - signer = new ECDSASigner((ECPrivateKey) privateKey); - } else { - throw new JOSEException("The \"" + privateKey.getAlgorithm() + - "\" algorithm is not supported by the Solution"); - } - - try { - // Check if payload is b64 encoded or un-encoded - if (GatewaySignatureHandlingUtils.isB64HeaderVerifiable(jwsObject)) { - // b64=true - jwsObject.sign(signer); - String serializedJws = jwsObject.serialize(); - detachedJWS = GatewaySignatureHandlingUtils.createDetachedJws(serializedJws); - } else { - // b64=false - // Produces the signature with un-encoded payload. - // which is the encoded header + ".." + the encoded signature - Base64URL signature = signer.sign(jwsHeader, - GatewaySignatureHandlingUtils.getSigningInput(jwsHeader, payloadString)); - detachedJWS = GatewaySignatureHandlingUtils.createDetachedJws(jwsHeader, signature); - } - } catch (JOSEException | UnsupportedEncodingException e) { - throw new OpenBankingExecutorException("Unable to compute JWS signature", e); - } - return detachedJWS; - } else { - throw new OpenBankingExecutorException("Signing key is not present"); - } - } - - /** - * Method to handle internal server errors in JWS Signature validation. - * - * @param obapiRequestContext OB response context object - * @param message error message - */ - @Generated(message = "Excluding from unit tests since the method is for exception" + - "handling") - public static void handleRequestInternalServerError( - OBAPIRequestContext obapiRequestContext, String message, String errorCode) { - - OpenBankingExecutorError error = new OpenBankingExecutorError(errorCode, - "Internal server error", message, OpenBankingErrorCodes.SERVER_ERROR_CODE); - ArrayList executorErrors = obapiRequestContext.getErrors(); - executorErrors.add(error); - obapiRequestContext.setError(true); - obapiRequestContext.setErrors(executorErrors); - } - - /** - * Method to handle internal server errors in JWS Signature validation. - * - * @param obapiResponseContext OB response context object - * @param message error message - */ - @Generated(message = "Excluding from unit tests since the method is for exception" + - "handling") - public static void handleResponseInternalServerError( - OBAPIResponseContext obapiResponseContext, String message, String errorCode) { - - OpenBankingExecutorError error = new OpenBankingExecutorError(errorCode, - "Internal server error", message, OpenBankingErrorCodes.SERVER_ERROR_CODE); - ArrayList executorErrors = obapiResponseContext.getErrors(); - executorErrors.add(error); - obapiResponseContext.setError(true); - obapiResponseContext.setErrors(executorErrors); - } - - /** - * Method to get the userName with tenant domain. - * - * @param userName username - * @return username with tenant domain - */ - public static String getUserNameWithTenantDomain(String userName) { - - String tenantDomain = PrivilegedCarbonContext.getThreadLocalCarbonContext().getTenantDomain(); - if (userName.endsWith(tenantDomain)) { - return userName; - } else { - return userName + "@" + tenantDomain; - } - } - - /** - * Retrieve security definitions defined in the swagger. - * This method will return cached values if present, else will read swagger and cache the result. - * - * @param obApiRequestContext ob api request context - * @return list of allowed auth flows for the elected resource - */ - @Generated(message = "Ignoring since the method has covered in other tests") - public static List getAllowedOAuthFlows(OBAPIRequestContext obApiRequestContext) { - - List oauthFlows = new ArrayList<>(); - String httpMethod = obApiRequestContext.getMsgInfo().getHttpMethod(); - String cacheKey = obApiRequestContext.getMsgInfo().getElectedResource() + ":" + httpMethod; - GatewayCacheKey apiSecurityCacheKey = GatewayCacheKey.of(cacheKey); - - try { - oauthFlows = - (List) GatewayDataHolder.getGatewayCache().getFromCacheOrRetrieve(apiSecurityCacheKey, - () -> { - OpenAPI openAPI = obApiRequestContext.getOpenAPI(); - String electedResource = obApiRequestContext.getMsgInfo().getElectedResource(); - return getAllowedOAuthFlowsFromSwagger(openAPI, electedResource, httpMethod); - }); - } catch (OpenBankingException e) { - log.error("Unable to cache or retrieve from API Security Cache", e); - } - return oauthFlows; - } - - /** - * Read allowed security schemes defined in the swagger for the given resource. - * - * @param openAPI open API object - * @param electedResource elected resource - * @param httpMethod http method - * @return allowed security scheme - */ - public static List getAllowedOAuthFlowsFromSwagger(OpenAPI openAPI, String electedResource, - String httpMethod) { - - Map securitySchemes = openAPI.getComponents().getSecuritySchemes(); - HashMap> oAuthFlows = new HashMap<>(); - - for (Object scheme : securitySchemes.keySet()) { - OAuthFlows flows = securitySchemes.get(scheme.toString()).getFlows(); - - if (flows != null) { - ArrayList allowedFlowsPerScheme = new ArrayList<>(); - - if (flows.getAuthorizationCode() != null) { - allowedFlowsPerScheme.add(GatewayConstants.AUTHORIZATION_CODE); - } - if (flows.getImplicit() != null) { - allowedFlowsPerScheme.add(GatewayConstants.IMPLICIT); - } - if (flows.getClientCredentials() != null) { - allowedFlowsPerScheme.add(GatewayConstants.CLIENT_CREDENTIALS); - } - if (flows.getPassword() != null) { - allowedFlowsPerScheme.add(GatewayConstants.PASSWORD_GRANT); - } - oAuthFlows.put(scheme.toString(), allowedFlowsPerScheme); - } - } - - // get security flows defined for the resource - PathItem electedPath = openAPI.getPaths().get(electedResource); - List resourceSecurity = null; - if (GatewayConstants.GET.equalsIgnoreCase(httpMethod)) { - resourceSecurity = electedPath.getGet().getSecurity(); - } else if (GatewayConstants.POST.equalsIgnoreCase(httpMethod)) { - resourceSecurity = electedPath.getPost().getSecurity(); - } else if (GatewayConstants.PUT.equalsIgnoreCase(httpMethod)) { - resourceSecurity = electedPath.getPut().getSecurity(); - } else if (GatewayConstants.PATCH.equalsIgnoreCase(httpMethod)) { - resourceSecurity = electedPath.getPatch().getSecurity(); - } else if (GatewayConstants.DELETE.equalsIgnoreCase(httpMethod)) { - resourceSecurity = electedPath.getDelete().getSecurity(); - } - - ArrayList allowedFlows = new ArrayList<>(); - List securityRequirementList = new ArrayList<>(); - if (resourceSecurity != null) { - for (Object security : resourceSecurity) { - // Adding the keys of each security requirement to a list - securityRequirementList.addAll(new ArrayList<>(((SecurityRequirement) security).keySet())); - } - } - - for (String requirement : securityRequirementList) { - if (GatewayConstants.DEFAULT.equalsIgnoreCase(requirement) || - GatewayConstants.OPENID.equalsIgnoreCase(requirement)) { - continue; - } - if (oAuthFlows.containsKey(requirement)) { - allowedFlows.addAll(oAuthFlows.get(requirement)); - } - } - - return allowedFlows; - } - - - /** - * Get bearer token payload. - * - * @param transportHeaders transport headers - * @return jwt token payload - * @throws OpenBankingExecutorException when authorization header not found in transport headers. - */ - public static String getBearerTokenPayload(Map transportHeaders) - throws OpenBankingExecutorException { - - String authorizationHeader = "Authorization"; - if (transportHeaders.containsKey(authorizationHeader)) { - try { - return transportHeaders.get(authorizationHeader).split(" ")[1].split("\\.")[1]; - } catch (ArrayIndexOutOfBoundsException e) { - log.debug("Invalid authorization header format", e); - throw new OpenBankingExecutorException("Invalid Credentials.", - String.valueOf(GatewayConstants.API_AUTH_INVALID_CREDENTIALS), - GatewayConstants.INVALID_CREDENTIALS); - } - } else { - log.debug("Missing Authorization header"); - throw new OpenBankingExecutorException("Missing Credentials.", - String.valueOf(GatewayConstants.API_AUTH_MISSING_CREDENTIALS), - GatewayConstants.MISSING_CREDENTIALS); - } - } - - /** - * Get type of the token (application/ application user). - * - * @param tokenPayload jwt token payload - * @return token type - */ - public static String getTokenType(String tokenPayload) throws OpenBankingExecutorException { - - try { - JSONObject decodedPayload = new JSONObject(new String(Base64.getUrlDecoder().decode(tokenPayload), - StandardCharsets.UTF_8)); - return decodedPayload.getString(GatewayConstants.AUTHORIZED_USER_TYPE_CLAIM_NAME); - } catch (RuntimeException e) { - log.error("Invalid tokenPayload", e); - throw new OpenBankingExecutorException("Invalid Credentials.", - String.valueOf(GatewayConstants.API_AUTH_INVALID_CREDENTIALS), - GatewayConstants.INVALID_CREDENTIALS); - } - } - - /** - * Validate the grant type against the swagger allowed auth flows. - * - * @param tokenType grant type in the token - * @param allowedOAuthFlows oauth flows allowed by swagger - * @throws OpenBankingExecutorException when incorrect grant type is provided - */ - public static void validateGrantType(String tokenType, List allowedOAuthFlows) - throws OpenBankingExecutorException { - - if ((GatewayConstants.APPLICATION.equalsIgnoreCase(tokenType) && - allowedOAuthFlows.contains(GatewayConstants.CLIENT_CREDENTIALS)) || - (GatewayConstants.APPLICATION_USER.equalsIgnoreCase(tokenType) && - allowedOAuthFlows.contains(GatewayConstants.AUTHORIZATION_CODE))) { - log.debug("Valid Access Token type found"); - } else { - log.error("Incorrect Access Token Type is provided"); - throw new OpenBankingExecutorException(GatewayConstants.INVALID_GRANT_TYPE, - OpenBankingErrorCodes.INVALID_GRANT_TYPE_CODE, "Incorrect Access Token Type provided"); - } - } - - /** - * Build Message and extract payload. - * - * @param axis2MC message context - * @return optional json message - * @throws OpenBankingException thrown if unable to build - */ - public static Optional buildMessagePayloadFromMessageContext( - org.apache.axis2.context.MessageContext axis2MC, Map headers) throws OpenBankingException { - - String requestPayload = null; - boolean isMessageContextBuilt = isMessageContextBuilt(axis2MC); - if (!isMessageContextBuilt) { - // Build Axis2 Message. - try { - RelayUtils.buildMessage(axis2MC); - } catch (IOException | XMLStreamException e) { - throw new OpenBankingException("Unable to build axis2 message", e); - } - } - - if (headers.containsKey(GatewayConstants.CONTENT_TYPE_TAG)) { - if (headers.get(GatewayConstants.CONTENT_TYPE_TAG).toString().contains( - GatewayConstants.TEXT_XML_CONTENT_TYPE) - || headers.get(GatewayConstants.CONTENT_TYPE_TAG).toString().contains( - GatewayConstants.APPLICATION_XML_CONTENT_TYPE) - || headers.get(GatewayConstants.CONTENT_TYPE_TAG).toString().contains( - GatewayConstants.JWT_CONTENT_TYPE)) { - - OMElement payload = axis2MC.getEnvelope().getBody().getFirstElement(); - if (payload != null) { - requestPayload = payload.toString(); - } else { - requestPayload = ""; - } - } else { - // Get JSON Stream and cast to string - try { - InputStream jsonPayload = JsonUtil.getJsonPayload(axis2MC); - if (jsonPayload != null) { - requestPayload = IOUtils.toString(JsonUtil.getJsonPayload(axis2MC), - StandardCharsets.UTF_8.name()); - } - - } catch (IOException e) { - throw new OpenBankingException("Unable to read payload stream", e); - } - } - } - return Optional.ofNullable(requestPayload); - } - - /** - * Util method to check whether the message context is already built. - * - * @param axis2MC axis2 message context - * @return true if message context is already built - */ - public static boolean isMessageContextBuilt(org.apache.axis2.context.MessageContext axis2MC) { - - boolean isMessageContextBuilt = false; - Object messageContextBuilt = axis2MC.getProperty(PassThroughConstants.MESSAGE_BUILDER_INVOKED); - if (messageContextBuilt != null) { - isMessageContextBuilt = (Boolean) messageContextBuilt; - } - return isMessageContextBuilt; - } - - /** - * Return JSON ResponseError for SynapseHandler. - * - * @param messageContext messages context. - * @param code response code. - * @param jsonPayload json payload. - */ - public static void returnSynapseHandlerJSONError(MessageContext messageContext, String code, String jsonPayload) { - - org.apache.axis2.context.MessageContext axis2MC = ((Axis2MessageContext) messageContext). - getAxis2MessageContext(); - axis2MC.setProperty(PassThroughConstants.MESSAGE_BUILDER_INVOKED, Boolean.TRUE); - try { - RelayUtils.discardRequestMessage(axis2MC); - } catch (AxisFault axisFault) { - log.error("ResponseError occurred while discarding the message", axisFault); - } - setJsonFaultPayloadToMessageContext(messageContext, jsonPayload); - sendSynapseHandlerFaultResponse(messageContext, code); - } - - /** - * Setting JSON payload as fault message to messageContext. - * @param messageContext messages context. - * @param payload json payload. - */ - private static void setJsonFaultPayloadToMessageContext(MessageContext messageContext, String payload) { - - org.apache.axis2.context.MessageContext axis2MessageContext = ((Axis2MessageContext) messageContext) - .getAxis2MessageContext(); - - axis2MessageContext.setProperty(Constants.Configuration.MESSAGE_TYPE, MediaType.APPLICATION_JSON); - - try { - JsonUtil.getNewJsonPayload(axis2MessageContext, payload, true, true); - } catch (AxisFault axisFault) { - log.error("Unable to set JSON payload to fault message", axisFault); - } - } - - /** - * Send synapseHandler fault response. - * @param messageContext messages context. - * @param status error code. - */ - private static void sendSynapseHandlerFaultResponse(MessageContext messageContext, String status) { - - org.apache.axis2.context.MessageContext axis2MC = ((Axis2MessageContext) messageContext). - getAxis2MessageContext(); - - axis2MC.setProperty(NhttpConstants.HTTP_SC, status); - messageContext.setResponse(true); - messageContext.setProperty("RESPONSE", "true"); - messageContext.setTo(null); - axis2MC.removeProperty(Constants.Configuration.CONTENT_TYPE); - Axis2Sender.sendBack(messageContext); - } - - /** - * Method to get json error body in OAuth2 format. - * @return json error body - */ - public static String getOAuth2JsonErrorBody(String error, String errorDescription) { - - JSONObject errorJSON = new JSONObject(); - errorJSON.put("error", error); - errorJSON.put("error_description", errorDescription); - return errorJSON.toString(); - } - - /** - * Convert X509Certificate to PEM encoded string. - * - * @param certificate X509Certificate - * @return PEM encoded string - */ - public static String getPEMEncodedCertificateString(X509Certificate certificate) - throws CertificateEncodingException { - - StringBuilder certificateBuilder = new StringBuilder(); - Base64.Encoder encoder = Base64.getEncoder(); - byte[] encoded = certificate.getEncoded(); - String base64Encoded = encoder.encodeToString(encoded); - - certificateBuilder.append(GatewayConstants.BEGIN_CERT); - certificateBuilder.append(base64Encoded); - certificateBuilder.append(GatewayConstants.END_CERT); - - return certificateBuilder.toString().replaceAll("\n", "+"); - } - - /** - * Extract Certificate from Message Context. - * - * @param ctx Message Context - * @return X509Certificate - */ - public static X509Certificate extractAuthCertificateFromMessageContext( - org.apache.axis2.context.MessageContext ctx) { - - Object sslCertObject = ctx.getProperty(GatewayConstants.AXIS2_MTLS_CERT_PROPERTY); - if (sslCertObject != null) { - X509Certificate[] certs = (X509Certificate[]) sslCertObject; - return certs[0]; - } else { - return null; - } - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/util/IdempotencyConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/util/IdempotencyConstants.java deleted file mode 100644 index b8991259..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/java/com/wso2/openbanking/accelerator/gateway/util/IdempotencyConstants.java +++ /dev/null @@ -1,60 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.util; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; - -/** - * IdempotencyConstants. - */ -public class IdempotencyConstants { - - // config parser keys - public static final String IDEMPOTENCY_ALLOWED_TIME = "Gateway.Idempotency.AllowedTimeDuration"; - public static final String IDEMPOTENCY_CACHE_TIME_TO_LIVE = - "Gateway.Cache.IdempotencyValidationCache.CacheTimeToLive"; - public static final String IDEMPOTENCY_KEY_HEADER = "Gateway.Idempotency.IdempotencyKeyHeader"; - public static final String IDEMPOTENCY_IS_ENABLED = "Gateway.Idempotency.IsEnabled"; - - public static final String HTTP_STATUS = "httpStatus"; - public static final String PAYLOAD = "payload"; - - - /** - * Error. - */ - public static class Error { - - public static final String DATE_MISSING = "Date header is missing in the request"; - public static final String EXECUTOR_IDEMPOTENCY_KEY_ERROR = - "Error while handling Idempotency check.:Header." + getPathIdemKey();; - public static final String EXECUTOR_IDEMPOTENCY_KEY_FRAUDULENT = - "Idempotency check failed.:Header." + getPathIdemKey(); - public static final String HEADER_INVALID = "Header Invalid"; - public static final String IDEMPOTENCY_HANDLE_ERROR = - "Error occurred while handling the idempotency available request"; - - private static String getPathIdemKey() { - - return (String) OpenBankingConfigParser.getInstance().getConfiguration() - .get(IdempotencyConstants.IDEMPOTENCY_KEY_HEADER); - } - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/resources/findbugs-exclude.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/resources/findbugs-exclude.xml deleted file mode 100644 index fd470b2c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/resources/findbugs-exclude.xml +++ /dev/null @@ -1,40 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/resources/findbugs-include.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/resources/findbugs-include.xml deleted file mode 100644 index 649d044e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,23 +0,0 @@ - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/core/DefaultRequestRouterTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/core/DefaultRequestRouterTest.java deleted file mode 100644 index 10892914..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/core/DefaultRequestRouterTest.java +++ /dev/null @@ -1,119 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.core; - -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.executor.test.util.TestUtil; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import io.swagger.v3.oas.models.OpenAPI; -import org.mockito.Mockito; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; -import org.wso2.carbon.apimgt.common.gateway.dto.MsgInfoDTO; - -import java.util.HashMap; -import java.util.Map; - -/** - * Test for default request router. - */ -public class DefaultRequestRouterTest { - - DefaultRequestRouter defaultRequestRouter; - OpenAPI openAPI; - - @BeforeClass - public void beforeClass() { - - defaultRequestRouter = new DefaultRequestRouter(); - defaultRequestRouter.setExecutorMap(TestUtil.initExecutors()); - openAPI = new OpenAPI(); - openAPI.setExtensions(new HashMap<>()); - } - - @Test(priority = 1) - public void testDCRRequestsForRouter() { - - OBAPIRequestContext obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - OBAPIResponseContext obapiResponseContext = Mockito.mock(OBAPIResponseContext.class); - MsgInfoDTO msgInfoDTO1 = new MsgInfoDTO(); - msgInfoDTO1.setResource("/anyAPIcall/register"); - Mockito.when(obapiRequestContext.getMsgInfo()).thenReturn(msgInfoDTO1); - Mockito.when(obapiRequestContext.getOpenAPI()).thenReturn(openAPI); - Mockito.when(obapiResponseContext.getMsgInfo()).thenReturn(msgInfoDTO1); - Assert.assertNotNull(defaultRequestRouter.getExecutorsForRequest(obapiRequestContext)); - Assert.assertNotNull(defaultRequestRouter.getExecutorsForResponse(obapiResponseContext)); - - } - - @Test(priority = 1) - public void testAccountRequestsForRouter() { - - OBAPIRequestContext obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - OBAPIResponseContext obapiResponseContext = Mockito.mock(OBAPIResponseContext.class); - MsgInfoDTO msgInfoDTO2 = new MsgInfoDTO(); - msgInfoDTO2.setResource("/anyAPIcall"); - Mockito.when(obapiRequestContext.getOpenAPI()).thenReturn(openAPI); - Mockito.when(obapiRequestContext.getMsgInfo()).thenReturn(msgInfoDTO2); - Mockito.when(obapiResponseContext.getMsgInfo()).thenReturn(msgInfoDTO2); - Assert.assertNotNull(defaultRequestRouter.getExecutorsForRequest(obapiRequestContext)); - Assert.assertNotNull(defaultRequestRouter.getExecutorsForResponse(obapiResponseContext)); - } - - @Test(priority = 2) - public void testNonRegulatoryAPIcall() { - - OBAPIRequestContext obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - OBAPIResponseContext obapiResponseContext = Mockito.mock(OBAPIResponseContext.class); - MsgInfoDTO msgInfoDTO2 = new MsgInfoDTO(); - msgInfoDTO2.setResource("/anyAPIcall"); - Map extensions = new HashMap<>(); - Map contextProps = new HashMap<>(); - extensions.put(GatewayConstants.API_TYPE_CUSTOM_PROP, GatewayConstants.API_TYPE_NON_REGULATORY); - contextProps.put(GatewayConstants.API_TYPE_CUSTOM_PROP, GatewayConstants.API_TYPE_NON_REGULATORY); - openAPI.setExtensions(extensions); - Mockito.when(obapiRequestContext.getOpenAPI()).thenReturn(openAPI); - Mockito.when(obapiRequestContext.getMsgInfo()).thenReturn(msgInfoDTO2); - Mockito.when(obapiResponseContext.getMsgInfo()).thenReturn(msgInfoDTO2); - Mockito.when(obapiResponseContext.getContextProps()).thenReturn(contextProps); - Assert.assertEquals(defaultRequestRouter.getExecutorsForRequest(obapiRequestContext).size(), 0); - Assert.assertEquals(defaultRequestRouter.getExecutorsForResponse(obapiResponseContext).size(), 0); - } - - @Test(priority = 2) - public void testRegulatoryAPIcall() { - - OBAPIRequestContext obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - OBAPIResponseContext obapiResponseContext = Mockito.mock(OBAPIResponseContext.class); - MsgInfoDTO msgInfoDTO2 = new MsgInfoDTO(); - msgInfoDTO2.setResource("/anyAPIcall"); - Map extensions = new HashMap<>(); - Map contextProps = new HashMap<>(); - extensions.put(GatewayConstants.API_TYPE_CUSTOM_PROP, GatewayConstants.API_TYPE_NON_REGULATORY); - contextProps.put(GatewayConstants.API_TYPE_CUSTOM_PROP, "regulatory"); - openAPI.setExtensions(extensions); - Mockito.when(obapiRequestContext.getOpenAPI()).thenReturn(openAPI); - Mockito.when(obapiRequestContext.getMsgInfo()).thenReturn(msgInfoDTO2); - Mockito.when(obapiResponseContext.getMsgInfo()).thenReturn(msgInfoDTO2); - Mockito.when(obapiResponseContext.getContextProps()).thenReturn(contextProps); - Assert.assertNotNull(defaultRequestRouter.getExecutorsForRequest(obapiRequestContext)); - Assert.assertNotNull(defaultRequestRouter.getExecutorsForResponse(obapiResponseContext)); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/core/TestOBExtensionImpl.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/core/TestOBExtensionImpl.java deleted file mode 100644 index a59b5579..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/core/TestOBExtensionImpl.java +++ /dev/null @@ -1,219 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.core; - -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.executor.test.TestConstants; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import org.apache.http.HttpStatus; -import org.json.JSONObject; -import org.mockito.Mockito; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; -import org.wso2.carbon.apimgt.common.gateway.dto.ExtensionResponseDTO; -import org.wso2.carbon.apimgt.common.gateway.dto.ExtensionResponseStatus; -import org.wso2.carbon.apimgt.common.gateway.dto.MsgInfoDTO; - -import java.util.HashMap; -import java.util.Map; - -/** - * Test for open Banking extension implementation. - */ -public class TestOBExtensionImpl { - - private static OBAPIRequestContext obapiRequestContext; - private static OBAPIResponseContext obapiResponseContext; - private static OBExtensionListenerImpl obExtensionListener = new OBExtensionListenerImpl(); - - @BeforeClass - public static void beforeClass() { - - } - - @Test(priority = 1) - public void testMinimumFlow() { - - obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - Mockito.when(obapiRequestContext.isError()).thenReturn(false); - Mockito.when(obapiRequestContext.getModifiedPayload()).thenReturn(null); - Mockito.when(obapiRequestContext.getAddedHeaders()).thenReturn(new HashMap<>()); - - obapiResponseContext = Mockito.mock(OBAPIResponseContext.class); - Mockito.when(obapiResponseContext.isError()).thenReturn(false); - Mockito.when(obapiResponseContext.getModifiedPayload()).thenReturn(null); - Mockito.when(obapiResponseContext.getAddedHeaders()).thenReturn(new HashMap<>()); - - ExtensionResponseDTO responseDTOForRequest = obExtensionListener.getResponseDTOForRequest(obapiRequestContext); - Assert.assertEquals(responseDTOForRequest.getResponseStatus(), ExtensionResponseStatus.CONTINUE.toString()); - Assert.assertNull(responseDTOForRequest.getHeaders()); - Assert.assertNull(responseDTOForRequest.getPayload()); - Assert.assertNull(responseDTOForRequest.getCustomProperty()); - Assert.assertEquals(responseDTOForRequest.getStatusCode(), 0); - - ExtensionResponseDTO responseDTOForResponse = - obExtensionListener.getResponseDTOForResponse(obapiResponseContext); - - Assert.assertEquals(responseDTOForResponse.getResponseStatus(), ExtensionResponseStatus.CONTINUE.toString()); - Assert.assertNull(responseDTOForResponse.getHeaders()); - Assert.assertNull(responseDTOForResponse.getPayload()); - Assert.assertNull(responseDTOForResponse.getCustomProperty()); - Assert.assertEquals(responseDTOForResponse.getStatusCode(), 0); - - } - - @Test(priority = 1) - public void testAddedHeaders() { - - obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - obapiResponseContext = Mockito.mock(OBAPIResponseContext.class); - Mockito.when(obapiRequestContext.isError()).thenReturn(false); - Mockito.when(obapiRequestContext.getModifiedPayload()).thenReturn(null); - MsgInfoDTO msgInfoDTO = new MsgInfoDTO(); - msgInfoDTO.setHeaders(new HashMap<>()); - Mockito.when(obapiRequestContext.getMsgInfo()).thenReturn(msgInfoDTO); - Map addedHeaders = new HashMap<>(); - addedHeaders.put("custom", "header"); - Mockito.when(obapiRequestContext.getAddedHeaders()).thenReturn(addedHeaders); - - ExtensionResponseDTO responseDTOForRequest = obExtensionListener.getResponseDTOForRequest(obapiRequestContext); - Assert.assertEquals(responseDTOForRequest.getResponseStatus(), ExtensionResponseStatus.CONTINUE.toString()); - Assert.assertEquals(responseDTOForRequest.getHeaders().get("custom"), "header"); - Assert.assertNull(responseDTOForRequest.getPayload()); - Assert.assertNull(responseDTOForRequest.getCustomProperty()); - Assert.assertEquals(responseDTOForRequest.getStatusCode(), 0); - - Mockito.when(obapiResponseContext.isError()).thenReturn(false); - Mockito.when(obapiResponseContext.getModifiedPayload()).thenReturn(null); - Mockito.when(obapiResponseContext.getMsgInfo()).thenReturn(msgInfoDTO); - Mockito.when(obapiResponseContext.getAddedHeaders()).thenReturn(addedHeaders); - - ExtensionResponseDTO responseDTOForResponse = - obExtensionListener.getResponseDTOForResponse(obapiResponseContext); - Assert.assertEquals(responseDTOForResponse.getResponseStatus(), ExtensionResponseStatus.CONTINUE.toString()); - Assert.assertEquals(responseDTOForResponse.getHeaders().get("custom"), "header"); - Assert.assertNull(responseDTOForResponse.getPayload()); - Assert.assertNull(responseDTOForResponse.getCustomProperty()); - Assert.assertEquals(responseDTOForResponse.getStatusCode(), 0); - - } - - @Test(priority = 1) - public void testModifiedPayload() { - - obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - obapiResponseContext = Mockito.mock(OBAPIResponseContext.class); - Mockito.when(obapiRequestContext.isError()).thenReturn(false); - Mockito.when(obapiRequestContext.getModifiedPayload()).thenReturn(TestConstants.CUSTOM_PAYLOAD); - Mockito.when(obapiRequestContext.getAddedHeaders()).thenReturn(new HashMap<>()); - - ExtensionResponseDTO responseDTOForRequest = obExtensionListener.getResponseDTOForRequest(obapiRequestContext); - Assert.assertEquals(responseDTOForRequest.getResponseStatus(), ExtensionResponseStatus.CONTINUE.toString()); - Assert.assertNull(responseDTOForRequest.getHeaders()); - Assert.assertNotNull(responseDTOForRequest.getPayload()); - Assert.assertNull(responseDTOForRequest.getCustomProperty()); - Assert.assertEquals(responseDTOForRequest.getStatusCode(), 0); - - Mockito.when(obapiResponseContext.isError()).thenReturn(false); - Mockito.when(obapiResponseContext.getModifiedPayload()).thenReturn(TestConstants.CUSTOM_PAYLOAD); - Mockito.when(obapiResponseContext.getAddedHeaders()).thenReturn(new HashMap<>()); - - ExtensionResponseDTO responseDTOForResponse = - obExtensionListener.getResponseDTOForResponse(obapiResponseContext); - Assert.assertEquals(responseDTOForResponse.getResponseStatus(), ExtensionResponseStatus.CONTINUE.toString()); - Assert.assertNull(responseDTOForResponse.getHeaders()); - Assert.assertNotNull(responseDTOForResponse.getPayload()); - Assert.assertNull(responseDTOForResponse.getCustomProperty()); - Assert.assertEquals(responseDTOForResponse.getStatusCode(), 0); - - } - - @Test(priority = 1) - public void testErrorFlow() { - - obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - Mockito.when(obapiRequestContext.isError()).thenReturn(true); - JSONObject errorJSON = new JSONObject(); - errorJSON.put("error", true); - Mockito.when(obapiRequestContext.getModifiedPayload()).thenReturn(errorJSON.toString()); - Mockito.when(obapiRequestContext.getAddedHeaders()).thenReturn(new HashMap<>()); - - obapiResponseContext = Mockito.mock(OBAPIResponseContext.class); - Mockito.when(obapiResponseContext.isError()).thenReturn(true); - Mockito.when(obapiResponseContext.getModifiedPayload()).thenReturn(errorJSON.toString()); - Mockito.when(obapiResponseContext.getAddedHeaders()).thenReturn(new HashMap<>()); - - ExtensionResponseDTO responseDTOForRequest = obExtensionListener.getResponseDTOForRequest(obapiRequestContext); - Assert.assertEquals(responseDTOForRequest.getResponseStatus(), ExtensionResponseStatus.RETURN_ERROR.toString()); - Assert.assertNull(responseDTOForRequest.getHeaders()); - Assert.assertNotNull(responseDTOForRequest.getPayload()); - Assert.assertNull(responseDTOForRequest.getCustomProperty()); - Assert.assertEquals(responseDTOForRequest.getStatusCode(), 500); - - ExtensionResponseDTO responseDTOForResponse = - obExtensionListener.getResponseDTOForResponse(obapiResponseContext); - - Assert.assertEquals(responseDTOForResponse.getResponseStatus(), - ExtensionResponseStatus.RETURN_ERROR.toString()); - Assert.assertNull(responseDTOForResponse.getHeaders()); - Assert.assertNotNull(responseDTOForResponse.getPayload()); - Assert.assertNull(responseDTOForResponse.getCustomProperty()); - Assert.assertEquals(responseDTOForResponse.getStatusCode(), 500); - - } - - @Test(priority = 1) - public void testFlowWithReturnResponseTrue() { - - Map contextProps = new HashMap<>(); - contextProps.put(GatewayConstants.IS_RETURN_RESPONSE, "true"); - contextProps.put(GatewayConstants.MODIFIED_STATUS, String.valueOf(HttpStatus.SC_CREATED)); - MsgInfoDTO msgInfoDTO = new MsgInfoDTO(); - msgInfoDTO.setHeaders(new HashMap<>()); - - obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - Mockito.when(obapiRequestContext.isError()).thenReturn(false); - Mockito.when(obapiRequestContext.getModifiedPayload()).thenReturn(null); - Mockito.when(obapiRequestContext.getAddedHeaders()).thenReturn(new HashMap<>()); - Mockito.when(obapiRequestContext.getMsgInfo()).thenReturn(msgInfoDTO); - Mockito.when(obapiRequestContext.getContextProps()).thenReturn(contextProps); - - obapiResponseContext = Mockito.mock(OBAPIResponseContext.class); - Mockito.when(obapiResponseContext.isError()).thenReturn(false); - Mockito.when(obapiResponseContext.getModifiedPayload()).thenReturn(null); - Mockito.when(obapiResponseContext.getAddedHeaders()).thenReturn(new HashMap<>()); - Mockito.when(obapiResponseContext.getMsgInfo()).thenReturn(msgInfoDTO); - Mockito.when(obapiResponseContext.getContextProps()).thenReturn(contextProps); - - ExtensionResponseDTO responseDTOForRequest = obExtensionListener.getResponseDTOForRequest(obapiRequestContext); - Assert.assertEquals(responseDTOForRequest.getResponseStatus(), ExtensionResponseStatus.RETURN_ERROR.toString()); - Assert.assertEquals(responseDTOForRequest.getStatusCode(), HttpStatus.SC_CREATED); - - ExtensionResponseDTO responseDTOForResponse = - obExtensionListener.getResponseDTOForResponse(obapiResponseContext); - - Assert.assertEquals(responseDTOForResponse.getResponseStatus(), - ExtensionResponseStatus.RETURN_ERROR.toString()); - Assert.assertEquals(responseDTOForResponse.getStatusCode(), HttpStatus.SC_CREATED); - - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/core/UtilityTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/core/UtilityTest.java deleted file mode 100644 index 58c87c1f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/core/UtilityTest.java +++ /dev/null @@ -1,422 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.core; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.gateway.executor.exception.OpenBankingExecutorException; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import io.swagger.parser.OpenAPIParser; -import io.swagger.v3.oas.models.OpenAPI; -import org.json.JSONObject; -import org.testng.Assert; -import org.testng.annotations.Test; - -import java.io.UnsupportedEncodingException; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -/** - * Test for utility methods used in gateway. - */ -public class UtilityTest { - - private static final String TEST_JWT = "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwI" + - "iwibmFtZSI6IkpvaG4gRG9lIiwiaWF0IjoxNTE2MjM5MDIyfQ.SflKxwRJSMeKKF2QT4fwpMeJf36POk6yJV_adQssw5c"; - private static final String B64_PAYLOAD = "eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6Ikpva" + - "G4gRG9lIiwiaWF0IjoxNTE2MjM5MDIyfQ"; - private static final String XML_PAYLOAD = "" + - "\n" + - " \n" + - " \n" + - " ABC/086\n" + - " TRF\n" + - " false\n" + - " \n" + - "

2012-09-29
\n" + - " \n" + - " \n" + - " \n" + - ""; - private String signingPayload = "\n" + - " \n" + - " \n" + - " ABC/086\n" + - " TRF\n" + - " false\n" + - " \n" + - "
2012-09-29
\n" + - "
\n" + - "
\n" + - "
\n" + - "
"; - private static final String APPLICATION = "APPLICATION"; - private static final String APPLICATION_USER = "APPLICATION_USER"; - - @Test(priority = 1) - public void testB64Encode() throws UnsupportedEncodingException { - - JSONObject payload = GatewayUtils.decodeBase64(B64_PAYLOAD); - Assert.assertEquals(payload.getString("sub"), "1234567890"); - Assert.assertEquals(payload.getString("name"), "John Doe"); - Assert.assertEquals(payload.getInt("iat"), 1516239022); - } - - @Test(priority = 1) - public void testJWTPayloadLoad() { - - Assert.assertEquals(GatewayUtils.getPayloadFromJWT(TEST_JWT), B64_PAYLOAD); - } - - @Test(priority = 1) - public void testBasicAuthHeader() { - - Assert.assertEquals(GatewayUtils.getBasicAuthHeader("admin", "admin"), - "Basic YWRtaW46YWRtaW4="); - } - - @Test(priority = 2) - public void testGetXMLPayloadToSign() throws OpenBankingException { - - Assert.assertEquals(GatewayUtils.getXMLPayloadToSign(XML_PAYLOAD), signingPayload); - } - - @Test (priority = 2) - public void testIsEligibleRequest() { - - Assert.assertTrue(GatewayUtils.isEligibleRequest(GatewayConstants.JSON_CONTENT_TYPE, - GatewayConstants.POST_HTTP_METHOD)); - Assert.assertTrue(GatewayUtils.isEligibleRequest(GatewayConstants.APPLICATION_XML_CONTENT_TYPE, - GatewayConstants.POST_HTTP_METHOD)); - Assert.assertTrue(GatewayUtils.isEligibleRequest(GatewayConstants.TEXT_XML_CONTENT_TYPE, - GatewayConstants.POST_HTTP_METHOD)); - Assert.assertTrue(GatewayUtils.isEligibleRequest(GatewayConstants.JSON_CONTENT_TYPE, - GatewayConstants.PUT_HTTP_METHOD)); - Assert.assertTrue(GatewayUtils.isEligibleRequest(GatewayConstants.APPLICATION_XML_CONTENT_TYPE, - GatewayConstants.PUT_HTTP_METHOD)); - Assert.assertTrue(GatewayUtils.isEligibleRequest(GatewayConstants.TEXT_XML_CONTENT_TYPE, - GatewayConstants.PUT_HTTP_METHOD)); - } - - @Test (priority = 3) - public void testIsEligibleResponse() { - - Assert.assertTrue(GatewayUtils.isEligibleResponse(GatewayConstants.JSON_CONTENT_TYPE, - GatewayConstants.POST_HTTP_METHOD)); - Assert.assertTrue(GatewayUtils.isEligibleResponse(GatewayConstants.APPLICATION_XML_CONTENT_TYPE, - GatewayConstants.POST_HTTP_METHOD)); - Assert.assertTrue(GatewayUtils.isEligibleResponse(GatewayConstants.TEXT_XML_CONTENT_TYPE, - GatewayConstants.POST_HTTP_METHOD)); - Assert.assertTrue(GatewayUtils.isEligibleResponse(GatewayConstants.JSON_CONTENT_TYPE, - GatewayConstants.PUT_HTTP_METHOD)); - Assert.assertTrue(GatewayUtils.isEligibleResponse(GatewayConstants.APPLICATION_XML_CONTENT_TYPE, - GatewayConstants.PUT_HTTP_METHOD)); - Assert.assertTrue(GatewayUtils.isEligibleResponse(GatewayConstants.TEXT_XML_CONTENT_TYPE, - GatewayConstants.PUT_HTTP_METHOD)); - Assert.assertTrue(GatewayUtils.isEligibleResponse(GatewayConstants.JSON_CONTENT_TYPE, - GatewayConstants.GET_HTTP_METHOD)); - Assert.assertTrue(GatewayUtils.isEligibleResponse(GatewayConstants.APPLICATION_XML_CONTENT_TYPE, - GatewayConstants.GET_HTTP_METHOD)); - Assert.assertTrue(GatewayUtils.isEligibleResponse(GatewayConstants.TEXT_XML_CONTENT_TYPE, - GatewayConstants.GET_HTTP_METHOD)); - } - - @Test(description = "Test the extraction of grant type from jwt token payload") - public void getTokenTypeForUserAccessTokens() throws OpenBankingExecutorException { - - String tokenPayload = "eyJzdWIiOiJhZG1pbkB3c28yLmNvbUBjYXJib24uc3VwZXIiLCJhdXQiOiJBUFBMSUNBVElPTl9VU0VSIiwiY" + - "XVkIjoiZldwTmNEVzFZM3FwVFVwcHp3SGFGMnZaWllBYSIsIm5iZiI6MTYxNzc5NjM3OCwiZ3JhbnRfdHlwZSI6ImNsaWVudF9j" + - "cmVkZW50aWFscyIsImNvbnNlbnRfaWQiOiJPQl8xMjM0IiwiYXpwIjoiZldwTmNEVzFZM3FwVFVwcHp3SGFGMnZaWllBYSIsInN" + - "jb3BlIjoiYWNjb3VudHMgcGF5bWVudHMiLCJpc3MiOiJodHRwczovL2xvY2FsaG9zdDo5NDQ2L29hdXRoMi90b2tlbiIsImNuZi" + - "I6eyJ4NXQjUzI1NiI6ImRtaVk1ZE03cE81VzdpbjhrUmFqZkFycXlUTU9uRlcyOVdCVU5rUUlYZTgifSwiZXhwIjoxNjE3Nzk5O" + - "Tc4LCJpYXQiOjE2MTc3OTYzNzgsImp0aSI6Ijk5NjQwN2RjLTY5MmYtNDk0Ni1hMGRlLTRlOWJkNTU3NWRmNSIsImFsZyI6IkhT" + - "MjU2In0"; - String grantType = GatewayUtils.getTokenType(tokenPayload); - Assert.assertEquals(grantType, APPLICATION_USER); - } - - - @Test - public void getTokenTypeForApplicationAccessTokens() throws OpenBankingExecutorException { - - String tokenPayload = "eyJzdWIiOiJhZG1pbkB3c28yLmNvbUBjYXJib24uc3VwZXIiLCJhdXQiOiJBUFBMSUNBVElPTiIsImF1ZCI6I" + - "mZXcE5jRFcxWTNxcFRVcHB6d0hhRjJ2WlpZQWEiLCJuYmYiOjE2MTc3OTYzNzgsImdyYW50X3R5cGUiOiJjbGllbnRfY3JlZGVu" + - "dGlhbHMiLCJhenAiOiJmV3BOY0RXMVkzcXBUVXBwendIYUYydlpaWUFhIiwic2NvcGUiOiJhY2NvdW50cyBwYXltZW50cyIsIml" + - "zcyI6Imh0dHBzOlwvXC9sb2NhbGhvc3Q6OTQ0Nlwvb2F1dGgyXC90b2tlbiIsImNuZiI6eyJ4NXQjUzI1NiI6ImRtaVk1ZE03cE" + - "81VzdpbjhrUmFqZkFycXlUTU9uRlcyOVdCVU5rUUlYZTgifSwiZXhwIjoxNjE3Nzk5OTc4LCJpYXQiOjE2MTc3OTYzNzgsImp0a" + - "SI6Ijk5NjQwN2RjLTY5MmYtNDk0Ni1hMGRlLTRlOWJkNTU3NWRmNSJ9"; - String grantType = GatewayUtils.getTokenType(tokenPayload); - Assert.assertEquals(grantType, APPLICATION); - } - - @Test(description = "Test the extraction of grant type from jwt token payload for GET requests") - public void getAllowedOAuthFlowsForGetRequest() { - - List oauthFlows = new ArrayList<>(); - oauthFlows.add("authorization_code"); - - OpenAPI openAPI = getOpenAPI(); - String electedResource = "/testResource"; - String httpMethod = "GET"; - - Assert.assertEquals(GatewayUtils.getAllowedOAuthFlowsFromSwagger(openAPI, electedResource, httpMethod), - oauthFlows); - } - - @Test(description = "Test the extraction of grant type from jwt token payload for POST requests") - public void getAllowedOAuthFlowsForPostRequest() { - - List oauthFlows = new ArrayList<>(); - oauthFlows.add("authorization_code"); - - OpenAPI openAPI = getOpenAPI(); - String electedResource = "/testResource"; - String httpMethod = "POST"; - - Assert.assertEquals(GatewayUtils.getAllowedOAuthFlowsFromSwagger(openAPI, electedResource, httpMethod), - oauthFlows); - } - - @Test(description = "Test the extraction of grant type from jwt token payload for PUT requests") - public void getAllowedOAuthFlowsForPutRequest() { - - List oauthFlows = new ArrayList<>(); - oauthFlows.add("authorization_code"); - - OpenAPI openAPI = getOpenAPI(); - String electedResource = "/testResource"; - String httpMethod = "PUT"; - - Assert.assertEquals(GatewayUtils.getAllowedOAuthFlowsFromSwagger(openAPI, electedResource, httpMethod), - oauthFlows); - } - - @Test(description = "Test the extraction of grant type from jwt token payload for DELETE requests") - public void getAllowedOAuthFlowsForDeleteRequest() { - - List oauthFlows = new ArrayList<>(); - oauthFlows.add("authorization_code"); - - OpenAPI openAPI = getOpenAPI(); - String electedResource = "/testResource"; - String httpMethod = "DELETE"; - - Assert.assertEquals(GatewayUtils.getAllowedOAuthFlowsFromSwagger(openAPI, electedResource, httpMethod), - oauthFlows); - } - - @Test(description = "Test the extraction of grant type from jwt token payload for PATCH requests") - public void getAllowedOAuthFlowsForPatchRequest() { - - List oauthFlows = new ArrayList<>(); - oauthFlows.add("authorization_code"); - - OpenAPI openAPI = getOpenAPI(); - String electedResource = "/testResource"; - String httpMethod = "PATCH"; - - Assert.assertEquals(GatewayUtils - .getAllowedOAuthFlowsFromSwagger(openAPI, electedResource, httpMethod), oauthFlows); - } - - @Test - public void checkValidityForCorrectClientCredentialsGrantType() throws OpenBankingExecutorException { - - List oauthFlows = new ArrayList<>(); - oauthFlows.add("client_credentials"); - String grantType = APPLICATION; - GatewayUtils.validateGrantType(grantType, oauthFlows); - } - - @Test - public void checkValidityForCorrectAuthCodeGrantType() throws OpenBankingExecutorException { - - List oauthFlows = new ArrayList<>(); - oauthFlows.add("authorization_code"); - String grantType = APPLICATION_USER; - GatewayUtils.validateGrantType(grantType, oauthFlows); - } - - @Test(expectedExceptions = OpenBankingExecutorException.class) - public void checkValidityForInvalidClientCredentialsGrantType() throws OpenBankingExecutorException { - - List oauthFlows = new ArrayList<>(); - oauthFlows.add("authorization_code"); - String grantType = APPLICATION; - GatewayUtils.validateGrantType(grantType, oauthFlows); - } - - @Test(expectedExceptions = OpenBankingExecutorException.class) - public void checkValidityForInvalidAuthCodeGrantType() throws OpenBankingExecutorException { - - List oauthFlows = new ArrayList<>(); - oauthFlows.add("client_credentials"); - String grantType = APPLICATION_USER; - GatewayUtils.validateGrantType(grantType, oauthFlows); - } - - @Test(expectedExceptions = OpenBankingExecutorException.class) - public void getTokenPayloadWhenAuhHeaderNotPresent() throws OpenBankingExecutorException { - - Map transportHeaders = new HashMap<>(); - GatewayUtils.getBearerTokenPayload(transportHeaders); - } - - @Test - public void getTokenPayloadWhenAuhHeaderPresent() throws OpenBankingExecutorException { - - String sampleToken = "Bearer abc.xyz.123"; - Map transportHeaders = new HashMap<>(); - transportHeaders.put("Authorization", sampleToken); - Assert.assertEquals(GatewayUtils.getBearerTokenPayload(transportHeaders), "xyz"); - } - - private OpenAPI getOpenAPI() { - - String swagger = "openapi: 3.0.1\n" + - "info:\n" + - " title: TestAPI\n" + - " version: \"1.0.0\"\n" + - "servers:\n" + - "- url: /testapi/{version}\n" + - "paths:\n" + - " /testResource:\n" + - " get:\n" + - " tags:\n" + - " - Client Registration\n" + - " summary: Get a Client Registration for a given Client ID\n" + - " responses:\n" + - " 200:\n" + - " description: Client registration retrieval success\n" + - " content:\n" + - " application/json:\n" + - " schema: {}\n" + - " security:\n" + - " - PSUOAuth2Security: \n" + - " - accounts\n" + - " - default:\n" + - " - accounts\n" + - " x-auth-type: Application\n" + - " x-throttling-tier: Unlimited\n" + - " post:\n" + - " tags:\n" + - " - Client Registration\n" + - " summary: Get a Client Registration for a given Client ID\n" + - " responses:\n" + - " 200:\n" + - " description: Client registration retrieval success\n" + - " content:\n" + - " application/json:\n" + - " schema: {}\n" + - " security:\n" + - " - PSUOAuth2Security: \n" + - " - accounts\n" + - " - default:\n" + - " - accounts\n" + - " x-auth-type: Application\n" + - " x-throttling-tier: Unlimited\n" + - " delete:\n" + - " tags:\n" + - " - Client Registration\n" + - " summary: Get a Client Registration for a given Client ID\n" + - " responses:\n" + - " 200:\n" + - " description: Client registration retrieval success\n" + - " content:\n" + - " application/json:\n" + - " schema: {}\n" + - " security:\n" + - " - PSUOAuth2Security: \n" + - " - accounts\n" + - " - default:\n" + - " - accounts\n" + - " x-auth-type: Application\n" + - " x-throttling-tier: Unlimited\n" + - " put:\n" + - " tags:\n" + - " - Client Registration\n" + - " summary: Get a Client Registration for a given Client ID\n" + - " responses:\n" + - " 200:\n" + - " description: Client registration retrieval success\n" + - " content:\n" + - " application/json:\n" + - " schema: {}\n" + - " security:\n" + - " - PSUOAuth2Security: \n" + - " - accounts\n" + - " - default:\n" + - " - accounts\n" + - " x-auth-type: Application\n" + - " x-throttling-tier: Unlimited\n" + - " patch:\n" + - " tags:\n" + - " - Client Registration\n" + - " summary: Get a Client Registration for a given Client ID\n" + - " responses:\n" + - " 200:\n" + - " description: Client registration retrieval success\n" + - " content:\n" + - " application/json:\n" + - " schema: {}\n" + - " security:\n" + - " - PSUOAuth2Security: \n" + - " - accounts\n" + - " - default:\n" + - " - accounts\n" + - " x-auth-type: Application\n" + - " x-throttling-tier: Unlimited\n" + - "components:\n" + - " securitySchemes:\n" + - " TPPOAuth2Security:\n" + - " type: oauth2\n" + - " description: TPP client credential authorisation flow with the ASPSP\n" + - " flows:\n" + - " clientCredentials:\n" + - " tokenUrl: https://authserver.example/token\n" + - " scopes: \n" + - " accounts: Ability to read Accounts information\n" + - " PSUOAuth2Security:\n" + - " type: oauth2\n" + - " description: >-\n" + - " OAuth flow, it is required when the PSU needs to perform SCA with the\n" + - " ASPSP when a TPP wants to access an ASPSP resource owned by the PSU\n" + - " flows:\n" + - " authorizationCode:\n" + - " authorizationUrl: 'https://authserver.example/authorization'\n" + - " tokenUrl: 'https://authserver.example/token'\n" + - " scopes:\n" + - " accounts: Ability to read Accounts information\n" + - " default:\n" + - " type: oauth2\n" + - " flows:\n" + - " implicit: \n" + - " authorizationUrl: https://test.com\n" + - " scopes:\n" + - " accounts: Ability to read Accounts information"; - OpenAPIParser parser = new OpenAPIParser(); - return parser.readContents(swagger, - null, null).getOpenAPI(); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/dcr/DCRExecutorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/dcr/DCRExecutorTest.java deleted file mode 100644 index f790554d..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/dcr/DCRExecutorTest.java +++ /dev/null @@ -1,838 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.dcr; - -import com.google.gson.JsonArray; -import com.google.gson.JsonObject; -import com.google.gson.JsonParser; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.gateway.cache.GatewayCache; -import com.wso2.openbanking.accelerator.gateway.cache.GatewayCacheKey; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.internal.GatewayDataHolder; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import org.apache.http.HttpStatus; -import org.mockito.InjectMocks; -import org.mockito.Mock; -import org.mockito.Mockito; -import org.mockito.MockitoAnnotations; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.reflect.internal.WhiteboxImpl; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; -import org.wso2.carbon.apimgt.common.gateway.dto.APIRequestInfoDTO; -import org.wso2.carbon.apimgt.common.gateway.dto.MsgInfoDTO; -import org.wso2.carbon.apimgt.common.gateway.dto.RequestContextDTO; -import org.wso2.carbon.apimgt.impl.APIManagerConfiguration; -import org.wso2.carbon.apimgt.impl.APIManagerConfigurationService; -import org.wso2.carbon.identity.core.util.IdentityUtil; - -import java.io.IOException; -import java.net.URISyntaxException; -import java.text.ParseException; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import javax.ws.rs.HttpMethod; - -import static org.mockito.Matchers.anyObject; -import static org.mockito.Matchers.anyString; -import static org.mockito.Matchers.eq; -import static org.mockito.Mockito.times; -import static org.mockito.Mockito.verify; - -/** - * Test for DCR executor. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({IdentityUtil.class, GatewayDataHolder.class, OpenBankingConfigParser.class}) -public class DCRExecutorTest { - - @Mock - Map urlMap = new HashMap<>(); - - @Mock - OpenBankingConfigurationService openBankingConfigurationService; - - @Mock - APIManagerConfigurationService apiManagerConfigurationService; - - @Mock - APIManagerConfiguration apiManagerConfiguration; - - @Mock - OpenBankingConfigParser openBankingConfigParser; - - @InjectMocks - DCRExecutor dcrExecutor = new DCRExecutor(); - - Map configMap = new HashMap<>(); - Map> configuredAPIList = new HashMap<>(); - JsonParser jsonParser = new JsonParser(); - - @BeforeTest - public void init() { - - MockitoAnnotations.initMocks(this); - - urlMap = new HashMap<>(); - urlMap.put("userName", "admin"); - urlMap.put("password", "admin".toCharArray()); - urlMap.put(GatewayConstants.IAM_HOSTNAME, "localhost"); - urlMap.put(GatewayConstants.IAM_DCR_URL, "dcr/register"); - urlMap.put(GatewayConstants.TOKEN_URL, "/token"); - urlMap.put(GatewayConstants.APP_CREATE_URL, "/appCreate"); - urlMap.put(GatewayConstants.KEY_MAP_URL, "/keyMap/application-id"); - urlMap.put(GatewayConstants.API_RETRIEVE_URL, "/apis"); - urlMap.put(GatewayConstants.API_SUBSCRIBE_URL, "/subscriptions/multiple"); - urlMap.put(GatewayConstants.API_GET_SUBSCRIBED, "/subscriptions"); - - configMap.put(GatewayConstants.REQUEST_ROUTER, - "com.wso2.openbanking.accelerator.gateway.executor.core.DefaultRequestRouter"); - configMap.put(GatewayConstants.GATEWAY_CACHE_EXPIRY, "1"); - configMap.put(GatewayConstants.GATEWAY_CACHE_MODIFIEDEXPIRY, "1"); - configMap.put(OpenBankingConstants.STORE_HOSTNAME, "localhost"); - configMap.put(OpenBankingConstants.TOKEN_ENDPOINT, "/token"); - configMap.put(OpenBankingConstants.APIM_APPCREATION, "/appCreation"); - configMap.put(OpenBankingConstants.APIM_KEYGENERATION, "/keygeneration"); - configMap.put(OpenBankingConstants.APIM_GETAPIS, "/getAPIs"); - configMap.put(OpenBankingConstants.APIM_SUBSCRIBEAPIS, "/subscribe"); - configMap.put(OpenBankingConstants.APIM_GETSUBSCRIPTIONS, "/getSubscriptions"); - configMap.put(OpenBankingConstants.OB_KM_NAME, "OBKM"); - Mockito.when(openBankingConfigurationService.getConfigurations()).thenReturn(configMap); - - List dcrRoles = new ArrayList<>(); - dcrRoles.add("AISP"); - dcrRoles.add("PISP"); - List accountRoles = new ArrayList<>(); - accountRoles.add("AISP"); - configuredAPIList.put("DynamicClientRegistration", dcrRoles); - configuredAPIList.put("AccountandTransactionAPI", accountRoles); - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - - String isDcrResponse = "{\"client_name\":\"application_test\",\"client_id\":\"provided_client_id0001\"," + - "\"client_secret\":\"provided_client_secret0001\",\"redirect_uris\":[\"\"]}"; - - String tokenResponse = "{\n" + - " \"access_token\": \"sdsdfcdvvfvfvfv\",\n" + - " \"scope\": \"accounts cdr-register\",\n" + - " \"token_type\": \"Bearer\",\n" + - " \"expires_in\": 3600\n" + - "}"; - String publishedAPIResponse = "{\n" + - " \"count\":2,\n" + - " \"list\":[\n" + - " {\n" + - " \"id\":\"01234567-0123-0123-0123-012345678901\",\n" + - " \"name\":\"DynamicClientRegistration\"\n" + - " },\n" + - " {\n" + - " \"id\":\"2962f3bb-8330-438e-baee-0ee1d6434ba4\",\n" + - " \"name\":\"AccountandTransactionAPI\"\n" + - " }\n" + - " ],\n" + - " \"pagination\":{\n" + - " \"offset\":2,\n" + - " \"limit\":2,\n" + - " \"total\":10,\n" + - " \"next\":\"/apis?limit=2&offset=4\",\n" + - " \"previous\":\"/apis?limit=2&offset=0\"\n" + - " }\n" + - "}"; - - String subscriptionResponse = "[\n" + - " {\n" + - " \"subscriptionId\":\"faae5fcc-cbae-40c4-bf43-89931630d313\",\n" + - " \"applicationId\":\"b3ade481-30b0-4b38-9a67-498a40873a6d\",\n" + - " \"apiId\":\"2962f3bb-8330-438e-baee-0ee1d6434ba4\",\n" + - " \"apiInfo\":{\n" + - " \"id\":\"01234567-0123-0123-0123-012345678901\",\n" + - " \"name\":\"DynamicClientRegistration\"\n" + - " },\n" + - " \"applicationInfo\":{\n" + - " \"applicationId\":\"01234567-0123-0123-0123-012345678901\",\n" + - " \"name\":\"DynamicClientRegistration\",\n" + - " \"throttlingPolicy\":\"Unlimited\",\n" + - " \"description\":\"Sample calculator application\",\n" + - " \"status\":\"APPROVED\",\n" + - " \"groups\":\"\",\n" + - " \"subscriptionCount\":0,\n" + - " \"attributes\":\"External Reference ID, Billing Tier\",\n" + - " \"owner\":\"admin\"\n" + - " },\n" + - " \"throttlingPolicy\":\"Unlimited\",\n" + - " \"requestedThrottlingPolicy\":\"Unlimited\",\n" + - " \"status\":\"UNBLOCKED\",\n" + - " \"redirectionParams\":\"\"\n" + - " }\n" + - "]"; - - String applicationSearchResponse = "{ \"count\": 1, \"list\": " + - "[ { \"applicationId\": \"01234567-0123-0123-0123-012345678901\", " + - "\"name\": \"CalculatorApp\", \"throttlingPolicy\": \"Unlimited\", " + - "\"description\": \"Sample calculator application\", \"status\": \"APPROVED\", " + - "\"groups\": \"\", \"subscriptionCount\": 0," + - " \"attributes\": \"External Reference ID, Billing Tier\", " + - "\"owner\": \"admin\" } ], " + - "\"pagination\": { \"offset\": 0, \"limit\": 1, \"total\": 10, \"next\": \"\", \"previous\": \"\" } }"; - - String createdApplicationResponse = "{\n" + - " \"applicationId\": \"01234567-0123-0123-0123-012345678901\",\n" + - " \"name\": \"CalculatorApp\"\n" + - "}"; - String keyMapResponse = "{\n" + - " \"keyMappingId\": \"92ab520c-8847-427a-a921-3ed19b15aad7\",\n" + - " \"keyManager\": \"Resident Key Manager\",\n" + - " \"consumerKey\": \"vYDoc9s7IgAFdkSyNDaswBX7ejoa\",\n" + - " \"consumerSecret\": \"TIDlOFkpzB7WjufO3OJUhy1fsvAa\"\n" + - "}"; - - String softwareStatement = "eyJhbGciOiJQUzI1NiIsImtpZCI6IkR3TUtkV01tajdQV2ludm9xZlF5WFZ6eVo2USIsInR5cCI6IkpXVCJ9." + - "eyJpc3MiOiJjZHItcmVnaXN0ZXIiLCJpYXQiOjE1NzE4MDgxNjcsImV4cCI6MjE0NzQ4MzY0NiwianRpIjoiM2JjMjA1YTFlYmM5NDNm" + - "YmI2MjRiMTRmY2IyNDExOTYiLCJvcmdfaWQiOiIzQjBCMEE3Qi0zRTdCLTRBMkMtOTQ5Ny1FMzU3QTcxRDA3QzgiLCJvcmdfbmFtZSI6" + - "Ik1vY2sgQ29tcGFueSBJbmMuIiwiY2xpZW50X25hbWUiOiJNb2NrIFNvZnR3YXJlIE5ldyIsImNsaWVudF9kZXNjcmlwdGlvbiI6IkEg" + - "bW9jayBzb2Z0d2FyZSBwcm9kdWN0IGZvciB0ZXN0aW5nIFNTQSIsImNsaWVudF91cmkiOiJodHRwczovL3d3dy5tb2NrY29tcGFueS5j" + - "b20uYXUiLCJyZWRpcmVjdF91cmlzIjpbImh0dHBzOi8vd3d3Lmdvb2dsZS5jb20vcmVkaXJlY3RzL3JlZGlyZWN0MSIsImh0dHBzOi8v" + - "d3d3Lmdvb2dsZS5jb20vcmVkaXJlY3RzL3JlZGlyZWN0MiJdLCJsb2dvX3VyaSI6Imh0dHBzOi8vd3d3Lm1vY2tjb21wYW55LmNvbS5h" + - "dS9sb2dvcy9sb2dvMS5wbmciLCJ0b3NfdXJpIjoiaHR0cHM6Ly93d3cubW9ja2NvbXBhbnkuY29tLmF1L3Rvcy5odG1sIiwicG9saWN5" + - "X3VyaSI6Imh0dHBzOi8vd3d3Lm1vY2tjb21wYW55LmNvbS5hdS9wb2xpY3kuaHRtbCIsImp3a3NfdXJpIjoiaHR0cHM6Ly9rZXlzdG9y" + - "ZS5vcGVuYmFua2luZ3Rlc3Qub3JnLnVrLzAwMTU4MDAwMDFIUVFyWkFBWC85YjV1c0RwYk50bXhEY1R6czdHektwLmp3a3MiLCJyZXZv" + - "Y2F0aW9uX3VyaSI6Imh0dHBzOi8vZ2lzdC5naXRodWJ1c2VyY29udGVudC5jb20vaW1lc2g5NC8zMTcyZTJlNDU3NTdjZGEwOGVjMjcy" + - "N2Y5MGI3MmNlZC9yYXcvZmYwZDNlYWJlNGNkZGNlNDdlZWMwMjI4ZjU5MjE3NTIyM2RkOTJiMi93c28yLWF1LWRjci1kZW1vLmp3a3Mi" + - "LCJyZWNpcGllbnRfYmFzZV91cmkiOiJodHRwczovL3d3dy5tb2NrY29tcGFueS5jb20uYXUiLCJzb2Z0d2FyZV9pZCI6InRlc3QxMjM0" + - "Iiwic29mdHdhcmVfcm9sZXMiOiJkYXRhLXJlY2lwaWVudC1zb2Z0d2FyZS1wcm9kdWN0IEFJU1AiLCJzY29wZSI6ImJhbms6YWNjb3Vu" + - "dHMuYmFzaWM6cmVhZCBiYW5rOmFjY291bnRzLmRldGFpbDpyZWFkIGJhbms6dHJhbnNhY3Rpb25zOnJlYWQgYmFuazpwYXllZXM6cmVh" + - "ZCBiYW5rOnJlZ3VsYXJfcGF5bWVudHM6cmVhZCBjb21tb246Y3VzdG9tZXIuYmFzaWM6cmVhZCBjb21tb246Y3VzdG9tZXIuZGV0YWls" + - "OnJlYWQgY2RyOnJlZ2lzdHJhdGlvbiJ9.O5xHyhgOyAcTyXLqaUD9O2Iz-Dv5i3_P-ADw1A7PrMZV9j8JdrvY0n0QfhV0YKhmiSTYtII" + - "RCFB_9EchBpnfPeVW4AJ9wt-JpQ2_TWCDSnGIlKb0fmepQkbcQmSRvecFpuECFWUIab6rDOz8IOMMuRXZrwghn3LaP5gKbbDT2NhCp0C" + - "GjBZ2RwriIEx4NZjLBXP4RIw7ZhicOdXL3_544vFs6rOs6IjEkK1z9pHaBfyU0j7BRNcCwPL0Y9_zo4VpZ81Bd8IB_AxIpRNOLcpsa5c" + - "c9oD5B-bqqTeWAkI_INjTlDXf-Rq5bBs7ldkuHh0fRNbI0gIyrpT_VyRL3IKIlw"; - - String dcrResponsePayload = "{\n" + - " \"software_id\": \"test1234\",\n" + - " \"software_statement\": \"eyJhbGciOiJQUzI1NiIsImtpZCI6IkR3TUtkV01tajdQV2ludm9xZlF5WFZ6eVo2U" + - "SIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJjZHItcmVnaXN0ZXIiLCJpYXQiOjE1NzE4MDgxNjcsImV4cCI6MjE0NzQ4MzY0Ni" + - "wianRpIjoiM2JjMjA1YTFlYmM5NDNmYmI2MjRiMTRmY2IyNDExOTYiLCJvcmdfaWQiOiIzQjBCMEE3Qi0zRTdCLTRBMkMtO" + - "TQ5Ny1FMzU3QTcxRDA3QzgiLCJvcmdfbmFtZSI6Ik1vY2sgQ29tcGFueSBJbmMuIiwiY2xpZW50X25hbWUiOiJNb2NrIFNv" + - "ZnR3YXJlIE5ldyIsImNsaWVudF9kZXNjcmlwdGlvbiI6IkEgbW9jayBzb2Z0d2FyZSBwcm9kdWN0IGZvciB0ZXN0aW5nIFNT" + - "QSIsImNsaWVudF91cmkiOiJodHRwczovL3d3dy5tb2NrY29tcGFueS5jb20uYXUiLCJyZWRpcmVjdF91cmlzIjpbImh0dHB" + - "zOi8vd3d3Lmdvb2dsZS5jb20vcmVkaXJlY3RzL3JlZGlyZWN0MSIsImh0dHBzOi8vd3d3Lmdvb2dsZS5jb20vcmVkaXJlY3" + - "RzL3JlZGlyZWN0MiJdLCJsb2dvX3VyaSI6Imh0dHBzOi8vd3d3Lm1vY2tjb21wYW55LmNvbS5hdS9sb2dvcy9sb2dvMS5wb" + - "mciLCJ0b3NfdXJpIjoiaHR0cHM6Ly93d3cubW9ja2NvbXBhbnkuY29tLmF1L3Rvcy5odG1sIiwicG9saWN5X3VyaSI6Imh0" + - "dHBzOi8vd3d3Lm1vY2tjb21wYW55LmNvbS5hdS9wb2xpY3kuaHRtbCIsImp3a3NfdXJpIjoiaHR0cHM6Ly9rZXlzdG9yZS5" + - "vcGVuYmFua2luZ3Rlc3Qub3JnLnVrLzAwMTU4MDAwMDFIUVFyWkFBWC85YjV1c0RwYk50bXhEY1R6czdHektwLmp3a3MiLC" + - "JyZXZvY2F0aW9uX3VyaSI6Imh0dHBzOi8vZ2lzdC5naXRodWJ1c2VyY29udGVudC5jb20vaW1lc2g5NC8zMTcyZTJlNDU3N" + - "TdjZGEwOGVjMjcyN2Y5MGI3MmNlZC9yYXcvZmYwZDNlYWJlNGNkZGNlNDdlZWMwMjI4ZjU5MjE3NTIyM2RkOTJiMi93c28y" + - "LWF1LWRjci1kZW1vLmp3a3MiLCJyZWNpcGllbnRfYmFzZV91cmkiOiJodHRwczovL3d3dy5tb2NrY29tcGFueS5jb20uYXU" + - "iLCJzb2Z0d2FyZV9pZCI6InRlc3QxMjM0Iiwic29mdHdhcmVfcm9sZXMiOiJkYXRhLXJlY2lwaWVudC1zb2Z0d2FyZS1wcm" + - "9kdWN0IEFJU1AiLCJzY29wZSI6ImJhbms6YWNjb3VudHMuYmFzaWM6cmVhZCBiYW5rOmFjY291bnRzLmRldGFpbDpyZWFkI" + - "GJhbms6dHJhbnNhY3Rpb25zOnJlYWQgYmFuazpwYXllZXM6cmVhZCBiYW5rOnJlZ3VsYXJfcGF5bWVudHM6cmVhZCBjb21t" + - "b246Y3VzdG9tZXIuYmFzaWM6cmVhZCBjb21tb246Y3VzdG9tZXIuZGV0YWlsOnJlYWQgY2RyOnJlZ2lzdHJhdGlvbiJ9.O5" + - "xHyhgOyAcTyXLqaUD9O2Iz-Dv5i3_P-ADw1A7PrMZV9j8JdrvY0n0QfhV0YKhmiSTYtIIRCFB_9EchBpnfPeVW4AJ9wt-Jp" + - "Q2_TWCDSnGIlKb0fmepQkbcQmSRvecFpuECFWUIab6rDOz8IOMMuRXZrwghn3LaP5gKbbDT2NhCp0CGjBZ2RwriIEx4NZjL" + - "BXP4RIw7ZhicOdXL3_544vFs6rOs6IjEkK1z9pHaBfyU0j7BRNcCwPL0Y9_zo4VpZ81Bd8IB_AxIpRNOLcpsa5cc9oD5B-b" + - "qqTeWAkI_INjTlDXf-Rq5bBs7ldkuHh0fRNbI0gIyrpT_VyRL3IKIlw\",\n" + - " \"grant_types\": [\n" + - " \"client_credentials\",\n" + - " \"authorization_code\",\n" + - " \"refresh_token\",\n" + - " \"urn:ietf:params:oauth:grant-type:jwt-bearer\"\n" + - " ],\n" + - " \"application_type\": \"web\",\n" + - " \"scope\": \"bank:accounts.basic:read bank:accounts.detail:read bank:transactions:read " + - "bank:payees:read bank:regular_payments:read common:customer.basic:read common:customer.detail:read" + - " cdr:registration\",\n" + - " \"client_id_issued_at\": 1619150285,\n" + - " \"redirect_uris\": [\n" + - " \"https://www.google.com/redirects/redirect1\",\n" + - " \"https://www.google.com/redirects/redirect2\"\n" + - " ],\n" + - " \"request_object_signing_alg\": \"PS256\",\n" + - " \"client_id\": \"uagAipmOU5quayzoznU1ddWg6tca\",\n" + - " \"token_endpoint_auth_method\": \"private_key_jwt\",\n" + - " \"response_types\": \"code id_token\",\n" + - " \"id_token_signed_response_alg\": \"PS256\"\n" + - "}"; - - String dcrResponsePayloadWithoutSSA = "{\n" + - " \"software_id\": \"test12345\",\n" + - "\"client_name\" : \"sample app name\",\n" + - " \"grant_types\": [\n" + - " \"client_credentials\",\n" + - " \"authorization_code\",\n" + - " \"refresh_token\",\n" + - " \"urn:ietf:params:oauth:grant-type:jwt-bearer\"\n" + - " ],\n" + - " \"application_type\": \"web\",\n" + - " \"scope\": \"bank:accounts.basic:read bank:accounts.detail:read bank:transactions:read " + - "bank:payees:read bank:regular_payments:read common:customer.basic:read common:customer.detail:read" + - " cdr:registration\",\n" + - " \"client_id_issued_at\": 1619150285,\n" + - " \"redirect_uris\": [\n" + - " \"https://www.google.com/redirects/redirect1\",\n" + - " \"https://www.google.com/redirects/redirect2\"\n" + - " ],\n" + - " \"request_object_signing_alg\": \"PS256\",\n" + - " \"client_id\": \"uagAipmOU5quayzoznU1ddWg6tca\",\n" + - " \"token_endpoint_auth_method\": \"private_key_jwt\",\n" + - " \"response_types\": \"code id_token\",\n" + - " \"id_token_signed_response_alg\": \"PS256\"\n" + - "}"; - - - @Test - public void testFilterRegulatorAPIs() { - - Map> configuredAPIList = new HashMap<>(); - List dcrRoles = new ArrayList<>(); - dcrRoles.add("AISP"); - dcrRoles.add("PISP"); - List accountRoles = new ArrayList<>(); - accountRoles.add("AISP"); - configuredAPIList.put("DynamicClientRegistration", dcrRoles); - configuredAPIList.put("AccountandTransactionAPI", accountRoles); - - JsonArray publishedAPIs = new JsonArray(); - JsonObject dcrApi = new JsonObject(); - dcrApi.addProperty("id", "1"); - dcrApi.addProperty("name", "DynamicClientRegistration"); - publishedAPIs.add(dcrApi); - - DCRExecutor dcrExecutor = new DCRExecutor(); - List filteredAPIList = dcrExecutor.filterRegulatoryAPIs(configuredAPIList, publishedAPIs, dcrRoles); - Assert.assertEquals(filteredAPIList.get(0), "1"); - } - @Test - public void isSubscriptionDeletionFailed() throws OpenBankingException, IOException { - DCRExecutor dcrExecutor = Mockito.spy(DCRExecutor.class); - Mockito.doReturn(false).when(dcrExecutor) - .callDelete(Mockito.anyString(), anyString()); - boolean subscriptionDeletionFailed = dcrExecutor.isSubscriptionDeletionFailed(Mockito.anyString(), - Mockito.anyString()); - Assert.assertTrue(subscriptionDeletionFailed); - } - - @Test - public void testExtractApplicationName() throws ParseException { - - Map configMap = new HashMap<>(); - configMap.put(OpenBankingConstants.DCR_USE_SOFTWAREID_AS_APPNAME, "true"); - configMap.put(OpenBankingConstants.DCR_APPLICATION_NAME_KEY, "software_client_name"); - String applicationName = dcrExecutor.getApplicationName(dcrResponsePayload, configMap); - Assert.assertEquals("test1234", applicationName); - } - - @Test - public void testExtractApplicationNameFromAppDetails() throws ParseException { - - Map configMap = new HashMap<>(); - configMap.put(OpenBankingConstants.DCR_USE_SOFTWAREID_AS_APPNAME, "true"); - configMap.put(OpenBankingConstants.DCR_APPLICATION_NAME_KEY, "software_client_name"); - String applicationName = dcrExecutor.getApplicationName(dcrResponsePayloadWithoutSSA, configMap); - Assert.assertEquals("test12345", applicationName); - } - - @Test - public void testExtractApplicationNameWhenSSANotContainsApplicationNameKey() throws ParseException { - - Map configMap = new HashMap<>(); - configMap.put(OpenBankingConstants.DCR_USE_SOFTWAREID_AS_APPNAME, "false"); - configMap.put(OpenBankingConstants.DCR_APPLICATION_NAME_KEY, "client_name"); - String applicationName = dcrExecutor.getApplicationName(dcrResponsePayloadWithoutSSA, configMap); - Assert.assertEquals("sample app name", applicationName); - } - - @Test - public void testExtractApplicationNameWithSoftwareIDEnabledFalse() throws ParseException { - - Map configMap = new HashMap<>(); - configMap.put(OpenBankingConstants.DCR_USE_SOFTWAREID_AS_APPNAME, "false"); - configMap.put(OpenBankingConstants.DCR_APPLICATION_NAME_KEY, "client_name"); - String applicationName = dcrExecutor.getApplicationName(dcrResponsePayload, configMap); - Assert.assertEquals("Mock Software New", applicationName); - } - - @Test - public void testExtractSoftwareRoles() throws ParseException { - - List roles = dcrExecutor.getRolesFromSSA(softwareStatement); - Assert.assertTrue(roles.contains("data-recipient-software-product")); - Assert.assertTrue(roles.contains("AISP")); - - } - - @Test - public void testGetUnauthorizedAPIs() { - - Map> configuredAPIList = new HashMap<>(); - List dcrRoles = new ArrayList<>(); - dcrRoles.add("AISP"); - dcrRoles.add("PISP"); - List accountRoles = new ArrayList<>(); - accountRoles.add("AISP"); - configuredAPIList.put("DynamicClientRegistration", dcrRoles); - configuredAPIList.put("AccountandTransactionAPI", accountRoles); - - JsonArray subscribedAPIs = new JsonArray(); - JsonObject dcrApi = new JsonObject(); - JsonObject apiInfo = new JsonObject(); - apiInfo.addProperty("name", "DynamicClientRegistration"); - dcrApi.add("apiInfo", apiInfo); - dcrApi.addProperty("subscriptionId", "1"); - subscribedAPIs.add(dcrApi); - - List allowedRoles = new ArrayList<>(); - allowedRoles.add("data-recipient-software-product"); - List unAuthorizedAPIs = dcrExecutor - .getUnAuthorizedAPIs(subscribedAPIs, configuredAPIList, allowedRoles); - Assert.assertTrue(unAuthorizedAPIs.contains("1")); - - } - - @Test - public void testNewAPIsToSubscribe() { - - List allowedAPIs = new ArrayList<>(); - allowedAPIs.add("1"); - allowedAPIs.add("2"); - - List subscribedAPIs = new ArrayList<>(); - subscribedAPIs.add("1"); - - List apisToSubscribe = dcrExecutor.getNewAPIsToSubscribe(allowedAPIs, subscribedAPIs); - Assert.assertTrue(apisToSubscribe.contains("2")); - - allowedAPIs.remove("2"); - apisToSubscribe = dcrExecutor.getNewAPIsToSubscribe(allowedAPIs, subscribedAPIs); - Assert.assertTrue(apisToSubscribe.isEmpty()); - - subscribedAPIs.remove("1"); - apisToSubscribe = dcrExecutor.getNewAPIsToSubscribe(allowedAPIs, subscribedAPIs); - Assert.assertTrue(apisToSubscribe.contains("1")); - - allowedAPIs.add("2"); - apisToSubscribe = dcrExecutor.getNewAPIsToSubscribe(allowedAPIs, subscribedAPIs); - Assert.assertTrue(apisToSubscribe.contains("1")); - Assert.assertTrue(apisToSubscribe.contains("2")); - } - - @Test - public void testPostProcessResponseForRegister() throws Exception { - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - Mockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParser); - Mockito.when(OpenBankingConfigParser.getInstance() - .getSoftwareEnvIdentificationSSAPropertyValueForSandbox()).thenReturn("sandbox"); - OBAPIResponseContext obapiResponseContext = Mockito.mock(OBAPIResponseContext.class); - MsgInfoDTO msgInfoDTO = Mockito.mock(MsgInfoDTO.class); - DCRExecutor dcrExecutor = Mockito.spy(DCRExecutor.class); - Mockito.doReturn(msgInfoDTO).when(obapiResponseContext).getMsgInfo(); - Mockito.doReturn(HttpMethod.POST).when(msgInfoDTO).getHttpMethod(); - Mockito.doReturn(HttpStatus.SC_CREATED).when(obapiResponseContext).getStatusCode(); - Mockito.doReturn(dcrResponsePayload).when(obapiResponseContext).getResponsePayload(); - - Mockito.when(openBankingConfigurationService.getAllowedAPIs()).thenReturn(configuredAPIList); - GatewayDataHolder.getInstance().setApiManagerConfiguration(apiManagerConfigurationService); - Mockito.when(apiManagerConfigurationService.getAPIManagerConfiguration()).thenReturn(apiManagerConfiguration); - Mockito.doReturn("admin").when(apiManagerConfiguration).getFirstProperty(anyString()); - Mockito.doReturn("localhost/services").when(apiManagerConfiguration) - .getFirstProperty("APIKeyValidator.ServerURL"); - - PowerMockito.mockStatic(IdentityUtil.class); - Mockito.when(IdentityUtil.getProperty("OAuth.OAuth2DCREPUrl")).thenReturn("localhost/api/dcr/register"); - Mockito.doReturn(jsonParser.parse(isDcrResponse)).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.IAM_DCR_URL).toString()), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(tokenResponse)).when(dcrExecutor) - .getToken(anyString(), eq(urlMap.get(GatewayConstants.TOKEN_URL).toString()), anyString()); - Mockito.doReturn(jsonParser.parse(isDcrResponse)).when(dcrExecutor) - .callGet(eq("dcr/register/uagAipmOU5quayzoznU1ddWg6tca"), - anyString(), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(createdApplicationResponse)).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.APP_CREATE_URL).toString()), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(keyMapResponse)).when(dcrExecutor) - .callPost(eq("/keyMap/01234567-0123-0123-0123-012345678901"), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(publishedAPIResponse)).when(dcrExecutor) - .callGet(eq(urlMap.get(GatewayConstants.API_RETRIEVE_URL).toString()), anyString(), anyString(), - anyString()); - Mockito.doReturn(jsonParser.parse(subscriptionResponse)).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.API_SUBSCRIBE_URL).toString()), anyString(), anyString()); - Mockito.doReturn(true).when(dcrExecutor) - .callDelete(eq("dcr/register/provided_client_id0001"), anyString()); - GatewayDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - DCRExecutor.setUrlMap(urlMap); - dcrExecutor.postProcessResponse(obapiResponseContext); - verify(obapiResponseContext, times(0)).setError(true); - } - - @Test - public void testPostProcessResponseForUpdate() throws IOException, OpenBankingException, URISyntaxException { - - String subscribedAPIResponse = "{ \"count\": 1, \"list\": " + - "[ { \"subscriptionId\": \"faae5fcc-cbae-40c4-bf43-89931630d313\", " + - "\"applicationId\": \"b3ade481-30b0-4b38-9a67-498a40873a6d\", " + - "\"apiId\": \"2962f3bb-8330-438e-baee-0ee1d6434ba4\", " + - "\"apiInfo\": { \"id\": \"01234567-0123-0123-0123-012345678901\", " + - "\"name\": \"AccountandTransactionAPI\"," + - " \"description\": \"A calculator API that supports basic operations\", " + - "\"context\": \"CalculatorAPI\", \"version\": \"1.0.0\", \"type\": \"WS\", " + - "\"provider\": \"admin\", \"lifeCycleStatus\": \"PUBLISHED\", " + - "\"thumbnailUri\": \"/apis/01234567-0123-0123-0123-012345678901/thumbnail\", " + - "\"avgRating\": 4.5, \"throttlingPolicies\": [ \"Unlimited\", \"Bronze\" ], " + - "\"advertiseInfo\": { \"advertised\": true, \"originalStoreUrl\": \"https://localhost:9443/store\", " + - "\"apiOwner\": \"admin\" }, " + - "\"businessInformation\": { \"businessOwner\": \"businessowner\", \"businessOwnerEmail\": " + - "\"businessowner@wso2.com\", \"technicalOwner\": \"technicalowner\"," + - " \"technicalOwnerEmail\": \"technicalowner@wso2.com\" }, \"isSubscriptionAvailable\": false, " + - "\"monetizationLabel\": \"Free\" }, " + - "\"applicationInfo\": { \"applicationId\": \"01234567-0123-0123-0123-012345678901\"," + - " \"name\": \"CalculatorApp\", \"throttlingPolicy\": \"Unlimited\", " + - "\"description\": \"Sample calculator application\", \"status\": \"APPROVED\", " + - "\"groups\": \"\", \"subscriptionCount\": 0, \"attributes\": \"External Reference ID, Billing Tier\"," + - " \"owner\": \"admin\" }, \"throttlingPolicy\": \"Unlimited\", \"requestedThrottlingPolicy\":" + - " \"Unlimited\", \"status\": \"UNBLOCKED\", \"redirectionParams\": \"\" } ], " + - "\"pagination\": { \"offset\": 0, \"limit\": 1, \"total\": 10, \"next\": \"\", \"previous\": \"\" } }"; - - OBAPIResponseContext obapiResponseContext = Mockito.mock(OBAPIResponseContext.class); - MsgInfoDTO msgInfoDTO = Mockito.mock(MsgInfoDTO.class); - DCRExecutor dcrExecutor = Mockito.spy(DCRExecutor.class); - Mockito.doReturn(msgInfoDTO).when(obapiResponseContext).getMsgInfo(); - Mockito.doReturn(HttpMethod.PUT).when(msgInfoDTO).getHttpMethod(); - Mockito.doReturn(HttpStatus.SC_OK).when(obapiResponseContext).getStatusCode(); - Mockito.doReturn(dcrResponsePayload).when(obapiResponseContext).getResponsePayload(); - - PowerMockito.mockStatic(IdentityUtil.class); - Mockito.when(IdentityUtil.getProperty("OAuth.OAuth2DCREPUrl")).thenReturn("localhost/api/dcr/register"); - GatewayDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - Map configMap = new HashMap<>(); - configMap.put(OpenBankingConstants.DCR_USE_SOFTWAREID_AS_APPNAME, true); - configMap.put(OpenBankingConstants.DCR_APPLICATION_NAME_KEY, "software_name"); - Mockito.when(openBankingConfigurationService.getConfigurations()).thenReturn(configMap); - Mockito.when(openBankingConfigurationService.getAllowedAPIs()).thenReturn(configuredAPIList); - - Mockito.doReturn(jsonParser.parse(isDcrResponse)).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.IAM_DCR_URL).toString()), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(tokenResponse)).when(dcrExecutor) - .getToken(anyString(), eq(urlMap.get(GatewayConstants.TOKEN_URL).toString()), anyString()); - Mockito.doReturn(jsonParser.parse(applicationSearchResponse)).when(dcrExecutor) - .callGet(eq(urlMap.get(GatewayConstants.APP_CREATE_URL).toString()), - anyString(), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(subscribedAPIResponse)).when(dcrExecutor) - .callGet(eq(urlMap.get(GatewayConstants.API_GET_SUBSCRIBED).toString()), - anyString(), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(publishedAPIResponse)).when(dcrExecutor) - .callGet(eq(urlMap.get(GatewayConstants.API_RETRIEVE_URL).toString()), anyString(), anyString(), - anyString()); - Mockito.doReturn(jsonParser.parse(subscriptionResponse)).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.API_SUBSCRIBE_URL).toString()), anyString(), anyString()); - Mockito.doReturn(true).when(dcrExecutor) - .callDelete(eq("dcr/register/provided_client_id0001"), anyString()); - DCRExecutor.setUrlMap(urlMap); - dcrExecutor.postProcessResponse(obapiResponseContext); - verify(obapiResponseContext, times(0)).setError(true); - } - - @Test - public void testPostProcessResponseForDelete() throws Exception { - - OBAPIResponseContext obapiResponseContext = Mockito.mock(OBAPIResponseContext.class); - APIRequestInfoDTO apiRequestInfoDTO = new APIRequestInfoDTO(); - apiRequestInfoDTO.setConsumerKey("clientId"); - MsgInfoDTO msgInfoDTO = Mockito.mock(MsgInfoDTO.class); - DCRExecutor dcrExecutor = Mockito.spy(DCRExecutor.class); - GatewayCache gatewayCache = Mockito.mock(GatewayCache.class); - - PowerMockito.mockStatic(IdentityUtil.class); - - Mockito.when(IdentityUtil.getProperty("OAuth.OAuth2DCREPUrl")).thenReturn("localhost/api/dcr/register"); - Mockito.doReturn(msgInfoDTO).when(obapiResponseContext).getMsgInfo(); - Mockito.doReturn(HttpMethod.DELETE).when(msgInfoDTO).getHttpMethod(); - Mockito.doReturn(HttpStatus.SC_NO_CONTENT).when(obapiResponseContext).getStatusCode(); - Mockito.doReturn(dcrResponsePayload).when(obapiResponseContext).getResponsePayload(); - Mockito.doReturn(apiRequestInfoDTO).when(obapiResponseContext).getApiRequestInfo(); - Mockito.when(openBankingConfigurationService.getAllowedAPIs()).thenReturn(configuredAPIList); - GatewayDataHolder.getInstance().setApiManagerConfiguration(apiManagerConfigurationService); - Mockito.when(apiManagerConfigurationService.getAPIManagerConfiguration()).thenReturn(apiManagerConfiguration); - Mockito.doReturn("admin").when(apiManagerConfiguration).getFirstProperty(anyString()); - Mockito.doReturn("localhost/services").when(apiManagerConfiguration) - .getFirstProperty("APIKeyValidator.ServerURL"); - GatewayDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - WhiteboxImpl.invokeMethod(GatewayDataHolder.getInstance(), "setGatewayCache", gatewayCache); - //GatewayDataHolder.setGatewayCache(gatewayCache); - - Mockito.doReturn("application").when(gatewayCache).getFromCache(GatewayCacheKey.of(anyString())); - Mockito.doReturn(jsonParser.parse(isDcrResponse)).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.IAM_DCR_URL).toString()), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(tokenResponse)).when(dcrExecutor) - .getToken(anyString(), eq(urlMap.get(GatewayConstants.TOKEN_URL).toString()), anyString()); - Mockito.doReturn(jsonParser.parse(applicationSearchResponse)).when(dcrExecutor) - .callGet(eq(urlMap.get(GatewayConstants.APP_CREATE_URL).toString()), - anyString(), anyString(), anyString()); - Mockito.doReturn(true).when(dcrExecutor).callDelete(anyString(), anyString()); - DCRExecutor.setUrlMap(urlMap); - dcrExecutor.postProcessResponse(obapiResponseContext); - verify(obapiResponseContext, times(0)).setError(true); - } - - @Test - public void testPostProcessRequestInvalidAuthenticationDCR() { - - OBAPIRequestContext obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - APIRequestInfoDTO apiRequestInfoDTO = new APIRequestInfoDTO(); - apiRequestInfoDTO.setConsumerKey("clientId"); - MsgInfoDTO msgInfoDTO = new MsgInfoDTO(); - DCRExecutor dcrExecutor = Mockito.spy(DCRExecutor.class); - - msgInfoDTO.setHttpMethod(HttpMethod.GET); - msgInfoDTO.setResource("/register/1234"); - apiRequestInfoDTO.setConsumerKey("123"); - Map requestHeaders = new HashMap<>(); - requestHeaders.put(GatewayConstants.CONTENT_TYPE_TAG, "application/jwt"); - requestHeaders.put(GatewayConstants.CONTENT_LENGTH, "1000"); - msgInfoDTO.setHeaders(requestHeaders); - RequestContextDTO requestContextDTO = new RequestContextDTO(); - requestContextDTO.setApiRequestInfo(apiRequestInfoDTO); - requestContextDTO.setMsgInfo(msgInfoDTO); - - Mockito.doReturn(msgInfoDTO).when(obapiRequestContext).getMsgInfo(); - Mockito.doReturn(apiRequestInfoDTO).when(obapiRequestContext).getApiRequestInfo(); - - dcrExecutor.postProcessRequest(obapiRequestContext); - verify(obapiRequestContext).setError(true); - } - - @Test - public void testPostProcessWithInvalidAuthenticationDCRApplicationAccessToken() throws Exception { - - OBAPIRequestContext obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - GatewayCache gatewayCache = Mockito.mock(GatewayCache.class); - DCRExecutor dcrExecutor = Mockito.spy(DCRExecutor.class); - MsgInfoDTO msgInfoDTO = new MsgInfoDTO(); - Map headers = new HashMap<>(); - headers.put(GatewayConstants.AUTH_HEADER, ""); - msgInfoDTO.setHeaders(headers); - msgInfoDTO.setResource("/register/123"); - msgInfoDTO.setHttpMethod(HttpMethod.DELETE); - APIRequestInfoDTO apiRequestInfoDTO = new APIRequestInfoDTO(); - apiRequestInfoDTO.setConsumerKey("123"); - DCRExecutor.setUrlMap(urlMap); - WhiteboxImpl.invokeMethod(GatewayDataHolder.getInstance(), "setGatewayCache", gatewayCache); - - Mockito.doReturn(msgInfoDTO).when(obapiRequestContext).getMsgInfo(); - Mockito.doReturn(apiRequestInfoDTO).when(obapiRequestContext).getApiRequestInfo(); - Mockito.doReturn(jsonParser.parse(isDcrResponse)).when(dcrExecutor) - .callGet(anyString(), anyString(), anyString(), anyString()); - Mockito.doNothing().when(gatewayCache).addToCache(anyObject(), anyObject()); - - dcrExecutor.postProcessRequest(obapiRequestContext); - verify(gatewayCache).addToCache(anyObject(), anyObject()); - } - - @Test - public void testAddApplicationNameToCacheInDelete() throws Exception { - - OBAPIRequestContext obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - GatewayCache gatewayCache = Mockito.mock(GatewayCache.class); - DCRExecutor dcrExecutor = Mockito.spy(DCRExecutor.class); - MsgInfoDTO msgInfoDTO = new MsgInfoDTO(); - msgInfoDTO.setResource("/register/123"); - msgInfoDTO.setHttpMethod(HttpMethod.DELETE); - APIRequestInfoDTO apiRequestInfoDTO = new APIRequestInfoDTO(); - apiRequestInfoDTO.setConsumerKey("123"); - DCRExecutor.setUrlMap(urlMap); - WhiteboxImpl.invokeMethod(GatewayDataHolder.getInstance(), "setGatewayCache", gatewayCache); - //GatewayDataHolder.setGatewayCache(gatewayCache); - - Mockito.doReturn(msgInfoDTO).when(obapiRequestContext).getMsgInfo(); - Mockito.doReturn(apiRequestInfoDTO).when(obapiRequestContext).getApiRequestInfo(); - Mockito.doReturn(jsonParser.parse(isDcrResponse)).when(dcrExecutor) - .callGet(anyString(), anyString(), anyString(), anyString()); - Mockito.doNothing().when(gatewayCache).addToCache(anyObject(), anyObject()); - - dcrExecutor.postProcessRequest(obapiRequestContext); - verify(gatewayCache).addToCache(anyObject(), anyObject()); - } - - @Test - public void testErrorScenarios() throws IOException, OpenBankingException, URISyntaxException { - - OBAPIResponseContext obapiResponseContext = Mockito.mock(OBAPIResponseContext.class); - MsgInfoDTO msgInfoDTO = Mockito.mock(MsgInfoDTO.class); - DCRExecutor dcrExecutor = Mockito.spy(DCRExecutor.class); - Mockito.doReturn(msgInfoDTO).when(obapiResponseContext).getMsgInfo(); - Mockito.doReturn(HttpMethod.POST).when(msgInfoDTO).getHttpMethod(); - Mockito.doReturn(HttpStatus.SC_CREATED).when(obapiResponseContext).getStatusCode(); - Mockito.doReturn(dcrResponsePayload).when(obapiResponseContext).getResponsePayload(); - - Mockito.when(openBankingConfigurationService.getAllowedAPIs()).thenReturn(configuredAPIList); - PowerMockito.mockStatic(OpenBankingConfigParser.class); - Mockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParser); - Mockito.when(OpenBankingConfigParser.getInstance().getSoftwareEnvIdentificationSSAPropertyValueForSandbox()) - .thenReturn("sandbox"); - GatewayDataHolder.getInstance().setApiManagerConfiguration(apiManagerConfigurationService); - Mockito.when(apiManagerConfigurationService.getAPIManagerConfiguration()).thenReturn(apiManagerConfiguration); - Mockito.doReturn("admin").when(apiManagerConfiguration).getFirstProperty(anyString()); - Mockito.doReturn("localhost/services").when(apiManagerConfiguration) - .getFirstProperty("APIKeyValidator.ServerURL"); - PowerMockito.mockStatic(IdentityUtil.class); - Mockito.when(IdentityUtil.getProperty("OAuth.OAuth2DCREPUrl")).thenReturn("localhost/api/dcr/register"); - - //when sp creation fails for token generation - Mockito.doReturn(null).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.IAM_DCR_URL).toString()), anyString(), anyString()); - Mockito.doReturn(true).when(dcrExecutor) - .callDelete(eq("localhost/api/openbanking/dynamic-client-registration/register/" + - "uagAipmOU5quayzoznU1ddWg6tca"), anyString()); - - GatewayDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - DCRExecutor.setUrlMap(urlMap); - dcrExecutor.postProcessResponse(obapiResponseContext); - verify(obapiResponseContext).setError(true); - - //when token call fails - Mockito.doReturn(jsonParser.parse(isDcrResponse)).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.IAM_DCR_URL).toString()), anyString(), anyString()); - Mockito.doReturn(null).when(dcrExecutor) - .getToken(anyString(), eq(urlMap.get(GatewayConstants.TOKEN_URL).toString()), anyString()); - Mockito.doReturn(true).when(dcrExecutor) - .callDelete(eq("dcr/register/provided_client_id0001"), anyString()); - dcrExecutor.postProcessResponse(obapiResponseContext); - verify(obapiResponseContext, times(2)).setError(true); - - //when retrieving client id and secret fails - Mockito.doReturn(jsonParser.parse(isDcrResponse)).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.IAM_DCR_URL).toString()), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(tokenResponse)).when(dcrExecutor) - .getToken(anyString(), eq(urlMap.get(GatewayConstants.TOKEN_URL).toString()), anyString()); - Mockito.doReturn(null).when(dcrExecutor) - .callGet(eq("dcr/register/uagAipmOU5quayzoznU1ddWg6tca"), - anyString(), anyString(), anyString()); - dcrExecutor.postProcessResponse(obapiResponseContext); - verify(obapiResponseContext, times(3)).setError(true); - - //when AM application creation fails - Mockito.doReturn(jsonParser.parse(isDcrResponse)).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.IAM_DCR_URL).toString()), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(tokenResponse)).when(dcrExecutor) - .getToken(anyString(), eq(urlMap.get(GatewayConstants.TOKEN_URL).toString()), anyString()); - Mockito.doReturn(jsonParser.parse(isDcrResponse)).when(dcrExecutor) - .callGet(eq("dcr/register/uagAipmOU5quayzoznU1ddWg6tca"), - anyString(), anyString(), anyString()); - Mockito.doReturn(null).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.APP_CREATE_URL).toString()), anyString(), anyString()); - dcrExecutor.postProcessResponse(obapiResponseContext); - verify(obapiResponseContext, times(4)).setError(true); - - //when key map response is null - Mockito.doReturn(jsonParser.parse(isDcrResponse)).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.IAM_DCR_URL).toString()), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(tokenResponse)).when(dcrExecutor) - .getToken(anyString(), eq(urlMap.get(GatewayConstants.TOKEN_URL).toString()), anyString()); - Mockito.doReturn(jsonParser.parse(isDcrResponse)).when(dcrExecutor) - .callGet(eq("dcr/register/uagAipmOU5quayzoznU1ddWg6tca"), - anyString(), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(createdApplicationResponse)).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.APP_CREATE_URL).toString()), anyString(), anyString()); - Mockito.doReturn(null).when(dcrExecutor) - .callPost(eq("/keyMap/01234567-0123-0123-0123-012345678901"), anyString(), anyString()); - ///appCreate/01234567-0123-0123-0123-012345678901 - Mockito.doReturn(true).when(dcrExecutor) - .callDelete(eq("/appCreate/01234567-0123-0123-0123-012345678901"), anyString()); - dcrExecutor.postProcessResponse(obapiResponseContext); - verify(obapiResponseContext, times(5)).setError(true); - - //when retrieving published apis get a null response - Mockito.doReturn(jsonParser.parse(isDcrResponse)).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.IAM_DCR_URL).toString()), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(tokenResponse)).when(dcrExecutor) - .getToken(anyString(), eq(urlMap.get(GatewayConstants.TOKEN_URL).toString()), anyString()); - Mockito.doReturn(jsonParser.parse(isDcrResponse)).when(dcrExecutor) - .callGet(eq("dcr/register/uagAipmOU5quayzoznU1ddWg6tca"), - anyString(), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(createdApplicationResponse)).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.APP_CREATE_URL).toString()), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(keyMapResponse)).when(dcrExecutor) - .callPost(eq("/keyMap/01234567-0123-0123-0123-012345678901"), anyString(), anyString()); - Mockito.doReturn(null).when(dcrExecutor) - .callGet(eq(urlMap.get(GatewayConstants.API_RETRIEVE_URL).toString()), anyString(), anyString(), - anyString()); - dcrExecutor.postProcessResponse(obapiResponseContext); - verify(obapiResponseContext, times(6)).setError(true); - - - //when subscribing to APIs get null response - Mockito.doReturn(jsonParser.parse(isDcrResponse)).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.IAM_DCR_URL).toString()), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(tokenResponse)).when(dcrExecutor) - .getToken(anyString(), eq(urlMap.get(GatewayConstants.TOKEN_URL).toString()), anyString()); - Mockito.doReturn(jsonParser.parse(isDcrResponse)).when(dcrExecutor) - .callGet(eq("dcr/register/uagAipmOU5quayzoznU1ddWg6tca"), - anyString(), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(createdApplicationResponse)).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.APP_CREATE_URL).toString()), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(keyMapResponse)).when(dcrExecutor) - .callPost(eq("/keyMap/01234567-0123-0123-0123-012345678901"), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(publishedAPIResponse)).when(dcrExecutor) - .callGet(eq(urlMap.get(GatewayConstants.API_RETRIEVE_URL).toString()), anyString(), anyString(), - anyString()); - Mockito.doReturn(null).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.API_SUBSCRIBE_URL).toString()), anyString(), anyString()); - - dcrExecutor.postProcessResponse(obapiResponseContext); - verify(obapiResponseContext, times(7)).setError(true); - - //when deleting internal SP get false - Mockito.doReturn(jsonParser.parse(isDcrResponse)).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.IAM_DCR_URL).toString()), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(tokenResponse)).when(dcrExecutor) - .getToken(anyString(), eq(urlMap.get(GatewayConstants.TOKEN_URL).toString()), anyString()); - Mockito.doReturn(jsonParser.parse(isDcrResponse)).when(dcrExecutor) - .callGet(eq("dcr/register/uagAipmOU5quayzoznU1ddWg6tca"), - anyString(), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(createdApplicationResponse)).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.APP_CREATE_URL).toString()), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(keyMapResponse)).when(dcrExecutor) - .callPost(eq("/keyMap/01234567-0123-0123-0123-012345678901"), anyString(), anyString()); - Mockito.doReturn(jsonParser.parse(publishedAPIResponse)).when(dcrExecutor) - .callGet(eq(urlMap.get(GatewayConstants.API_RETRIEVE_URL).toString()), anyString(), anyString(), - anyString()); - Mockito.doReturn(jsonParser.parse(subscriptionResponse)).when(dcrExecutor) - .callPost(eq(urlMap.get(GatewayConstants.API_SUBSCRIBE_URL).toString()), anyString(), anyString()); - Mockito.doReturn(false).when(dcrExecutor) - .callDelete(eq("dcr/register/provided_client_id0001"), anyString()); - dcrExecutor.postProcessResponse(obapiResponseContext); - verify(obapiResponseContext, times(8)).setError(true); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/common/reporting/data/executor/CommonReportingDataExecutorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/common/reporting/data/executor/CommonReportingDataExecutorTest.java deleted file mode 100644 index 43328482..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/common/reporting/data/executor/CommonReportingDataExecutorTest.java +++ /dev/null @@ -1,125 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.impl.common.reporting.data.executor; - -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import org.mockito.Mockito; -import org.testng.Assert; -import org.testng.annotations.Test; -import org.wso2.carbon.apimgt.common.gateway.dto.APIRequestInfoDTO; -import org.wso2.carbon.apimgt.common.gateway.dto.MsgInfoDTO; - -import java.util.HashMap; -import java.util.Map; - -/** - * Test for common reporting data executor. - */ -public class CommonReportingDataExecutorTest { - - @Test(priority = 1) - public void testPreRequestFlow() { - - String userAgent = "testUserAgent"; - String electedResource = "/test"; - String messageId = "test_message_id"; - String apiId = "test_apiId"; - String version = "1.0.0"; - String apiName = "testAPI"; - String httpMethod = "GET"; - String sampleToken = "eyJhbGciOiJSUzI1NiJ9.eyJzdWIiOiJ0cHAiLCJhdXQiOiJBUFBMSUNBVElPTiIsImNvbnNlbnRfaWQiOiJ4eX" + - "pfY29uc25ldF9pZCJ9.wTIt8KUMRjpWD5WML6GYM1BlBOHzTdroVjuDaeFyb7gmVoCPF9zUjzIS1FDbHCh-xf8n5RvlruTwSDgu" + - "g3BUMpvy3fJT5C2dQQiXbkH-aofSrWCBjvHSN1D5GJGnd0TmD9cN2nEfFEbhLao8IeJS8Gj-NvMVP7bloZ_USyiD273gLnmWl53" + - "e2pSYYp7N_b97Ci-nH4WnooHq4HS5f94G85CIJQm6vIUjm0wnzQyVg9Uh_BipUrtV1PMz1h5ugOrv003kBmV10oszj4BSZhscuW" + - "4xSe07jgIN7xPvsx02hzSVHjTXA4hWhP7YeTzSCvpVcDPREtjIT800cf35akhv0w"; - - MsgInfoDTO msgInfoDTO = new MsgInfoDTO(); - msgInfoDTO.setHttpMethod(httpMethod); - - Map headers = new HashMap<>(); - headers.put("Authorization", sampleToken); - headers.put("User-Agent", userAgent); - msgInfoDTO.setHeaders(headers); - msgInfoDTO.setElectedResource(electedResource); - msgInfoDTO.setMessageId(messageId); - - APIRequestInfoDTO apiRequestInfoDTO = new APIRequestInfoDTO(); - apiRequestInfoDTO.setApiId(apiId); - apiRequestInfoDTO.setVersion(version); - - OBAPIRequestContext obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - Mockito.when(obapiRequestContext.getAnalyticsData()).thenReturn(new HashMap<>()); - Mockito.when(obapiRequestContext.getMsgInfo()).thenReturn(msgInfoDTO); - Mockito.when(obapiRequestContext.getApiRequestInfo()).thenReturn(apiRequestInfoDTO); - - CommonReportingDataExecutor commonReportingDataExecutor = Mockito.spy(CommonReportingDataExecutor.class); - Mockito.doReturn(apiName).when(commonReportingDataExecutor).getApiName(Mockito.any()); - commonReportingDataExecutor.preProcessRequest(obapiRequestContext); - - Assert.assertEquals(obapiRequestContext.getAnalyticsData().size(), 7); - Assert.assertEquals(obapiRequestContext.getAnalyticsData().get("userAgent"), userAgent); - Assert.assertEquals(obapiRequestContext.getAnalyticsData().get("electedResource"), electedResource); - Assert.assertEquals(obapiRequestContext.getAnalyticsData().get("httpMethod"), httpMethod); - Assert.assertEquals(obapiRequestContext.getAnalyticsData().get("apiName"), apiName); - Assert.assertEquals(obapiRequestContext.getAnalyticsData().get("apiSpecVersion"), version); - Assert.assertEquals(obapiRequestContext.getAnalyticsData().get("messageId"), messageId); - Assert.assertTrue((long) obapiRequestContext.getAnalyticsData().get("timestamp") > 0); - } - - @Test(priority = 2) - public void testPostRequestFlow() { - - String username = "tpp@wso2.conm"; - String clientId = "test_client_id"; - String consentId = "test_consent_id"; - - APIRequestInfoDTO apiRequestInfoDTO = new APIRequestInfoDTO(); - apiRequestInfoDTO.setUsername(username); - apiRequestInfoDTO.setConsumerKey(clientId); - - OBAPIRequestContext obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - Mockito.when(obapiRequestContext.getApiRequestInfo()).thenReturn(apiRequestInfoDTO); - Mockito.when(obapiRequestContext.getAnalyticsData()).thenReturn(new HashMap<>()); - Mockito.when(obapiRequestContext.getConsentId()).thenReturn(consentId); - - CommonReportingDataExecutor commonReportingDataExecutor = Mockito.spy(CommonReportingDataExecutor.class); - commonReportingDataExecutor.postProcessRequest(obapiRequestContext); - - Assert.assertEquals(obapiRequestContext.getAnalyticsData().size(), 3); - Assert.assertEquals(obapiRequestContext.getAnalyticsData().get("consumerId"), username); - Assert.assertEquals(obapiRequestContext.getAnalyticsData().get("consentId"), consentId); - Assert.assertEquals(obapiRequestContext.getAnalyticsData().get("clientId"), clientId); - } - - @Test(priority = 3) - public void testPreResponseFlow() { - CommonReportingDataExecutor commonReportingDataExecutor = new CommonReportingDataExecutor(); - OBAPIResponseContext obapiResponseContext = Mockito.mock(OBAPIResponseContext.class); - Mockito.when(obapiResponseContext.getAnalyticsData()).thenReturn(new HashMap<>()); - Mockito.when(obapiResponseContext.getModifiedPayload()).thenReturn("testPayloadOfLength21"); - Mockito.when(obapiResponseContext.getStatusCode()).thenReturn(200); - - commonReportingDataExecutor.preProcessResponse(obapiResponseContext); - Assert.assertTrue(obapiResponseContext.getAnalyticsData().size() == 2); - Assert.assertEquals(obapiResponseContext.getAnalyticsData().get("responsePayloadSize"), (long) 21); - Assert.assertEquals(obapiResponseContext.getAnalyticsData().get("statusCode"), 200); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/consent/TestEnforcementExecutor.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/consent/TestEnforcementExecutor.java deleted file mode 100644 index 7ca77016..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/consent/TestEnforcementExecutor.java +++ /dev/null @@ -1,110 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.impl.consent; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OpenBankingExecutorError; -import com.wso2.openbanking.accelerator.gateway.executor.test.TestConstants; -import com.wso2.openbanking.accelerator.gateway.internal.GatewayDataHolder; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import org.json.JSONObject; -import org.mockito.Mockito; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -import java.io.File; -import java.io.UnsupportedEncodingException; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Map; - -/** - * Test for enforcement executor. - */ -public class TestEnforcementExecutor { - - private static ConsentEnforcementExecutor consentEnforcementExecutor; - private static OBAPIRequestContext obapiRequestContext; - private String jwtToken = "eyJjdXN0b20iOiJwYXlsb2FkIn0"; - - @BeforeClass - public static void beforeClass() throws OpenBankingException { - - GatewayDataHolder dataHolder = GatewayDataHolder.getInstance(); - String path = "src/test/resources"; - File file = new File(path); - String absolutePathForTestResources = file.getAbsolutePath(); - dataHolder.setKeyStoreLocation(absolutePathForTestResources + "/wso2carbon.jks"); - dataHolder.setKeyAlias("wso2carbon"); - dataHolder.setKeyPassword("wso2carbon"); - dataHolder.setKeyStorePassword("wso2carbon".toCharArray()); - - consentEnforcementExecutor = new ConsentEnforcementExecutor(); - } - - @Test(priority = 1) - public void testSigningKeyRetrieval() { - - Assert.assertNotNull(consentEnforcementExecutor.getJWTSigningKey()); - } - - @Test(priority = 2) - public void testJWTGeneration() { - - String jwtToken = consentEnforcementExecutor.generateJWT(TestConstants.CUSTOM_PAYLOAD); - Assert.assertNotNull(jwtToken); - String[] parts = jwtToken.split("\\."); - Assert.assertEquals(parts.length, 3); - } - - @Test(priority = 2) - public void testValidationPayloadCreation() { - - Map headers = new HashMap<>(); - headers.put("customHeader", "headerValue"); - headers.put("customHeader2", "headerValue2"); - JSONObject jsonObject = - consentEnforcementExecutor.createValidationRequestPayload(headers, - TestConstants.CUSTOM_PAYLOAD, new HashMap<>()); - Assert.assertNotNull(jsonObject); - Assert.assertEquals(((JSONObject) jsonObject.get(ConsentEnforcementExecutor.HEADERS_TAG)).get("customHeader"), - "headerValue"); - Assert.assertEquals(((JSONObject) jsonObject.get(ConsentEnforcementExecutor.BODY_TAG)).get("custom"), - "payload"); - } - - @Test(priority = 3) - public void testB64Decoder() throws UnsupportedEncodingException { - - JSONObject jsonObject = GatewayUtils.decodeBase64(jwtToken); - Assert.assertEquals(jsonObject.get("custom").toString(), "payload"); - } - - @Test - public void testHandlerError() { - OBAPIRequestContext obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - ArrayList errors = new ArrayList<>(); - Mockito.when(obapiRequestContext.getErrors()).thenReturn(errors); - consentEnforcementExecutor.handleError(obapiRequestContext, "Error", "Error", - "400"); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/error/handler/OBDefaultErrorHandlerTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/error/handler/OBDefaultErrorHandlerTest.java deleted file mode 100644 index 810e7b59..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/error/handler/OBDefaultErrorHandlerTest.java +++ /dev/null @@ -1,106 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.impl.error.handler; - -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OpenBankingExecutorError; -import org.mockito.Mockito; -import org.testng.annotations.Test; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Map; - -import static org.mockito.Mockito.times; -import static org.mockito.Mockito.verify; - -/** - * Test class for OBDefaultErrorHandler. - */ -public class OBDefaultErrorHandlerTest { - - Map contextProps = new HashMap<>(); - - @Test - public void testPreRequestFlow() { - - OBAPIRequestContext obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - Mockito.when(obapiRequestContext.isError()).thenReturn(true); - Mockito.when(obapiRequestContext.getErrors()).thenReturn(getErrorList()); - Mockito.when(obapiRequestContext.getContextProps()).thenReturn(contextProps); - Mockito.when(obapiRequestContext.getAnalyticsData()).thenReturn(new HashMap<>()); - - OBDefaultErrorHandler commonReportingDataExecutor = Mockito.spy(OBDefaultErrorHandler.class); - commonReportingDataExecutor.preProcessRequest(obapiRequestContext); - verify(obapiRequestContext, times(0)).setError(false); - } - - @Test - public void testPostRequestFlow() { - - OBAPIRequestContext obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - Mockito.when(obapiRequestContext.isError()).thenReturn(true); - Mockito.when(obapiRequestContext.getErrors()).thenReturn(getErrorList()); - Mockito.when(obapiRequestContext.getContextProps()).thenReturn(contextProps); - Mockito.when(obapiRequestContext.getAnalyticsData()).thenReturn(new HashMap<>()); - - OBDefaultErrorHandler commonReportingDataExecutor = Mockito.spy(OBDefaultErrorHandler.class); - commonReportingDataExecutor.postProcessRequest(obapiRequestContext); - verify(obapiRequestContext, times(0)).setError(false); - } - - @Test - public void testPreResponseFlow() { - - OBAPIResponseContext obapiResponseContext = Mockito.mock(OBAPIResponseContext.class); - Mockito.when(obapiResponseContext.isError()).thenReturn(true); - Mockito.when(obapiResponseContext.getErrors()).thenReturn(getErrorList()); - Mockito.when(obapiResponseContext.getContextProps()).thenReturn(contextProps); - Mockito.when(obapiResponseContext.getAnalyticsData()).thenReturn(new HashMap<>()); - - OBDefaultErrorHandler commonReportingDataExecutor = Mockito.spy(OBDefaultErrorHandler.class); - commonReportingDataExecutor.preProcessResponse(obapiResponseContext); - verify(obapiResponseContext, times(0)).setError(false); - } - - @Test - public void testPostResponseFlow() { - - OBAPIResponseContext obapiResponseContext = Mockito.mock(OBAPIResponseContext.class); - Mockito.when(obapiResponseContext.isError()).thenReturn(true); - Mockito.when(obapiResponseContext.getErrors()).thenReturn(getErrorList()); - Mockito.when(obapiResponseContext.getContextProps()).thenReturn(contextProps); - Mockito.when(obapiResponseContext.getAnalyticsData()).thenReturn(new HashMap<>()); - - OBDefaultErrorHandler commonReportingDataExecutor = Mockito.spy(OBDefaultErrorHandler.class); - commonReportingDataExecutor.postProcessResponse(obapiResponseContext); - verify(obapiResponseContext, times(0)).setError(false); - } - - private ArrayList getErrorList() { - - OpenBankingExecutorError error = new OpenBankingExecutorError("400", "Invalid Request", - "Mandatory parameter is missing", "400"); - - ArrayList errors = new ArrayList<>(); - errors.add(error); - return errors; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/mtls/cert/validation/executor/CertRevocationValidationExecutorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/mtls/cert/validation/executor/CertRevocationValidationExecutorTest.java deleted file mode 100644 index c9f7b963..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/mtls/cert/validation/executor/CertRevocationValidationExecutorTest.java +++ /dev/null @@ -1,229 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.impl.mtls.cert.validation.executor; - -import com.wso2.openbanking.accelerator.common.exception.CertificateValidationException; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.CertificateUtils; -import com.wso2.openbanking.accelerator.gateway.cache.CertificateRevocationCache; -import com.wso2.openbanking.accelerator.gateway.cache.GatewayCacheKey; -import com.wso2.openbanking.accelerator.gateway.executor.service.CertValidationService; -import com.wso2.openbanking.accelerator.gateway.executor.util.CertificateValidationUtils; -import com.wso2.openbanking.accelerator.gateway.executor.util.TestValidationUtil; -import com.wso2.openbanking.accelerator.gateway.internal.TPPCertValidatorDataHolder; -import org.mockito.Mock; -import org.mockito.Mockito; -import org.mockito.MockitoAnnotations; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.reflect.internal.WhiteboxImpl; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; - -import java.security.cert.CertificateException; -import java.security.cert.X509Certificate; -import java.util.Collections; - -/** - * Test cases for MTLSCertValidationExecutor class. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({CertificateValidationUtils.class, TPPCertValidatorDataHolder.class, - CertValidationService.class, CertificateRevocationCache.class}) -public class CertRevocationValidationExecutorTest { - - CertRevocationValidationExecutor certRevocationValidationExecutor; - @Mock - TPPCertValidatorDataHolder tppCertValidatorDataHolder; - @Mock - CertValidationService certValidationService; - - private X509Certificate testPeerCertificate; - private X509Certificate testPeerCertificateIssuer; - private X509Certificate eidasPeerCertificate; - private X509Certificate expiredPeerCertificate; - - @BeforeClass - public void init() throws CertificateValidationException, CertificateException, OpenBankingException { - MockitoAnnotations.initMocks(this); - this.certRevocationValidationExecutor = new CertRevocationValidationExecutor(); - - this.testPeerCertificate = TestValidationUtil.getTestClientCertificate(); - this.testPeerCertificateIssuer = TestValidationUtil.getTestClientCertificateIssuer(); - this.eidasPeerCertificate = TestValidationUtil.getTestEidasCertificate(); - this.expiredPeerCertificate = TestValidationUtil.getExpiredSelfCertificate(); - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - - - @Test(description = "When expired certificate is provided, then should return true") - public void testIsCertValidWithExpiredCert() { - Assert.assertTrue(CertificateUtils.isExpired(expiredPeerCertificate)); - } - - @Test(description = "When certificate validation success, then should return false") - public void testIsCertRevokedWithNonCachedCert() throws Exception { - CertificateRevocationCache mock = Mockito.mock(CertificateRevocationCache.class); - PowerMockito.mockStatic(CertificateRevocationCache.class); - PowerMockito.when(CertificateRevocationCache.getInstance()) - .thenReturn(mock); - - boolean isCertRevoked = WhiteboxImpl.invokeMethod(this.certRevocationValidationExecutor, - "isCertRevoked", testPeerCertificate); - Assert.assertFalse(isCertRevoked); - } - - @Test(description = "When cached certificate provided, then return false") - public void testIsCertRevokedWithCachedCert() throws Exception { - CertificateRevocationCache mock = Mockito.mock(CertificateRevocationCache.class); - Mockito.doReturn(true).when(mock).getFromCache(Mockito.any(GatewayCacheKey.class)); - - PowerMockito.mockStatic(CertificateRevocationCache.class); - PowerMockito.when(CertificateRevocationCache.getInstance()) - .thenReturn(mock); - - boolean isCertRevoked = WhiteboxImpl.invokeMethod(this.certRevocationValidationExecutor, - "isCertRevoked", testPeerCertificate); - - Assert.assertFalse(isCertRevoked); - } - - @Test(description = "When self signed certificate provided, then should return true") - public void testIsCertRevocationSuccessWithSelfSignedCert() throws Exception { - Mockito.when(tppCertValidatorDataHolder.isCertificateRevocationValidationEnabled()).thenReturn(true); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()) - .thenReturn(tppCertValidatorDataHolder); - - boolean isCertRevocationSuccess = WhiteboxImpl.invokeMethod(this.certRevocationValidationExecutor, - "isCertRevocationSuccess", expiredPeerCertificate); - - Assert.assertTrue(isCertRevocationSuccess); - } - - @Test(description = "When isCertificateRevocationValidationEnabled is false, then should return true") - public void testIsCertRevocationSuccessWithDisabledRevocationValidation() throws Exception { - Mockito.when(tppCertValidatorDataHolder.isCertificateRevocationValidationEnabled()).thenReturn(false); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()) - .thenReturn(tppCertValidatorDataHolder); - - boolean isCertRevocationSuccess = WhiteboxImpl.invokeMethod(this.certRevocationValidationExecutor, - "isCertRevocationSuccess", expiredPeerCertificate); - - Assert.assertTrue(isCertRevocationSuccess); - } - - @Test(description = "When certificate issuer is in excluded list, then should return true") - public void testIsCertRevocationSuccessWithExcludedIssuers() throws Exception { - Mockito.when(tppCertValidatorDataHolder.isCertificateRevocationValidationEnabled()).thenReturn(true); - Mockito.when(tppCertValidatorDataHolder.getCertificateRevocationValidationExcludedIssuers()) - .thenReturn(Collections.singletonList(eidasPeerCertificate.getIssuerDN().getName())); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()) - .thenReturn(tppCertValidatorDataHolder); - - boolean isCertRevocationSuccess = WhiteboxImpl.invokeMethod(this.certRevocationValidationExecutor, - "isCertRevocationSuccess", eidasPeerCertificate); - - Assert.assertTrue(isCertRevocationSuccess); - } - - @Test(description = "When peer certificate is valid, then should return true") - public void testIsCertRevocationSuccessWithValidCerts() throws Exception { - Mockito.when(tppCertValidatorDataHolder.getCertificateRevocationValidationExcludedIssuers()).thenReturn( - Collections.singletonList("")); - Mockito.when(tppCertValidatorDataHolder.isCertificateRevocationValidationEnabled()).thenReturn(true); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito - .when(TPPCertValidatorDataHolder.getInstance()) - .thenReturn(tppCertValidatorDataHolder); - - PowerMockito.mockStatic(CertificateValidationUtils.class); - PowerMockito.when(CertificateValidationUtils.getIssuerCertificateFromTruststore( - Mockito.any(X509Certificate.class))).thenReturn(testPeerCertificateIssuer); - - Mockito.when(certValidationService.verify(Mockito.any(X509Certificate.class), - Mockito.any(X509Certificate.class), Mockito.anyInt(), Mockito.anyInt(), Mockito.anyInt(), - Mockito.anyInt())).thenReturn(true); - - PowerMockito.mockStatic(CertValidationService.class); - PowerMockito - .when(CertValidationService.getInstance()) - .thenReturn(certValidationService); - - boolean isCertRevocationSuccess = WhiteboxImpl.invokeMethod(this.certRevocationValidationExecutor, - "isCertRevocationSuccess", testPeerCertificate); - - Assert.assertTrue(isCertRevocationSuccess); - } - - @Test(description = "When peer certificate is invalid, then should throw CertificateValidationException") - public void testIsCertRevocationSuccessWithInValidCert() throws Exception { - Mockito.when(tppCertValidatorDataHolder.getCertificateRevocationValidationExcludedIssuers()) - .thenReturn(Collections.singletonList("")); - Mockito.when(tppCertValidatorDataHolder.isCertificateRevocationValidationEnabled()).thenReturn(true); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito - .when(TPPCertValidatorDataHolder.getInstance()) - .thenReturn(tppCertValidatorDataHolder); - - Mockito.when(certValidationService.verify(Mockito.any(X509Certificate.class), - Mockito.any(X509Certificate.class), Mockito.anyInt())).thenReturn(true); - - PowerMockito.mockStatic(CertValidationService.class); - PowerMockito - .when(CertValidationService.getInstance()) - .thenReturn(certValidationService); - - boolean isCertRevocationSuccess = WhiteboxImpl.invokeMethod(this.certRevocationValidationExecutor, - "isCertRevocationSuccess", testPeerCertificate); - - Assert.assertFalse(isCertRevocationSuccess); - } - - @Test(description = "When certificate revocation validation not configured, then should return true") - public void testIsCertRevocationSuccessWithFalseCertificateRevocationValidation() throws Exception { - Mockito.when(tppCertValidatorDataHolder.isCertificateRevocationValidationEnabled()).thenReturn(false); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito - .when(TPPCertValidatorDataHolder.getInstance()) - .thenReturn(tppCertValidatorDataHolder); - - boolean isCertRevocationSuccess = WhiteboxImpl.invokeMethod(this.certRevocationValidationExecutor, - "isCertRevocationSuccess", testPeerCertificate); - - Assert.assertTrue(isCertRevocationSuccess); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/mtls/cert/validation/executor/MTLSEnforcementExecutorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/mtls/cert/validation/executor/MTLSEnforcementExecutorTest.java deleted file mode 100644 index e2f92491..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/mtls/cert/validation/executor/MTLSEnforcementExecutorTest.java +++ /dev/null @@ -1,45 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.impl.mtls.cert.validation.executor; - -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import org.mockito.Mockito; -import org.testng.annotations.Test; - -import java.util.ArrayList; -/** - * Test class for MTLSEnforcementExecutor. - */ -public class MTLSEnforcementExecutorTest { - - @Test(description = "When an error occurs, then it should set errors to the obApiRequestContext") - public void testPreProcessRequest() throws Exception { - MTLSEnforcementExecutor mtlsEnforcementExecutor = new MTLSEnforcementExecutor(); - - OBAPIRequestContext obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - Mockito.when(obapiRequestContext.getErrors()).thenReturn(new ArrayList<>()); - - mtlsEnforcementExecutor.preProcessRequest(obapiRequestContext); - - Mockito.verify(obapiRequestContext, Mockito.times(1)).setError(true); - Mockito.verify(obapiRequestContext, Mockito.times(1)).setErrors(Mockito.any()); - - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/selfcare/portal/UserPermissionValidationExecutorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/selfcare/portal/UserPermissionValidationExecutorTest.java deleted file mode 100644 index 425385c7..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/selfcare/portal/UserPermissionValidationExecutorTest.java +++ /dev/null @@ -1,94 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.impl.selfcare.portal; - -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -import java.util.Optional; - -import static org.mockito.Matchers.anyString; -import static org.mockito.Mockito.when; - -/** - * UserPermissionValidationExecutorTest. - *

- * Contains unit tests for UserPermissionValidationExecutor class - */ -@PrepareForTest({GatewayUtils.class}) -@PowerMockIgnore({"jdk.internal.reflect.*"}) -public class UserPermissionValidationExecutorTest { - - private UserPermissionValidationExecutor uut; - - @BeforeClass - public void setup() { - this.uut = new UserPermissionValidationExecutor(); - } - - @Test(description = "when valid url provided, return user IDs") - public void testGetUserIdsFromQueryParamsWithValidUrl() { - PowerMockito.mockStatic(GatewayUtils.class); - when(GatewayUtils.getUserNameWithTenantDomain(anyString())).thenReturn("admin@wso2.com@carbon.super"); - - final String url = "https://localhost:9446/api/consent/admin/search?userIDs=admin@wso2.com&limit=25"; - Optional optUserIds = uut.getUserIdsFromQueryParams(url); - - Assert.assertTrue(optUserIds.isPresent()); - Assert.assertEquals(optUserIds.get(), "admin@wso2.com@carbon.super"); - } - - @Test(description = "when invalid url provided, return empty") - public void testGetUserIdsFromQueryParamsWithInvalidUrl() { - final String url = "https://localhost:9446/api/consent/admin/search?limit=25"; - Optional optUserIds = uut.getUserIdsFromQueryParams(url); - - Assert.assertFalse(optUserIds.isPresent()); - } - - @Test(description = "when valid customer care officer's scopes received from access token, return true") - public void testIsCustomerCareOfficerWithValidScope() { - Assert.assertTrue(uut.isCustomerCareOfficer("consentmgt " + - GatewayConstants.CUSTOMER_CARE_OFFICER_SCOPE + " openid")); - } - - @Test(description = "when invalid scope received from access token, return false") - public void testIsCustomerCareOfficerWithInvalidScope() { - Assert.assertFalse(uut.isCustomerCareOfficer(" ")); - Assert.assertFalse(uut.isCustomerCareOfficer("consentmgt consents:read_self openid")); - } - - @Test(description = "when userId is matching with access token subject, return true") - public void testIsUserIdMatchesTokenSub() { - Assert.assertTrue(uut.isUserIdMatchesTokenSub("amy@gold.com@carbon.super", "amy@gold.com@carbon.super")); - Assert.assertTrue(uut.isUserIdMatchesTokenSub("amy@gold.com", "amy@gold.com")); - - Assert.assertFalse(uut.isUserIdMatchesTokenSub("mark@gold.com", "amy@gold.com")); - Assert.assertFalse(uut.isUserIdMatchesTokenSub("amy@gold.com@carbon.super", "amy@gold.com")); - Assert.assertFalse(uut.isUserIdMatchesTokenSub("amy@gold.com", "amy@gold.com@carbon.super")); - Assert.assertFalse(uut.isUserIdMatchesTokenSub(" ", "amy@gold.com@carbon.super")); - Assert.assertFalse(uut.isUserIdMatchesTokenSub("amy@gold.com", "")); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/tpp/validation/executor/APITPPValidationExecutorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/tpp/validation/executor/APITPPValidationExecutorTest.java deleted file mode 100644 index 4e760c78..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/tpp/validation/executor/APITPPValidationExecutorTest.java +++ /dev/null @@ -1,159 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.impl.tpp.validation.executor; - -import com.wso2.openbanking.accelerator.common.model.PSD2RoleEnum; -import io.swagger.v3.oas.models.Operation; -import io.swagger.v3.oas.models.PathItem; -import io.swagger.v3.oas.models.security.SecurityRequirement; -import org.powermock.reflect.internal.WhiteboxImpl; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -import java.util.Arrays; -import java.util.Collections; -import java.util.HashMap; -import java.util.HashSet; -import java.util.List; -import java.util.Map; -import java.util.Set; - -/** - * Test for API TPP validation executor. - */ -public class APITPPValidationExecutorTest { - - private APITPPValidationExecutor apitppValidationExecutor; - private Map> allowedScopes; - private SecurityRequirement securityRequirement; - - @BeforeClass - public void init() { - this.allowedScopes = new HashMap<>(); - allowedScopes.put("accounts", Arrays.asList("AISP", "PISP")); - allowedScopes.put("payments", Collections.singletonList("PISP")); - - securityRequirement = new SecurityRequirement(); - securityRequirement.put("PSUOAuth2Security", Arrays.asList("accounts", "payments")); - securityRequirement.put("default", Collections.singletonList("accounts")); - - this.apitppValidationExecutor = new APITPPValidationExecutor(); - } - - @Test(description = "when valid scopes provided, then requiredPSD2Roles list should contain roles") - public void testGetRolesFromScopesWithValidScopes() throws Exception { - Set scopes = new HashSet<>(); - scopes.add("accounts"); - scopes.add("payments"); - - List roleList = WhiteboxImpl.invokeMethod(this.apitppValidationExecutor, - "getRolesFromScopes", allowedScopes, scopes); - - Assert.assertTrue(roleList.size() != 0); - Assert.assertTrue(roleList.contains(PSD2RoleEnum.AISP)); - } - - @Test(description = "when invalid scopes provided, then requiredPSD2Roles list should be empty") - public void testGetRolesFromScopesWithInvalidScopes() throws Exception { - Set scopes = new HashSet<>(); - scopes.add("default"); - - List roleList = WhiteboxImpl.invokeMethod(this.apitppValidationExecutor, - "getRolesFromScopes", allowedScopes, scopes); - - Assert.assertEquals(roleList.size(), 0); - } - - @Test(description = "when security requirement provided for GET API, then set of scopes should return") - public void testExtractScopesFromSwaggerAPIWithGet() throws Exception { - Operation get = new Operation(); - get.setSecurity(Collections.singletonList(securityRequirement)); - - PathItem pathItem = new PathItem(); - pathItem.setGet(get); - - Set scopes = WhiteboxImpl.invokeMethod(this.apitppValidationExecutor, - "extractScopesFromSwaggerAPI", pathItem, "GET"); - - Assert.assertTrue(scopes.size() != 0); - Assert.assertTrue(scopes.contains("accounts")); - } - - @Test(description = "when security requirement provided for POST API, then set of scopes should return") - public void testExtractScopesFromSwaggerAPIWithPost() throws Exception { - Operation post = new Operation(); - post.setSecurity(Collections.singletonList(securityRequirement)); - - PathItem pathItem = new PathItem(); - pathItem.setPost(post); - - Set scopes = WhiteboxImpl.invokeMethod(this.apitppValidationExecutor, - "extractScopesFromSwaggerAPI", pathItem, "POST"); - - Assert.assertTrue(scopes.size() != 0); - Assert.assertTrue(scopes.contains("accounts")); - } - - @Test(description = "when security requirement provided for PUT API, then set of scopes should return") - public void testExtractScopesFromSwaggerAPIWithPut() throws Exception { - Operation put = new Operation(); - put.setSecurity(Collections.singletonList(securityRequirement)); - - PathItem pathItem = new PathItem(); - pathItem.setPut(put); - - Set scopes = WhiteboxImpl.invokeMethod(this.apitppValidationExecutor, - "extractScopesFromSwaggerAPI", pathItem, "PUT"); - - Assert.assertTrue(scopes.size() != 0); - Assert.assertTrue(scopes.contains("accounts")); - } - - @Test(description = "when security requirement provided for PATCH API, then set of scopes should return") - public void testExtractScopesFromSwaggerAPIWithPatch() throws Exception { - Operation patch = new Operation(); - patch.setSecurity(Collections.singletonList(securityRequirement)); - - PathItem pathItem = new PathItem(); - pathItem.setPatch(patch); - - Set scopes = WhiteboxImpl.invokeMethod(this.apitppValidationExecutor, - "extractScopesFromSwaggerAPI", pathItem, "PATCH"); - - Assert.assertTrue(scopes.size() != 0); - Assert.assertTrue(scopes.contains("accounts")); - } - - @Test(description = "when security requirement provided for DELETE API, then set of scopes should return") - public void testExtractScopesFromSwaggerAPIWithDelete() throws Exception { - Operation delete = new Operation(); - delete.setSecurity(Collections.singletonList(securityRequirement)); - - PathItem pathItem = new PathItem(); - pathItem.setDelete(delete); - - Set scopes = WhiteboxImpl.invokeMethod(this.apitppValidationExecutor, - "extractScopesFromSwaggerAPI", pathItem, "DELETE"); - - Assert.assertTrue(scopes.size() != 0); - Assert.assertTrue(scopes.contains("accounts")); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/tpp/validation/executor/DCRTPPValidationExecutorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/tpp/validation/executor/DCRTPPValidationExecutorTest.java deleted file mode 100644 index e6c42d31..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/impl/tpp/validation/executor/DCRTPPValidationExecutorTest.java +++ /dev/null @@ -1,90 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.impl.tpp.validation.executor; - -import com.wso2.openbanking.accelerator.common.model.PSD2RoleEnum; -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.gateway.executor.util.TestValidationUtil; -import net.minidev.json.JSONObject; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.reflect.internal.WhiteboxImpl; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; - -import java.util.List; - -/** - * Test for DCR TPP validation executor. - */ -@PowerMockIgnore({"org.mockito.*", "javax.script.*"}) -public class DCRTPPValidationExecutorTest { - - private static final String BODY = "body"; - private static final String SOFTWARE_STATEMENT = "software_statement"; - - private DCRTPPValidationExecutor dcrtppValidationExecutor; - - @BeforeClass - public void init() { - this.dcrtppValidationExecutor = new DCRTPPValidationExecutor(); - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - - @Test(description = "when valid SSA provided, then return extracted software roles as string") - public void testGetRolesFromSsaAsString() throws Exception { - JSONObject requestBody = JWTUtils.decodeRequestJWT(TestValidationUtil.REQUEST_BODY_WITH_SSA, BODY); - String expectedSSA = requestBody.getAsString(SOFTWARE_STATEMENT); - - String actualSSA = WhiteboxImpl.invokeMethod(this.dcrtppValidationExecutor, - "getSSAFromPayload", TestValidationUtil.REQUEST_BODY_WITH_SSA); - - Assert.assertEquals(actualSSA, expectedSSA); - } - - @Test(description = "when valid software statement provided, then requiredPSD2Roles list should return") - public void testGetRolesFromSSAWithValidSSA() throws Exception { - JSONObject requestBody = JWTUtils.decodeRequestJWT(TestValidationUtil.REQUEST_BODY_WITH_SSA, BODY); - // extract software statement - String softwareStatement = requestBody.getAsString(SOFTWARE_STATEMENT); - - List rolesFromSSA = this.dcrtppValidationExecutor.getRolesFromSSA(softwareStatement); - - Assert.assertEquals(rolesFromSSA.size(), 2); - Assert.assertTrue(rolesFromSSA.contains(PSD2RoleEnum.AISP)); - } - - @Test(description = "when invalid software roles provided, then empty requiredPSD2Roles list should return") - public void testGetRolesFromSSAWithInvalidRoles() throws Exception { - - JSONObject requestBody = JWTUtils.decodeRequestJWT(TestValidationUtil.REQUEST_BODY_WITH_SSA_SINGLE_ROLE, BODY); - // extract software statement - String softwareStatement = requestBody.getAsString(SOFTWARE_STATEMENT); - List rolesFromSSA = this.dcrtppValidationExecutor.getRolesFromSSA(softwareStatement); - - Assert.assertTrue(rolesFromSSA.isEmpty()); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/jws/JwsRequestSignatureHandlingExecutorTests.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/jws/JwsRequestSignatureHandlingExecutorTests.java deleted file mode 100644 index 5a13dade..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/jws/JwsRequestSignatureHandlingExecutorTests.java +++ /dev/null @@ -1,299 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.jws; - -import com.nimbusds.jose.JWSAlgorithm; -import com.nimbusds.jose.JWSHeader; -import com.nimbusds.jose.JWSObject; -import com.nimbusds.jose.Payload; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.error.OpenBankingErrorCodes; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.mockito.Mockito; -import org.mockito.MockitoAnnotations; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.reflect.internal.WhiteboxImpl; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; -import org.wso2.carbon.apimgt.common.gateway.dto.MsgInfoDTO; - -import java.lang.reflect.InvocationTargetException; -import java.lang.reflect.Method; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.HashSet; -import java.util.List; -import java.util.Map; - -import javax.ws.rs.HttpMethod; - -import static org.powermock.api.mockito.PowerMockito.mock; - -/** - * Test class for JwsRequestSignatureHandlingExecutor. - */ - -@PowerMockIgnore({"jdk.internal.reflect.*", "javax.management.*"}) -@PrepareForTest({OpenBankingConfigParser.class}) - -public class JwsRequestSignatureHandlingExecutorTests { - - private static final Log log = LogFactory.getLog(JwsRequestSignatureHandlingExecutorTests.class); - - private OBAPIRequestContext obapiRequestContextMock; - - private MsgInfoDTO msgInfoDTO; - - JwsRequestSignatureHandlingExecutor jwsRequestSignatureHandlingExecutor; - - Map sampleRequestHeaders = new HashMap<>(); - - public JwsRequestSignatureHandlingExecutorTests() throws Exception { - sampleRequestHeaders.put("Authorization", "Bearer 2YotnFZFEjr1zCsicMWpAA"); - sampleRequestHeaders.put("x-idempotency-key", "FRESCO.21302.GFX.20"); - sampleRequestHeaders.put("x-jws-signature", "TGlmZSdzIGEgam91cm5leSBub3QgYSBkZXN0aW5hdGlvbiA=" + - "..T2ggZ29vZCBldmVuaW5nIG1yIHR5bGVyIGdvaW5nIGRvd24gPw=="); - sampleRequestHeaders.put("x-fapi-auth-date", "Sun, 10 Sep 2017 19:43:31 GMT"); - sampleRequestHeaders.put("x-fapi-customer-ip-address", "104.25.212.99"); - sampleRequestHeaders.put("x-fapi-interaction-id", "93bac548-d2de-4546-b106-880a5018460d"); - sampleRequestHeaders.put("Content-Type", "text/xml"); - sampleRequestHeaders.put("Accept", "application/json"); - } - - String samplejwsSignature = "V2hhdCB3ZSBnb3QgaGVyZQ0K..aXMgZmFpbHVyZSB0byBjb21tdW5pY2F0ZQ0K"; - - String sampleJWEsignature = "V2hhdCB3ZSBnb3QgaGVyZQ0K.V2hhdCB3ZSBnb3QgaGVyZQ0K." + - "V2hhdCB3ZSBnb3QgaGVyZQ0K.V2hhdCB3ZSBnb3QgaGVyZQ0K.V2hhdCB3ZSBnb3QgaGVyZQ0K"; - - String sampleRequestPayload = "{\"Data\":{\"Initiation\":{\"InstructionIdentification\":" + - "\"ACME412\",\"EndToEndIdentification\":\"FRESCO.21302.GFX.20\",\"InstructedAmount\":" + - "{\"Amount\":\"165.88\",\"Currency\":\"GBP\"},\"CreditorAccount\":" + - "{\"SchemeName\":\"UK.OBIE.SortCodeAccountNumber\",\"Identification\":\"08080021325698\"" + - "\"Name\":\"ACME Inc\",\"SecondaryIdentification\":\"0002\"}," + - "\"RemittanceInformation\":{\"Reference\":\"FRESCO-101\",\"Unstructured\":\"" + - "Internal ops code 5120101\"}}}," + - "\"Risk\":{" + - "{\"PaymentContextCode\":\"EcommerceGoods\",\"MerchantCategoryCode\":\"5967\"," + - "\"MerchantCustomerIdentification\":\"053598653254\",\"DeliveryAddress\":{" + - "\"AddressLine\":[\"Flat 7\",\"Acacia Lodge\"]," + - "\"StreetName\":\"Acacia Avenue\",\"BuildingNumber\":\"27\",\"PostCode\":\"GU31 2ZZ\"" + - "\"TownName\":\"Sparsholt,\"CountrySubDivision\":\"Wessex\",\"Country\":\"UK\"}}}"; - - String requestHeaderb64False = "{\n" + - "\"alg\": \"PS256\",\n" + - "\"kid\": \"12345\",\n" + - "\"b64\": false, \n" + - "\"http://openbanking.org.uk/iat\": 1739485930,\n" + - "\"http://openbanking.org.uk/iss\": \"http://openbanking.org.uk\", \n" + - "\"crit\": [ \"b64\", \"http://openbanking.org.uk/iat\",\n" + - "\"http://openbanking.org.uk/iss\"] \n" + - "}"; - String requestHeaderb64None = "{\n" + - "\"alg\": \"PS256\",\n" + - "\"kid\": \"12345\",\n" + - "\"http://openbanking.org.uk/iat\": 1739485930,\n" + - "\"http://openbanking.org.uk/iss\": \"http://openbanking.org.uk\", \n" + - "\"crit\": [ \"http://openbanking.org.uk/iat\",\n" + - "\"http://openbanking.org.uk/iss\"] \n" + - "}"; - - @BeforeClass - public void initClass() { - - MockitoAnnotations.initMocks(this); - jwsRequestSignatureHandlingExecutor = new JwsRequestSignatureHandlingExecutor(); - } - - /** - * Test a request with an error. - * @throws Exception - */ - @Test - public void testWithErrorRequest() throws Exception { - - // Mocking request headers - obapiRequestContextMock = mock(OBAPIRequestContext.class); - msgInfoDTO = mock(MsgInfoDTO.class); - PowerMockito.when(obapiRequestContextMock.isError()).thenReturn(true); - PowerMockito.when(msgInfoDTO.getHttpMethod()). - thenReturn(HttpMethod.POST); - - boolean isPreProcessValidationPassed = WhiteboxImpl.invokeMethod( - this.jwsRequestSignatureHandlingExecutor, - "preProcessValidation", obapiRequestContextMock, sampleRequestHeaders); - log.debug("Preprocess Validation passed? " + isPreProcessValidationPassed); - - //should return false - Assert.assertFalse(isPreProcessValidationPassed); - } - - /** - * Test if a JWS consisting of 3 parts are returned at reconstructing JWS. - * @throws Exception - */ - @Test - public void testReconstructJWS() throws Exception { - - String reconstructedJWS = WhiteboxImpl.invokeMethod(this.jwsRequestSignatureHandlingExecutor, - "reconstructJws", samplejwsSignature, sampleRequestPayload); - log.debug("The reconstructed JWS, " + reconstructedJWS); - - Assert.assertTrue(reconstructedJWS.split("\\.").length == 3); - } - - /** - * Test reconstructing a JWS with a Payload with empty string. - * @throws Exception - */ - @Test - public void testReconstructJWSException() throws Exception { - - Method method = JwsRequestSignatureHandlingExecutor.class.getDeclaredMethod( - "reconstructJws", String.class, String.class); - method.setAccessible(true); - - Assert.expectThrows(Exception.class, ()-> - method.invoke(this.jwsRequestSignatureHandlingExecutor, samplejwsSignature, "")); - - } - - /** - * Test reconstructing a JWS if the passed value is a JWE. - * @throws Exception - */ - @Test - public void testReconstructJWE() throws Exception { - - Method method = JwsRequestSignatureHandlingExecutor.class.getDeclaredMethod( - "reconstructJws", String.class, String.class); - method.setAccessible(true); - - Assert.expectThrows(Exception.class, ()-> - method.invoke(this.jwsRequestSignatureHandlingExecutor, - sampleJWEsignature, sampleRequestPayload)); - } - - /** - * Test a request with a JWS having b64 header claim set to false. - * @throws Exception - */ - @Test - public void testb64FalseSigningInput() throws Exception { - - JWSAlgorithm signJWSAlg = JWSAlgorithm.parse("PS256"); - HashSet hs = new HashSet(); - hs.add("crit"); - JWSHeader header = new JWSHeader(signJWSAlg, null, null, hs, null, - null, null, null, null, null, "samplekid", - null, null); - byte[] input = WhiteboxImpl.invokeMethod(this.jwsRequestSignatureHandlingExecutor, - "getSigningInput", header, sampleRequestPayload); - Assert.assertNotNull(input); - } - - /** - * Test with a JOSE header b64 claim set to false. - * @throws Exception - */ - @Test - public void testb64Verifiability() throws Exception { - - JWSHeader header = JWSHeader.parse(requestHeaderb64False); - JWSObject jwsObject = new JWSObject(header, new Payload(sampleRequestPayload)); - Method method = JwsRequestSignatureHandlingExecutor.class.getDeclaredMethod( - "isB64HeaderVerifiable", JWSObject.class); - method.setAccessible(true); - boolean isB64Verifiable = (boolean) method.invoke(this.jwsRequestSignatureHandlingExecutor, jwsObject); - - Assert.assertFalse(isB64Verifiable); - } - - /** - * Test a JOSE header with no b64 claim. - * @throws Exception - */ - @Test - public void testb64VerifiabilityWithNoClaim() throws Exception { - - JWSHeader header = JWSHeader.parse(requestHeaderb64None); - JWSObject jwsObject = new JWSObject(header, new Payload(sampleRequestPayload)); - Method method = JwsRequestSignatureHandlingExecutor.class.getDeclaredMethod( - "isB64HeaderVerifiable", JWSObject.class); - method.setAccessible(true); - boolean isB64Verifiable = (boolean) method.invoke(this.jwsRequestSignatureHandlingExecutor, jwsObject); - - Assert.assertTrue(isB64Verifiable); - } - - @Test - public void testHandleRequestInternalServerError() { - - obapiRequestContextMock = mock(OBAPIRequestContext.class); - - GatewayUtils.handleRequestInternalServerError(obapiRequestContextMock, "Error", - OpenBankingErrorCodes.SERVER_ERROR_CODE); - } - - @Test - public void testHandleJwsSignatureErrors() throws NoSuchMethodException, InvocationTargetException, - IllegalAccessException { - - obapiRequestContextMock = mock(OBAPIRequestContext.class); - Method method = JwsRequestSignatureHandlingExecutor.class.getDeclaredMethod( - "handleJwsSignatureErrors", OBAPIRequestContext.class, String.class, String.class); - - method.invoke(this.jwsRequestSignatureHandlingExecutor, obapiRequestContextMock, "Error", - OpenBankingErrorCodes.BAD_REQUEST_CODE); - } - - @Test - public void testValidateClaims() throws NoSuchMethodException, InvocationTargetException, - IllegalAccessException { - - List alg = new ArrayList<>(); - alg.add("PS256"); - OpenBankingConfigParser openBankingConfigParserMock = Mockito.mock(OpenBankingConfigParser.class); - Mockito.doReturn(alg).when(openBankingConfigParserMock).getJwsRequestSigningAlgorithms(); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - - obapiRequestContextMock = mock(OBAPIRequestContext.class); - JWSAlgorithm signJWSAlg = JWSAlgorithm.parse("PS256"); - HashSet hs = new HashSet(); - hs.add("crit"); - JWSHeader header = new JWSHeader(signJWSAlg, null, null, hs, null, - null, null, null, null, null, "samplekid", - null, null); - Method method = JwsRequestSignatureHandlingExecutor.class.getDeclaredMethod( - "validateClaims", OBAPIRequestContext.class, JWSHeader.class, String.class, - String.class); - - boolean result = (boolean) method.invoke(this.jwsRequestSignatureHandlingExecutor, - obapiRequestContextMock, header, "TestApp", null); - - Assert.assertTrue(result); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/jws/JwsResponseSignatureHandlingExecutorTests.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/jws/JwsResponseSignatureHandlingExecutorTests.java deleted file mode 100644 index 6ed4fc82..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/jws/JwsResponseSignatureHandlingExecutorTests.java +++ /dev/null @@ -1,292 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.jws; - -import com.nimbusds.jose.JOSEObjectType; -import com.nimbusds.jose.JWSAlgorithm; -import com.nimbusds.jose.JWSHeader; -import com.nimbusds.jose.JWSObject; -import com.nimbusds.jose.Payload; -import com.nimbusds.jose.util.Base64URL; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import com.wso2.openbanking.accelerator.gateway.util.GatewaySignatureHandlingUtils; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import org.mockito.Mock; -import org.mockito.Mockito; -import org.mockito.MockitoAnnotations; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; -import org.wso2.carbon.apimgt.common.gateway.dto.APIRequestInfoDTO; -import org.wso2.carbon.apimgt.common.gateway.dto.MsgInfoDTO; -import org.wso2.carbon.apimgt.common.gateway.extensionlistener.PayloadHandler; - -import java.io.UnsupportedEncodingException; -import java.text.ParseException; -import java.util.HashMap; -import java.util.Map; -import java.util.Optional; - -/** - * Test class for JwsRequestSignatureHandlingExecutor. - */ - -@PowerMockIgnore({"jdk.internal.reflect.*", "javax.management.*"}) -@PrepareForTest({OpenBankingConfigParser.class}) -public class JwsResponseSignatureHandlingExecutorTests { - - JwsResponseSignatureHandlingExecutor jwsResponseSignatureHandlingExecutor; - @Mock - OBAPIRequestContext obapiRequestContext; - @Mock - OBAPIResponseContext obapiResponseContext; - @Mock - APIRequestInfoDTO apiRequestInfoDTO; - @Mock - MsgInfoDTO msgInfoDTO; - @Mock - PayloadHandler payloadHandler; - Map headers = new HashMap<>(); - - private String kid = "1234"; - - private HashMap criticalParameters = new HashMap<>(); - - private String sampleResponsePayload = "{\n" + - " \"Data\": {\n" + - " \"Initiation\": {\n" + - " \"FileType\": \"UK.OBIE.pain.001.001.08\",\n" + - " \"FileHash\": \"sof6XBU7RAkxekFddW38uJ2h2TBknlgLLiRSCP7qVdw=\",\n" + - " \"FileReference\": \"test\"\n" + - " }\n" + - " }\n" + - "}"; - - private String sampleXmlResponsePayload = "\n" + - " \n" + - " \n" + - " ABC/120928/CCT001\n" + - " 2012-09-28T14:07:00\n" + - " 2\n" + - " 70\n" + - " \n" + - " ABC Corporation\n" + - " \n" + - " Times Square\n" + - " 7\n" + - " NY 10036\n" + - " New York\n" + - " US\n" + - " \n" + - " \n" + - " \n" + - ""; - - private String sampleJWS = "eyJodHRwOlwvXC9vcGVuYmFua2luZy5vwvaXNzIjoiMDAxNTgwMDAwMUhRUXFMyNTYifQ." + - "TQolWv8OZM90Wq6mqL2TZ_Sj6cQjefo5mCgWLP33qg5WH38oFh1YBaBQ7daAFALFIN6jMw." + - "hgdyyUcKNoX8bYZzdQvyYBIMkoyxI39rpYUyumxKQEFbNzysihO_f4js5k4L"; - - String requestHeaderb64False = "{\n" + - "\"alg\": \"PS256\",\n" + - "\"kid\": \"12345\",\n" + - "\"b64\": false, \n" + - "\"http://openbanking.org.uk/iat\": 1739485930,\n" + - "\"http://openbanking.org.uk/iss\": \"http://openbanking.org.uk\", \n" + - "\"crit\": [ \"b64\", \"http://openbanking.org.uk/iat\",\n" + - "\"http://openbanking.org.uk/iss\"] \n" + - "}"; - - String requestHedaerb64True = "{\n" + - "\"alg\": \"PS256\",\n" + - "\"kid\": \"12345\",\n" + - "\"b64\": true, \n" + - "\"http://openbanking.org.uk/iat\": 1739485930,\n" + - "\"http://openbanking.org.uk/iss\": \"http://openbanking.org.uk\", \n" + - "\"crit\": [ \"b64\", \"http://openbanking.org.uk/iat\",\n" + - "\"http://openbanking.org.uk/iss\"] \n" + - "}"; - - String requestHedaerb64None = "{\n" + - "\"alg\": \"PS256\",\n" + - "\"kid\": \"12345\",\n" + - "\"http://openbanking.org.uk/iat\": 1739485930,\n" + - "\"http://openbanking.org.uk/iss\": \"http://openbanking.org.uk\", \n" + - "\"crit\": [ \"http://openbanking.org.uk/iat\",\n" + - "\"http://openbanking.org.uk/iss\"] \n" + - "}"; - - @BeforeClass - public void initClass() { - - MockitoAnnotations.initMocks(this); - - jwsResponseSignatureHandlingExecutor = new JwsResponseSignatureHandlingExecutor(); - - obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - obapiResponseContext = Mockito.mock(OBAPIResponseContext.class); - apiRequestInfoDTO = Mockito.mock(APIRequestInfoDTO.class); - msgInfoDTO = Mockito.mock(MsgInfoDTO.class); - payloadHandler = Mockito.mock(PayloadHandler.class); - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - - /** - * Test the returned JWS Header with the input parameter. - */ - @Test - public void testJWSHeader() { - - criticalParameters.put("iss", "issuer"); - criticalParameters.put("iat", 123456); - criticalParameters.put("tan", "trustAnchor"); - JWSHeader jwsHeader = GatewaySignatureHandlingUtils.constructJWSHeader(kid, criticalParameters, - JWSAlgorithm.parse("ES256")); - Assert.assertTrue("ES256".equals(jwsHeader.getAlgorithm().getName())); - } - - /** - * Test make JWSObject. - */ - @Test - public void testConstructJWSObject() { - - criticalParameters.put("iss", "issuer"); - criticalParameters.put("iat", 123456); - criticalParameters.put("tan", "trustAnchor"); - JWSHeader jwsHeader = new JWSHeader.Builder(JWSAlgorithm.parse("EC")) - .keyID(kid) - .type(JOSEObjectType.JOSE) - .criticalParams(criticalParameters.keySet()) - .customParams(criticalParameters) - .build(); - - JWSObject jwsObject = GatewaySignatureHandlingUtils.constructJWSObject(jwsHeader, sampleResponsePayload); - - Assert.assertTrue("EC".equals(jwsObject.getHeader().getAlgorithm().getName())); - } - - /** - * Test input to sign when b64 claim set to true. - */ - @Test - public void getSigningInput() throws UnsupportedEncodingException { - - criticalParameters.put("iss", "issuer"); - criticalParameters.put("iat", 123456); - criticalParameters.put("tan", "trustAnchor"); - criticalParameters.put("b64", true); - JWSHeader jwsHeader = new JWSHeader.Builder(JWSAlgorithm.parse("EC")) - .keyID(kid) - .type(JOSEObjectType.JOSE) - .criticalParams(criticalParameters.keySet()) - .customParams(criticalParameters) - .build(); - Object signingInput = GatewaySignatureHandlingUtils.getSigningInput(jwsHeader, sampleResponsePayload); - Assert.assertNotNull(signingInput); - } - - /** - * Test a header with b64 claim set to true. - */ - @Test - public void testHeaderWithB64True() throws ParseException { - - JWSHeader header = JWSHeader.parse(requestHedaerb64True); - JWSObject jwsObject = new JWSObject(header, new Payload(sampleResponsePayload)); - boolean isB64Verifiable = GatewaySignatureHandlingUtils.isB64HeaderVerifiable(jwsObject); - - Assert.assertTrue(isB64Verifiable); - } - - /** - * Test a header with b64 claim not set. - */ - @Test - public void testHeaderWithB64NotSet() throws ParseException { - - JWSHeader header = JWSHeader.parse(requestHedaerb64None); - JWSObject jwsObject = new JWSObject(header, new Payload(sampleResponsePayload)); - boolean isB64Verifiable = GatewaySignatureHandlingUtils.isB64HeaderVerifiable(jwsObject); - - Assert.assertTrue(isB64Verifiable); - } - - /** - * Test a header with b64 claim set to false. - */ - @Test - public void testHeaderWithB64False() throws ParseException { - - JWSHeader header = JWSHeader.parse(requestHeaderb64False); - JWSObject jwsObject = new JWSObject(header, new Payload(sampleResponsePayload)); - boolean isB64Verifiable = GatewaySignatureHandlingUtils.isB64HeaderVerifiable(jwsObject); - - - Assert.assertFalse(isB64Verifiable); - } - - /** - * Test creating a detached JWS with serialized JWS. - */ - @Test - public void testDetachedJWS() throws ParseException { - - JWSHeader header = JWSHeader.parse(requestHeaderb64False); - Base64URL signature = Base64URL.encode("signature"); - String detachedJWS = GatewaySignatureHandlingUtils.createDetachedJws(header, signature); - String[] jwsParts = detachedJWS.split("\\."); - - Assert.assertEquals("", jwsParts[1]); - } - - @Test - public void testExtractRequestPayloadForJsonPayloads() throws OpenBankingException { - - headers.put(GatewayConstants.CONTENT_TYPE_TAG, GatewayConstants.JSON_CONTENT_TYPE); - Mockito.doReturn(sampleResponsePayload).when(obapiRequestContext).getRequestPayload(); - - Optional payload = GatewayUtils.extractRequestPayload(obapiRequestContext, headers); - Assert.assertNotNull(payload.get()); - } - - @Test - public void testExtractResponsePayloadForJsonPayloads() throws OpenBankingException { - - headers.put(GatewayConstants.CONTENT_TYPE_TAG, GatewayConstants.JSON_CONTENT_TYPE); - Mockito.doReturn(sampleResponsePayload).when(obapiResponseContext).getResponsePayload(); - - Optional payload = GatewayUtils.extractResponsePayload(obapiResponseContext, headers); - Assert.assertNotNull(payload.get()); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/CRLValidatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/CRLValidatorTest.java deleted file mode 100644 index e3d62725..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/CRLValidatorTest.java +++ /dev/null @@ -1,314 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.revocation; - -import com.wso2.openbanking.accelerator.common.exception.CertificateValidationException; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.HTTPClientUtils; -import com.wso2.openbanking.accelerator.gateway.executor.model.RevocationStatus; -import com.wso2.openbanking.accelerator.gateway.executor.util.CertificateValidationUtils; -import com.wso2.openbanking.accelerator.gateway.executor.util.TestValidationUtil; -import com.wso2.openbanking.accelerator.gateway.internal.TPPCertValidatorDataHolder; -import org.apache.commons.io.FileUtils; -import org.apache.http.HttpEntity; -import org.apache.http.HttpStatus; -import org.apache.http.StatusLine; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpGet; -import org.apache.http.impl.client.CloseableHttpClient; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.powermock.reflect.internal.WhiteboxImpl; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -import java.io.ByteArrayInputStream; -import java.io.File; -import java.io.IOException; -import java.io.InputStream; -import java.net.MalformedURLException; -import java.nio.charset.StandardCharsets; -import java.security.PublicKey; -import java.security.cert.CRLException; -import java.security.cert.CertificateException; -import java.security.cert.X509CRL; -import java.security.cert.X509Certificate; -import java.text.SimpleDateFormat; -import java.util.Collections; -import java.util.Date; - -/** - * Test for CRL validator. - */ -@PrepareForTest({TPPCertValidatorDataHolder.class, HTTPClientUtils.class, CertificateValidationUtils.class}) -@PowerMockIgnore({"javax.security.auth.x500.*", "jdk.internal.reflect.*"}) -public class CRLValidatorTest extends PowerMockTestCase { - - String path = "src/test/resources"; - File file = new File(path); - String absolutePathForTestResources = file.getAbsolutePath(); - - private CRLValidator crlValidator; - private X509Certificate eidasPeerCertificate; - private X509Certificate eidasPeerCertificateIssuer; - private X509Certificate expiredPeerCertificate; - - @BeforeClass - public void initClass() throws CertificateValidationException, CertificateException, OpenBankingException { - this.eidasPeerCertificate = TestValidationUtil.getTestEidasCertificate(); - this.eidasPeerCertificateIssuer = TestValidationUtil.getTestEidasCertificateIssuer(); - this.expiredPeerCertificate = TestValidationUtil.getExpiredSelfCertificate(); - this.crlValidator = new CRLValidator(3); - } - - @BeforeMethod - public void initMethods() throws IOException, OpenBankingException { - StatusLine statusLineMock = Mockito.mock(StatusLine.class); - Mockito.doReturn(HttpStatus.SC_OK).when(statusLineMock).getStatusCode(); - - File file = new File(absolutePathForTestResources + "/test_crl_entries.pem"); - byte[] crlBytes = FileUtils.readFileToString(file, StandardCharsets.UTF_8).getBytes(StandardCharsets.UTF_8); - InputStream inStream = new ByteArrayInputStream(crlBytes); - - HttpEntity httpEntityMock = Mockito.mock(HttpEntity.class); - Mockito.doReturn(inStream).when(httpEntityMock).getContent(); - - CloseableHttpResponse httpResponseMock = Mockito.mock(CloseableHttpResponse.class); - Mockito.doReturn(statusLineMock).when(httpResponseMock).getStatusLine(); - Mockito.doReturn(httpEntityMock).when(httpResponseMock).getEntity(); - - CloseableHttpClient closeableHttpClientMock = Mockito.mock(CloseableHttpClient.class); - Mockito.doReturn(httpResponseMock).when(closeableHttpClientMock).execute(Mockito.any(HttpGet.class)); - - PowerMockito.mockStatic(HTTPClientUtils.class); - PowerMockito.when(HTTPClientUtils.getHttpsClient()).thenReturn(closeableHttpClientMock); - } - - @Test - public void testCRLValidatorConstructor() { - Assert.assertSame(new CRLValidator(3).getRetryCount(), 3); - } - - @Test(description = "when valid certificate provided, then X509URL should not be null") - public void testDownloadCRLFromWeb() throws Exception { - TPPCertValidatorDataHolder tppCertValidatorDataHolder = Mockito.mock(TPPCertValidatorDataHolder.class); - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()).thenReturn(tppCertValidatorDataHolder); - - // Date needs to be an old date than X509 next update date - Date dateMock = new SimpleDateFormat("dd/MM/yyyy").parse("17/03/2021"); - PowerMockito.mockStatic(CertificateValidationUtils.class); - PowerMockito.when(CertificateValidationUtils.getNewDate()).thenReturn(dateMock); - - Assert.assertNotNull(this.crlValidator.checkRevocationStatus(eidasPeerCertificate, eidasPeerCertificateIssuer)); - - } - - @Test(description = "when valid proxy provided, then X509URL should not be null") - public void testDownloadCRLFromWebWithProxy() throws Exception { - TPPCertValidatorDataHolder tppCertValidatorDataHolder = Mockito.mock(TPPCertValidatorDataHolder.class); - Mockito.doReturn(true).when(tppCertValidatorDataHolder).isCertificateRevocationProxyEnabled(); - Mockito.doReturn("localhost").when(tppCertValidatorDataHolder).getCertificateRevocationProxyHost(); - Mockito.doReturn(8080).when(tppCertValidatorDataHolder).getCertificateRevocationProxyPort(); - - // Date needs to be an old date than X509 next update date - Date dateMock = new SimpleDateFormat("dd/MM/yyyy").parse("17/03/2021"); - PowerMockito.mockStatic(CertificateValidationUtils.class); - PowerMockito.when(CertificateValidationUtils.getNewDate()).thenReturn(dateMock); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()).thenReturn(tppCertValidatorDataHolder); - - Assert.assertNotNull(this.crlValidator.checkRevocationStatus(eidasPeerCertificate, eidasPeerCertificateIssuer)); - } - - @Test(description = "when invalid proxy host provided, then throw CertificateValidationException", - expectedExceptions = CertificateValidationException.class) - public void testDownloadCRLFromWebWithInvalidProxy() throws CertificateValidationException { - TPPCertValidatorDataHolder tppCertValidatorDataHolder = Mockito.mock(TPPCertValidatorDataHolder.class); - Mockito.doReturn(true).when(tppCertValidatorDataHolder).isCertificateRevocationProxyEnabled(); - Mockito.doReturn(" ").when(tppCertValidatorDataHolder).getCertificateRevocationProxyHost(); - Mockito.doReturn(8080).when(tppCertValidatorDataHolder).getCertificateRevocationProxyPort(); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()).thenReturn(tppCertValidatorDataHolder); - - Assert.assertNotNull(this.crlValidator.checkRevocationStatus(eidasPeerCertificate, eidasPeerCertificateIssuer)); - } - - @Test(description = "when invalid http response received, then throw CertificateValidationException", - expectedExceptions = CertificateValidationException.class) - public void testDownloadCRLFromWebWithInvalidHTTPResponse() throws CertificateValidationException, IOException, - OpenBankingException { - StatusLine statusLineMock = Mockito.mock(StatusLine.class); - Mockito.doReturn(HttpStatus.SC_BAD_REQUEST).when(statusLineMock).getStatusCode(); - - CloseableHttpResponse httpResponseMock = Mockito.mock(CloseableHttpResponse.class); - Mockito.doReturn(statusLineMock).when(httpResponseMock).getStatusLine(); - - CloseableHttpClient closeableHttpClientMock = Mockito.mock(CloseableHttpClient.class); - Mockito.doReturn(httpResponseMock).when(closeableHttpClientMock).execute(Mockito.any(HttpGet.class)); - - PowerMockito.mockStatic(HTTPClientUtils.class); - PowerMockito.when(HTTPClientUtils.getHttpsClient()).thenReturn(closeableHttpClientMock); - - TPPCertValidatorDataHolder tppCertValidatorDataHolder = Mockito.mock(TPPCertValidatorDataHolder.class); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()).thenReturn(tppCertValidatorDataHolder); - - Assert.assertNotNull(this.crlValidator.checkRevocationStatus(eidasPeerCertificate, eidasPeerCertificateIssuer)); - } - - @Test(description = "when invalid http response received, then throw CertificateValidationException", - expectedExceptions = CertificateValidationException.class) - public void testDownloadCRLFromWebWhenThrowingIOException() throws Exception { - CloseableHttpClient closeableHttpClientMock = Mockito.mock(CloseableHttpClient.class); - Mockito.doThrow(IOException.class).when(closeableHttpClientMock).execute(Mockito.any(HttpGet.class)); - - PowerMockito.mockStatic(HTTPClientUtils.class); - PowerMockito.when(HTTPClientUtils.getHttpsClient()).thenReturn(closeableHttpClientMock); - - TPPCertValidatorDataHolder tppCertValidatorDataHolder = Mockito.mock(TPPCertValidatorDataHolder.class); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()).thenReturn(tppCertValidatorDataHolder); - - Assert.assertNotNull(this.crlValidator.checkRevocationStatus(eidasPeerCertificate, eidasPeerCertificateIssuer)); - } - - @Test(description = "when invalid http response received, then throw CertificateValidationException", - expectedExceptions = CertificateValidationException.class) - public void testDownloadCRLFromWebWhenThrowingCertificateException() throws Exception { - CloseableHttpClient closeableHttpClientMock = Mockito.mock(CloseableHttpClient.class); - Mockito.doThrow(CertificateException.class).when(closeableHttpClientMock).execute(Mockito.any(HttpGet.class)); - - PowerMockito.mockStatic(HTTPClientUtils.class); - PowerMockito.when(HTTPClientUtils.getHttpsClient()).thenReturn(closeableHttpClientMock); - - TPPCertValidatorDataHolder tppCertValidatorDataHolder = Mockito.mock(TPPCertValidatorDataHolder.class); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()).thenReturn(tppCertValidatorDataHolder); - - Assert.assertNotNull(this.crlValidator.checkRevocationStatus(eidasPeerCertificate, eidasPeerCertificateIssuer)); - } - - @Test(description = "when invalid http response received, then throw CertificateValidationException", - expectedExceptions = CertificateValidationException.class) - public void testDownloadCRLFromWebWhenThrowingCRLException() throws Exception { - CloseableHttpClient closeableHttpClientMock = Mockito.mock(CloseableHttpClient.class); - Mockito.doThrow(CRLException.class).when(closeableHttpClientMock).execute(Mockito.any(HttpGet.class)); - - PowerMockito.mockStatic(HTTPClientUtils.class); - PowerMockito.when(HTTPClientUtils.getHttpsClient()).thenReturn(closeableHttpClientMock); - - TPPCertValidatorDataHolder tppCertValidatorDataHolder = Mockito.mock(TPPCertValidatorDataHolder.class); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()).thenReturn(tppCertValidatorDataHolder); - - Assert.assertNotNull(this.crlValidator.checkRevocationStatus(eidasPeerCertificate, eidasPeerCertificateIssuer)); - } - - @Test(description = "when invalid http response received, then throw CertificateValidationException", - expectedExceptions = CertificateValidationException.class) - public void testDownloadCRLFromWebWhenThrowingMalformedURLException() throws Exception { - CloseableHttpClient closeableHttpClientMock = Mockito.mock(CloseableHttpClient.class); - Mockito.doThrow(MalformedURLException.class).when(closeableHttpClientMock).execute(Mockito.any(HttpGet.class)); - - PowerMockito.mockStatic(HTTPClientUtils.class); - PowerMockito.when(HTTPClientUtils.getHttpsClient()).thenReturn(closeableHttpClientMock); - - TPPCertValidatorDataHolder tppCertValidatorDataHolder = Mockito.mock(TPPCertValidatorDataHolder.class); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()).thenReturn(tppCertValidatorDataHolder); - - Assert.assertNotNull(this.crlValidator.checkRevocationStatus(eidasPeerCertificate, eidasPeerCertificateIssuer)); - } - - @Test(description = "when invalid X509URL, then throw CertificateValidationException", - expectedExceptions = CertificateValidationException.class) - public void testIsValidX509CRLFromIssuer() throws Exception { - X509CRL x509CRLMock = Mockito.mock(X509CRL.class); - WhiteboxImpl.invokeMethod(this.crlValidator, "isValidX509Crl", x509CRLMock, - eidasPeerCertificate, eidasPeerCertificateIssuer); - } - - @Test(description = "when X509URL next update date is invalid, then throw CertificateValidationException", - expectedExceptions = CertificateValidationException.class) - public void testIsValidX509CRLFromNextUpdate() throws Exception { - X509CRL x509CRLMock = Mockito.mock(X509CRL.class); - final Date today = new Date(); - final Date yesterday = new Date(today.getTime() - (1000 * 60 * 60 * 24)); - Mockito.doReturn(yesterday).when(x509CRLMock).getNextUpdate(); - - WhiteboxImpl.invokeMethod(this.crlValidator, "isValidX509CRLFromNextUpdate", x509CRLMock, - today, today); - } - - @Test(description = "when X509URL next update date is null, then return false") - public void testIsValidX509CRLFromNextUpdateWithNullDate() throws Exception { - X509CRL x509CRLMock = Mockito.mock(X509CRL.class); - boolean result = WhiteboxImpl.invokeMethod(this.crlValidator, - "isValidX509CRLFromNextUpdate", x509CRLMock, null, null); - - Assert.assertFalse(result); - } - - @Test(description = "when X509URL verification failed, then throw CertificateValidationException", - expectedExceptions = CertificateValidationException.class) - public void testIsValidX509CRLFromIssuerWithFailedVerification() throws Exception { - X509CRL x509CRLMock = Mockito.mock(X509CRL.class); - Mockito.doReturn(eidasPeerCertificate.getIssuerDN()).when(x509CRLMock).getIssuerDN(); - Mockito.doThrow(CRLException.class).when(x509CRLMock).verify(Mockito.any(PublicKey.class)); - - WhiteboxImpl.invokeMethod(this.crlValidator, "isValidX509CRLFromIssuer", x509CRLMock, - eidasPeerCertificate, eidasPeerCertificateIssuer); - } - - @Test(description = "when CRL URL list is empty, then throw CertificateValidationException", - expectedExceptions = CertificateValidationException.class) - public void testGetCRLRevocationStatusWithEmptyCRLUrls() throws CertificateValidationException { - CRLValidator.getCRLRevocationStatus(null, null, 0, Collections.emptyList(), false, "", 0); - } - - @Test(description = "when invalid cert provided, then throw CertificateValidationException", - expectedExceptions = CertificateValidationException.class) - public void testGetCRLUrlsWithInvalidCert() throws CertificateValidationException { - CRLValidator.getCRLUrls(expiredPeerCertificate); - } - - @Test(description = "when certificate is revoked, then return revoked revocation status") - public void testGetRevocationStatusFromCRLWithRevokedCert() throws Exception { - X509CRL x509CRLMock = Mockito.mock(X509CRL.class); - Mockito.doReturn(true).when(x509CRLMock).isRevoked(eidasPeerCertificate); - - RevocationStatus actual = WhiteboxImpl.invokeMethod(this.crlValidator, "getRevocationStatusFromCRL", - x509CRLMock, eidasPeerCertificate); - - Assert.assertSame(actual, RevocationStatus.REVOKED); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/OCSPValidatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/OCSPValidatorTest.java deleted file mode 100644 index 21d5a7f7..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/revocation/OCSPValidatorTest.java +++ /dev/null @@ -1,224 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.revocation; - -import com.wso2.openbanking.accelerator.common.exception.CertificateValidationException; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.HTTPClientUtils; -import com.wso2.openbanking.accelerator.gateway.executor.model.RevocationStatus; -import com.wso2.openbanking.accelerator.gateway.executor.util.CertificateValidationUtils; -import com.wso2.openbanking.accelerator.gateway.executor.util.TestValidationUtil; -import com.wso2.openbanking.accelerator.gateway.internal.TPPCertValidatorDataHolder; -import org.apache.http.HttpStatus; -import org.apache.http.StatusLine; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpPost; -import org.apache.http.impl.client.CloseableHttpClient; -import org.bouncycastle.cert.ocsp.CertificateStatus; -import org.bouncycastle.cert.ocsp.OCSPReq; -import org.bouncycastle.cert.ocsp.RevokedStatus; -import org.bouncycastle.cert.ocsp.SingleResp; -import org.bouncycastle.cert.ocsp.UnknownStatus; -import org.mockito.Mock; -import org.mockito.Mockito; -import org.mockito.MockitoAnnotations; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.powermock.reflect.internal.WhiteboxImpl; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -import java.io.IOException; -import java.nio.charset.StandardCharsets; -import java.security.cert.CertificateException; -import java.security.cert.X509Certificate; -import java.util.List; - -/** - * Test for OCSP validator. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({TPPCertValidatorDataHolder.class, HTTPClientUtils.class}) -public class OCSPValidatorTest extends PowerMockTestCase { - - @Mock - TPPCertValidatorDataHolder tppCertValidatorDataHolder; - OCSPValidator ocspValidator; - - private X509Certificate testPeerCertificateIssuer; - private X509Certificate eidasPeerCertificate; - private X509Certificate eidasPeerCertificateIssuer; - private X509Certificate expiredPeerCertificate; - - @BeforeClass - public void init() throws CertificateValidationException, CertificateException, OpenBankingException { - MockitoAnnotations.initMocks(this); - this.ocspValidator = new OCSPValidator(1); - this.testPeerCertificateIssuer = TestValidationUtil.getTestClientCertificateIssuer(); - this.eidasPeerCertificate = TestValidationUtil.getTestEidasCertificate(); - this.eidasPeerCertificateIssuer = TestValidationUtil.getTestEidasCertificateIssuer(); - this.expiredPeerCertificate = TestValidationUtil.getExpiredSelfCertificate(); - } - - @Test(description = "when valid certificate provided, then list of crl urls should return") - public void testGetAIALocationsWithValidCert() throws CertificateValidationException { - List crlUrls = OCSPValidator.getAIALocations(eidasPeerCertificate); - Assert.assertFalse(crlUrls.isEmpty()); - } - - @Test(description = "when invalid certificate provided, then CertificateValidationException should throw", - expectedExceptions = CertificateValidationException.class) - public void testGetAIALocationsWithInvalidCert() throws CertificateValidationException { - OCSPValidator.getAIALocations(expiredPeerCertificate); - } - - @Test(description = "when valid certificate provided, then OCSP object should return") - public void testGenerateOCSPRequestWithValidCert() throws Exception { - OCSPReq ocspRequest = WhiteboxImpl.invokeMethod(this.ocspValidator, - "generateOCSPRequest", testPeerCertificateIssuer, expiredPeerCertificate.getSerialNumber()); - Assert.assertNotNull(ocspRequest); - } - - @Test - public void testSetRequestProperties() throws Exception { - byte[] bytes = "test msg".getBytes(StandardCharsets.UTF_8); - HttpPost httpPost = new HttpPost(); - - WhiteboxImpl.invokeMethod(this.ocspValidator, - "setRequestProperties", bytes, httpPost); - - Assert.assertTrue(httpPost.containsHeader(CertificateValidationUtils.HTTP_CONTENT_TYPE)); - Assert.assertTrue(httpPost.containsHeader(CertificateValidationUtils.HTTP_ACCEPT)); - Assert.assertEquals(httpPost.getEntity().getContentType().getValue(), CertificateValidationUtils.CONTENT_TYPE); - } - - @Test(description = "when invalid proxy provided, then throw CertificateValidationException", - expectedExceptions = CertificateValidationException.class) - public void testCheckRevocationStatusWithInvalidProxy() throws Exception { - CloseableHttpClient closeableHttpClientMock = Mockito.mock(CloseableHttpClient.class); - - PowerMockito.mockStatic(HTTPClientUtils.class); - PowerMockito.when(HTTPClientUtils.getHttpsClient()).thenReturn(closeableHttpClientMock); - - Mockito.when(tppCertValidatorDataHolder.isCertificateRevocationProxyEnabled()).thenReturn(true); - Mockito.when(tppCertValidatorDataHolder.getCertificateRevocationProxyHost()).thenReturn(" "); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()) - .thenReturn(tppCertValidatorDataHolder); - - this.ocspValidator - .checkRevocationStatus(eidasPeerCertificate, eidasPeerCertificateIssuer); - } - - @Test(description = "when issuer cert is null, then throw CertificateValidationException", - expectedExceptions = CertificateValidationException.class) - public void testCheckRevocationStatusWithNullCerts() throws CertificateValidationException { - this.ocspValidator.checkRevocationStatus(null, null); - } - - @Test(description = "when invalid response received, then throw CertificateValidationException", - expectedExceptions = CertificateValidationException.class) - public void testCheckRevocationStatusWithInvalidResponse() throws CertificateValidationException, IOException, - OpenBankingException { - StatusLine statusLineMock = Mockito.mock(StatusLine.class); - Mockito.doReturn(HttpStatus.SC_BAD_REQUEST).when(statusLineMock).getStatusCode(); - - CloseableHttpClient closeableHttpClientMock = Mockito.mock(CloseableHttpClient.class); - CloseableHttpResponse httpResponseMock = Mockito.mock(CloseableHttpResponse.class); - Mockito.doReturn(statusLineMock).when(httpResponseMock).getStatusLine(); - - Mockito.doReturn(httpResponseMock).when(closeableHttpClientMock).execute(Mockito.any(HttpPost.class)); - - PowerMockito.mockStatic(HTTPClientUtils.class); - PowerMockito.when(HTTPClientUtils.getHttpsClient()).thenReturn(closeableHttpClientMock); - - Mockito.doReturn(true).when(tppCertValidatorDataHolder).isCertificateRevocationProxyEnabled(); - Mockito.doReturn("localhost").when(tppCertValidatorDataHolder).getCertificateRevocationProxyHost(); - Mockito.doReturn(8080).when(tppCertValidatorDataHolder).getCertificateRevocationProxyPort(); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()).thenReturn(tppCertValidatorDataHolder); - - this.ocspValidator.checkRevocationStatus(eidasPeerCertificate, eidasPeerCertificateIssuer); - } - - @Test(description = "when invalid response received, then throw CertificateValidationException", - expectedExceptions = CertificateValidationException.class) - public void testCheckRevocationStatusWhenThrowingIOException() throws CertificateValidationException, IOException, - OpenBankingException { - CloseableHttpClient closeableHttpClientMock = Mockito.mock(CloseableHttpClient.class); - - Mockito.doThrow(IOException.class).when(closeableHttpClientMock).execute(Mockito.any(HttpPost.class)); - - PowerMockito.mockStatic(HTTPClientUtils.class); - PowerMockito.when(HTTPClientUtils.getHttpsClient()).thenReturn(closeableHttpClientMock); - - TPPCertValidatorDataHolder tppCertValidatorDataHolderMock = Mockito.mock(TPPCertValidatorDataHolder.class); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()).thenReturn(tppCertValidatorDataHolderMock); - - this.ocspValidator.checkRevocationStatus(eidasPeerCertificate, eidasPeerCertificateIssuer); - } - - @Test - public void testOCSPValidatorConstructor() { - Assert.assertSame(new OCSPValidator(3).getRetryCount(), 3); - } - - - @Test(description = "when certificate is good, then return good revocation status") - public void testGetRevocationStatusFromOCSGood() throws Exception { - SingleResp singleRespMock = Mockito.mock(SingleResp.class); - Mockito.doReturn(CertificateStatus.GOOD).when(singleRespMock).getCertStatus(); - - RevocationStatus revocationStatus = WhiteboxImpl.invokeMethod(this.ocspValidator, - "getRevocationStatusFromOCSP", singleRespMock); - - Assert.assertSame(revocationStatus, RevocationStatus.GOOD); - } - - @Test(description = "when certificate is revoked, then return revoked revocation status") - public void testGetRevocationStatusFromOCSRevoked() throws Exception { - SingleResp singleRespMock = Mockito.mock(SingleResp.class); - RevokedStatus revokedStatusMock = Mockito.mock(RevokedStatus.class); - Mockito.doReturn(revokedStatusMock).when(singleRespMock).getCertStatus(); - - RevocationStatus revocationStatus = WhiteboxImpl.invokeMethod(this.ocspValidator, - "getRevocationStatusFromOCSP", singleRespMock); - - Assert.assertSame(revocationStatus, RevocationStatus.REVOKED); - } - - @Test(description = "when certificate status is unknown, then return unknown revocation status") - public void testGetRevocationStatusFromOCSUnknown() throws Exception { - SingleResp singleRespMock = Mockito.mock(SingleResp.class); - UnknownStatus unknownStatusMock = Mockito.mock(UnknownStatus.class); - Mockito.doReturn(unknownStatusMock).when(singleRespMock).getCertStatus(); - - RevocationStatus revocationStatus = WhiteboxImpl.invokeMethod(this.ocspValidator, - "getRevocationStatusFromOCSP", singleRespMock); - - Assert.assertSame(revocationStatus, RevocationStatus.UNKNOWN); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/service/CertValidationServiceTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/service/CertValidationServiceTest.java deleted file mode 100644 index dfe30851..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/service/CertValidationServiceTest.java +++ /dev/null @@ -1,351 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.service; - -import com.wso2.openbanking.accelerator.common.caching.OpenBankingBaseCache; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.CertificateValidationException; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.exception.TPPValidationException; -import com.wso2.openbanking.accelerator.common.model.PSD2RoleEnum; -import com.wso2.openbanking.accelerator.gateway.cache.GatewayCacheKey; -import com.wso2.openbanking.accelerator.gateway.cache.TppValidationCache; -import com.wso2.openbanking.accelerator.gateway.executor.model.RevocationStatus; -import com.wso2.openbanking.accelerator.gateway.executor.revocation.OCSPValidator; -import com.wso2.openbanking.accelerator.gateway.executor.util.TestValidationUtil; -import com.wso2.openbanking.accelerator.gateway.internal.TPPCertValidatorDataHolder; -import org.mockito.Mock; -import org.mockito.Mockito; -import org.mockito.MockitoAnnotations; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.reflect.internal.WhiteboxImpl; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; - -import java.security.cert.CertificateException; -import java.security.cert.X509Certificate; -import java.util.Arrays; -import java.util.Collections; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -/** - * Test for certificate validation service. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({OpenBankingConfigParser.class, OCSPValidator.class, TPPCertValidatorDataHolder.class, - TppValidationCache.class}) -public class CertValidationServiceTest { - - @Mock - OpenBankingConfigParser openBankingConfigParser; - @Mock - TPPCertValidatorDataHolder tppCertValidatorDataHolder; - @Mock - TppValidationCache tppValidationCache; - CertValidationService certValidationService; - private X509Certificate testPeerCertificate; - private X509Certificate testPeerCertificateIssuer; - private X509Certificate eidasPeerCertificate; - - @BeforeClass - public void init() throws CertificateValidationException, CertificateException, OpenBankingException { - MockitoAnnotations.initMocks(this); - certValidationService = CertValidationService.getInstance(); - this.testPeerCertificate = TestValidationUtil.getTestClientCertificate(); - this.testPeerCertificateIssuer = TestValidationUtil.getTestClientCertificateIssuer(); - this.eidasPeerCertificate = TestValidationUtil.getTestEidasCertificate(); - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - - @Test(description = "when valid certificate provided, then should return true") - public void testVerifyWithValidCertificate() throws Exception { - Map validators = new HashMap<>(); - validators.put(1, "OCSP"); - - Mockito.when(openBankingConfigParser.getCertificateRevocationValidators()).thenReturn(validators); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()) - .thenReturn(openBankingConfigParser); - - PowerMockito.mockStatic(OCSPValidator.class); - PowerMockito.when(OCSPValidator.getOCSPRevocationStatus(Mockito.any(X509Certificate.class), - Mockito.any(X509Certificate.class), Mockito.anyInt(), Mockito.anyListOf(String.class), - Mockito.anyBoolean(), Mockito.anyString(), Mockito.anyInt())) - .thenReturn(RevocationStatus.GOOD); - - boolean isVerified = this.certValidationService - .verify(testPeerCertificate, testPeerCertificateIssuer, 1); - Assert.assertTrue(isVerified); - } - - @Test(description = "when valid certificate provided, then should return true") - public void testUpdatedVerifyWithValidCertificate() throws Exception { - Map validators = new HashMap<>(); - validators.put(1, "OCSP"); - - Mockito.when(openBankingConfigParser.getCertificateRevocationValidators()).thenReturn(validators); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()) - .thenReturn(openBankingConfigParser); - - PowerMockito.mockStatic(OCSPValidator.class); - PowerMockito.when(OCSPValidator.getOCSPRevocationStatus(Mockito.any(X509Certificate.class), - Mockito.any(X509Certificate.class), Mockito.anyInt(), Mockito.anyListOf(String.class), - Mockito.anyBoolean(), Mockito.anyString(), Mockito.anyInt())) - .thenReturn(RevocationStatus.GOOD); - - boolean isVerified = this.certValidationService - .verify(testPeerCertificate, testPeerCertificateIssuer, 1, 5000, - 5000, 5000); - Assert.assertTrue(isVerified); - } - - @Test(description = "when invalid certificate provided, then should return false") - public void testVerifyWithValidInvalidCertificate() throws Exception { - Map validators = new HashMap<>(); - validators.put(1, "OCSP"); - - Mockito.when(openBankingConfigParser.getCertificateRevocationValidators()).thenReturn(validators); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()) - .thenReturn(openBankingConfigParser); - - PowerMockito.mockStatic(OCSPValidator.class); - PowerMockito.when(OCSPValidator.getOCSPRevocationStatus(Mockito.any(X509Certificate.class), - Mockito.any(X509Certificate.class), Mockito.anyInt(), Mockito.anyListOf(String.class), - Mockito.anyBoolean(), Mockito.anyString(), Mockito.anyInt())) - .thenReturn(RevocationStatus.REVOKED); - - boolean isVerified = this.certValidationService - .verify(testPeerCertificate, testPeerCertificateIssuer, 1); - Assert.assertFalse(isVerified); - - boolean isVerifiedUpdated = this.certValidationService - .verify(testPeerCertificate, testPeerCertificateIssuer, 1, - 5000, 5000, 5000); - Assert.assertFalse(isVerifiedUpdated); - } - - @Test(description = "when invalid validator configured, then should return false") - public void testVerifyWithInvalidValidator() throws Exception { - Map validators = new HashMap<>(); - validators.put(1, "INVALID VALIDATOR"); - - Mockito.when(openBankingConfigParser.getCertificateRevocationValidators()).thenReturn(validators); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()) - .thenReturn(openBankingConfigParser); - - boolean isVerified = this.certValidationService - .verify(testPeerCertificate, testPeerCertificateIssuer, 1); - Assert.assertFalse(isVerified); - - boolean isVerifiedUpdated = this.certValidationService - .verify(testPeerCertificate, testPeerCertificateIssuer, 1, - 5000, 5000, 5000); - Assert.assertFalse(isVerifiedUpdated); - } - - @Test(description = "when cert roles and scope roles are equal, then return true") - public void testIsRequiredRolesMatchWithValidScopes() throws Exception { - boolean isValid = WhiteboxImpl.invokeMethod(this.certValidationService, - "isRequiredRolesMatchWithScopes", eidasPeerCertificate, - Arrays.asList(PSD2RoleEnum.AISP, PSD2RoleEnum.PISP)); - - Assert.assertTrue(isValid); - } - - @Test(description = "when cert roles and scope roles are not equal, then throw TPPValidationException", - expectedExceptions = {TPPValidationException.class}) - public void testIsRequiredRolesMatchWithInvalidScopes() throws Exception { - WhiteboxImpl.invokeMethod(this.certValidationService, - "isRequiredRolesMatchWithScopes", eidasPeerCertificate, - Arrays.asList(PSD2RoleEnum.AISP, PSD2RoleEnum.PISP, PSD2RoleEnum.ASPSP)); - } - - @Test(description = "When tpp certificate and roles are valid, then should return true") - public void testValidateTPPWithValidRoles() throws TPPValidationException, - CertificateValidationException, OpenBankingException { - Mockito.when(tppCertValidatorDataHolder.isTppValidationEnabled()).thenReturn(true); - Mockito.when(tppCertValidatorDataHolder.getTPPValidationServiceImpl()).thenReturn("/dummy/path"); - - TPPValidationService mockTppValidationService = Mockito.mock(TPPValidationService.class); - Mockito.when(mockTppValidationService.getCacheKey(Mockito.any(X509Certificate.class), - Mockito.anyListOf(PSD2RoleEnum.class), Mockito.anyMapOf(String.class, Object.class))) - .thenReturn("dummy-cache-key"); - Mockito.when(mockTppValidationService.validate(Mockito.any(X509Certificate.class), - Mockito.anyListOf(PSD2RoleEnum.class), Mockito.anyMapOf(String.class, Object.class))).thenReturn(true); - - Mockito.when(tppCertValidatorDataHolder.getTppValidationService()).thenReturn(mockTppValidationService); - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()) - .thenReturn(tppCertValidatorDataHolder); - - PowerMockito.mockStatic(TppValidationCache.class); - PowerMockito.when(TppValidationCache.getInstance()) - .thenReturn(tppValidationCache); - - Assert.assertTrue(certValidationService.validateTppRoles(eidasPeerCertificate, - Arrays.asList(PSD2RoleEnum.AISP, PSD2RoleEnum.PISP))); - } - - @Test(description = "When role is failing, then should return false ") - public void testValidateTPPWithValidationError() throws TPPValidationException, - CertificateValidationException, OpenBankingException { - Mockito.when(tppCertValidatorDataHolder.isTppValidationEnabled()).thenReturn(true); - Mockito.when(tppCertValidatorDataHolder.getTPPValidationServiceImpl()).thenReturn("/dummy/path"); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()) - .thenReturn(openBankingConfigParser); - - TPPValidationService mockTppValidationService = Mockito.mock(TPPValidationService.class); - Mockito.when(mockTppValidationService.getCacheKey(Mockito.any(X509Certificate.class), - Mockito.anyListOf(PSD2RoleEnum.class), Mockito.anyMapOf(String.class, Object.class))) - .thenReturn("dummy-cache-key"); - - Mockito.when(tppCertValidatorDataHolder.getTppValidationService()).thenReturn(mockTppValidationService); - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()) - .thenReturn(tppCertValidatorDataHolder); - - Mockito.when(tppValidationCache.getFromCacheOrRetrieve(Mockito.any(GatewayCacheKey.class), - Mockito.any(OpenBankingBaseCache.OnDemandRetriever.class))).thenThrow(OpenBankingException.class); - PowerMockito.mockStatic(TppValidationCache.class); - PowerMockito.when(TppValidationCache.getInstance()) - .thenReturn(tppValidationCache); - - Assert.assertFalse(certValidationService.validateTppRoles(eidasPeerCertificate, - Arrays.asList(PSD2RoleEnum.AISP, PSD2RoleEnum.PISP))); - } - - @Test(description = "When TPPValidationImpl path configuration is empty, then should throw TPPValidationException", - expectedExceptions = {TPPValidationException.class}) - public void testValidateTPPWithEmptyImplConfig() throws TPPValidationException, CertificateValidationException { - Mockito.when(tppCertValidatorDataHolder.isTppValidationEnabled()).thenReturn(true); - Mockito.when(tppCertValidatorDataHolder.getTPPValidationServiceImpl()).thenReturn(""); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()) - .thenReturn(tppCertValidatorDataHolder); - - Assert.assertFalse(certValidationService - .validateTppRoles(eidasPeerCertificate, Arrays.asList(PSD2RoleEnum.AISP, PSD2RoleEnum.PISP))); - } - - @Test(description = "When TPPValidationImpl path configuration is invalid, then throw TPPValidationException", - expectedExceptions = {TPPValidationException.class}) - public void testValidateTPPWithInvalidImplConfig() throws TPPValidationException, CertificateValidationException { - Mockito.when(tppCertValidatorDataHolder.isTppValidationEnabled()).thenReturn(true); - Mockito.when(tppCertValidatorDataHolder.getTPPValidationServiceImpl()).thenReturn("dummy-path"); - Mockito.when(tppCertValidatorDataHolder.getTppValidationService()).thenReturn(null); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()) - .thenReturn(tppCertValidatorDataHolder); - - Assert.assertFalse(certValidationService - .validateTppRoles(eidasPeerCertificate, Arrays.asList(PSD2RoleEnum.AISP, PSD2RoleEnum.PISP))); - } - - @Test(description = "When tpp certificate and roles are valid, then should return true") - public void testValidateTPPWithCustomRoleValidation() - throws TPPValidationException, CertificateValidationException { - Mockito.when(tppCertValidatorDataHolder.isTppValidationEnabled()).thenReturn(false); - Mockito.when(tppCertValidatorDataHolder.isPsd2RoleValidationEnabled()).thenReturn(true); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()) - .thenReturn(tppCertValidatorDataHolder); - - Assert.assertTrue(certValidationService - .validateTppRoles(eidasPeerCertificate, Arrays.asList(PSD2RoleEnum.AISP, PSD2RoleEnum.PISP))); - } - - @Test(description = "When both tppValidationEnabled and psd2RoleValidationEnabled are false, " + - "then should throw TPPValidationException", expectedExceptions = TPPValidationException.class) - public void testValidateTPPWithInvalidConfigs() throws TPPValidationException, CertificateValidationException { - Mockito.when(tppCertValidatorDataHolder.isTppValidationEnabled()).thenReturn(false); - Mockito.when(tppCertValidatorDataHolder.isPsd2RoleValidationEnabled()).thenReturn(false); - - PowerMockito.mockStatic(TPPCertValidatorDataHolder.class); - PowerMockito.when(TPPCertValidatorDataHolder.getInstance()) - .thenReturn(tppCertValidatorDataHolder); - - Assert.assertTrue(certValidationService - .validateTppRoles(eidasPeerCertificate, Arrays.asList(PSD2RoleEnum.AISP, PSD2RoleEnum.PISP))); - } - - @Test(description = "when valid certificate and roles provided, then return true") - public void testIsRequiredRolesMatchWithScopes() throws Exception { - List requiredPSD2Roles = Arrays.asList(PSD2RoleEnum.AISP, PSD2RoleEnum.PISP); - - boolean isValid = WhiteboxImpl.invokeMethod(this.certValidationService, - "isRequiredRolesMatchWithScopes", eidasPeerCertificate, requiredPSD2Roles); - - Assert.assertTrue(isValid); - } - - @Test(description = "when invalid certificate and roles provided, then throw TPPValidationException", - expectedExceptions = {TPPValidationException.class}) - public void testIsRequiredRolesMatchWithScopesWithInvalidRoles() throws Exception { - WhiteboxImpl.invokeMethod(this.certValidationService, - "isRequiredRolesMatchWithScopes", eidasPeerCertificate, Collections.singletonList(PSD2RoleEnum.ASPSP)); - } - - @Test(description = "when valid certificate provided, then should return GOOD revocation status") - public void testIsRevokedWithValidCert() throws Exception { - OCSPValidator mockOCSPValidator = Mockito.mock(OCSPValidator.class); - Mockito.when(mockOCSPValidator.checkRevocationStatus(Mockito.any(X509Certificate.class), - Mockito.any(X509Certificate.class))).thenReturn(RevocationStatus.GOOD); - - RevocationStatus result = WhiteboxImpl.invokeMethod(this.certValidationService, - "isRevoked", mockOCSPValidator, eidasPeerCertificate, testPeerCertificateIssuer); - - Assert.assertSame(result, RevocationStatus.GOOD); - } - - @Test(description = "when valid certificate provided, then should return UNKNOWN revocation status") - public void testIsRevokedWithInvalidCert() throws Exception { - OCSPValidator mockOCSPValidator = Mockito.mock(OCSPValidator.class); - Mockito.when(mockOCSPValidator.checkRevocationStatus(Mockito.any(X509Certificate.class), - Mockito.any(X509Certificate.class))).thenThrow(CertificateValidationException.class); - - RevocationStatus result = WhiteboxImpl.invokeMethod(this.certValidationService, - "isRevoked", mockOCSPValidator, eidasPeerCertificate, testPeerCertificateIssuer); - - Assert.assertSame(result, RevocationStatus.UNKNOWN); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/service/RevocationValidatorFactoryTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/service/RevocationValidatorFactoryTest.java deleted file mode 100644 index bb429dc1..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/service/RevocationValidatorFactoryTest.java +++ /dev/null @@ -1,67 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.service; - -import com.wso2.openbanking.accelerator.gateway.executor.revocation.CRLValidator; -import com.wso2.openbanking.accelerator.gateway.executor.revocation.OCSPValidator; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -/** - * Test for revocation validation service. - */ -public class RevocationValidatorFactoryTest { - - private RevocationValidatorFactory revocationValidatorFactory; - - @BeforeClass - public void init() { - this.revocationValidatorFactory = new RevocationValidatorFactory(); - } - - @Test(description = "when valid revocation validator, then return valid validator object") - public void testGetValidatorWithValidStr() { - Assert.assertTrue(revocationValidatorFactory - .getValidator("OCSP", 1) instanceof OCSPValidator); - Assert.assertTrue(revocationValidatorFactory - .getValidator("CRL", 1) instanceof CRLValidator); - } - - @Test(description = "when invalid revocation validator, then return null") - public void testGetValidatorWithInvalidStr() { - Assert.assertNull(revocationValidatorFactory - .getValidator("INVALID", 1)); - } - - @Test(description = "when valid revocation validator, then return valid validator object") - public void testUpdatedGetValidatorWithValidStr() { - Assert.assertTrue(revocationValidatorFactory - .getValidator("OCSP", 1, 5000, 5000, 5000) instanceof OCSPValidator); - Assert.assertTrue(revocationValidatorFactory - .getValidator("CRL", 1, 5000, 5000, 5000) instanceof CRLValidator); - } - - @Test(description = "when invalid revocation validator, then return null") - public void testUpdatedGetValidatorWithInvalidStr() { - Assert.assertNull(revocationValidatorFactory - .getValidator("INVALID", 1, 5000, 5000, 5000)); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/test/TestConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/test/TestConstants.java deleted file mode 100644 index 34dc3767..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/test/TestConstants.java +++ /dev/null @@ -1,61 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.test; - -import java.util.AbstractMap; -import java.util.Map; -import java.util.stream.Collectors; -import java.util.stream.Stream; - -/** - * Constants for testing. - */ -public class TestConstants { - - public static final String INVALID_EXECUTOR_CLASS = - "com.wso2.openbanking.accelerator.gateway.executor.test.executor.InvalidClass"; - public static final String VALID_EXECUTOR_CLASS = - "com.wso2.openbanking.accelerator.gateway.executor.test.executor.MockOBExecutor"; - - public static final Map VALID_EXECUTOR_MAP = Stream.of( - new AbstractMap.SimpleImmutableEntry<>(1, VALID_EXECUTOR_CLASS)) - .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue)); - - public static final Map INVALID_EXECUTOR_MAP = Stream.of( - new AbstractMap.SimpleImmutableEntry<>(1, INVALID_EXECUTOR_CLASS)) - .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue)); - public static final String CUSTOM_PAYLOAD = "{\"custom\":\"payload\"}"; - public static final Map> FULL_VALIDATOR_MAP = Stream.of( - new AbstractMap.SimpleImmutableEntry<>("Default", VALID_EXECUTOR_MAP), - new AbstractMap.SimpleImmutableEntry<>("DCR", VALID_EXECUTOR_MAP)) - .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue)); - public static String authHeader = "eyJ4NXQiOiJNell4TW1Ga09HWXdNV0kwWldObU5EY3hOR1l3WW1NNFpUQ" + - "TNNV0kyTkRBelpHUXpOR00wWkdSbE5qSmtPREZrWkRSaU9URmtNV0ZoTXpVMlpHVmxOZyIsImtpZCI6Ik16" + - "WXhNbUZrT0dZd01XSTBaV05tTkRjeE5HWXdZbU00WlRBM01XSTJOREF6WkdRek5HTTBaR1JsTmpKa09ERmt" + - "aRFJpT1RGa01XRmhNelUyWkdWbE5nX1JTMjU2IiwiYWxnIjoiUlMyNTYifQ.eyJzdWIiOiJhZG1pbiIsImF" + - "1dCI6IkFQUExJQ0FUSU9OIiwiYXVkIjoiaENQUHFwbndTMWFKajd0Zkd6ckVVY3J0R2M0YSIsIm5iZiI6MT" + - "YxNDU5MDE4NSwiYXpwIjoiaENQUHFwbndTMWFKajd0Zkd6ckVVY3J0R2M0YSIsInNjb3BlIjoicmVhZDpwZ" + - "XRzIHdyaXRlOnBldHMiLCJpc3MiOiJodHRwczpcL1wvbG9jYWxob3N0Ojk0NDNcL29hdXRoMlwvdG9rZW4i" + - "LCJleHAiOjE2MTQ1OTM3ODUsImlhdCI6MTYxNDU5MDE4NSwianRpIjoiZGU3M2E3MzUtNmQzZi00MWI2LWE" + - "yYTYtN2U0ZTI1YWQxYTQxIn0.Jsz_0Wo79oBcVb2ibIVuJZ7pnsmIvU1r-RlFANiUYbjyTm8gF5b5CBf5uT" + - "JvKBM5BkqOSRbfgZdCMKZ7l83yZ5OSYDckwJ7rYKlcyzz50DKXlNW2-4J6d87uC1EOA10mg4pPC9LAH7Zdm" + - "MtN1JMY13xevKzoYB9FyuFgdLIHI4ALQOxeMAJZm_Y5_qBJgj1usE1FUmQUCdTc4aY3EbYkM9gZRO8Oc4HI" + - "7nn8eLwVShQqEdVDdtzsn0GJXLUlljxCfSAkVmP3vkxW1ZyC9OmmroONhdeEoJmy4Dr3JWwMNNNuzzFbT8K" + - "ycU1HwUqHTD5nL5Gs5jzSx8E-JxvdRotUCw"; -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/test/executor/MockOBExecutor.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/test/executor/MockOBExecutor.java deleted file mode 100644 index 7180e681..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/test/executor/MockOBExecutor.java +++ /dev/null @@ -1,60 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.test.executor; - -import com.wso2.openbanking.accelerator.gateway.executor.core.OpenBankingGatewayExecutor; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; - -/** - * Mock Open banking executor for testing. - */ -public class MockOBExecutor implements OpenBankingGatewayExecutor { - - @Override - public void preProcessRequest(OBAPIRequestContext obapiRequestContext) { - - obapiRequestContext.setModifiedPayload("{}"); - } - - @Override - public void preProcessResponse(OBAPIResponseContext obapiResponseContext) { - - } - - /** - * Method to handle post response. - * - * @param obapiResponseContext OB response context object - */ - @Override - public void postProcessResponse(OBAPIResponseContext obapiResponseContext) { - - } - - /** - * Method to handle post request. - * - * @param obapiRequestContext OB request context object - */ - @Override - public void postProcessRequest(OBAPIRequestContext obapiRequestContext) { - - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/test/executor/TestRouter.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/test/executor/TestRouter.java deleted file mode 100644 index c46fe4c3..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/test/executor/TestRouter.java +++ /dev/null @@ -1,44 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.test.executor; - -import com.wso2.openbanking.accelerator.gateway.executor.core.AbstractRequestRouter; -import com.wso2.openbanking.accelerator.gateway.executor.core.OpenBankingGatewayExecutor; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIResponseContext; - -import java.util.List; - -/** - * Router for testing. - */ -public class TestRouter extends AbstractRequestRouter { - - @Override - public List getExecutorsForRequest(OBAPIRequestContext requestContext) { - - return super.getExecutorMap().get(requestContext.getMsgInfo().getHeaders().get("test-prop")); - } - - @Override - public List getExecutorsForResponse(OBAPIResponseContext requestContext) { - - return super.getExecutorMap().get("VALID"); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/test/util/TestUtil.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/test/util/TestUtil.java deleted file mode 100644 index 94e1016d..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/test/util/TestUtil.java +++ /dev/null @@ -1,52 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.test.util; - -import com.wso2.openbanking.accelerator.common.util.OpenBankingUtils; -import com.wso2.openbanking.accelerator.gateway.executor.core.OpenBankingGatewayExecutor; -import com.wso2.openbanking.accelerator.gateway.executor.test.TestConstants; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -/** - * Util for testing. - */ -public class TestUtil { - - public static Map> initExecutors() { - - Map> executors = new HashMap<>(); - Map> fullValidatorMap = TestConstants.FULL_VALIDATOR_MAP; - for (Map.Entry> stringMapEntry : fullValidatorMap.entrySet()) { - List executorList = new ArrayList<>(); - Map executorNames = stringMapEntry.getValue(); - for (Map.Entry executorEntity : executorNames.entrySet()) { - OpenBankingGatewayExecutor object = (OpenBankingGatewayExecutor) - OpenBankingUtils.getClassInstanceFromFQN(executorEntity.getValue()); - executorList.add(object); - } - executors.put(stringMapEntry.getKey(), executorList); - } - return executors; - - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/util/CertificateValidationUtilsTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/util/CertificateValidationUtilsTest.java deleted file mode 100644 index d171b6b9..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/util/CertificateValidationUtilsTest.java +++ /dev/null @@ -1,184 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.util; - -import com.wso2.openbanking.accelerator.common.exception.CertificateValidationException; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.CertificateUtils; -import com.wso2.openbanking.accelerator.gateway.executor.model.OBAPIRequestContext; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.Test; - -import java.io.File; -import java.io.IOException; -import java.security.KeyStore; -import java.security.KeyStoreException; -import java.security.NoSuchAlgorithmException; -import java.security.cert.Certificate; -import java.security.cert.CertificateException; -import java.security.cert.X509Certificate; -import java.time.Instant; -import java.time.temporal.ChronoUnit; -import java.util.ArrayList; -import java.util.Date; -import java.util.Optional; - -import javax.security.cert.CertificateEncodingException; - -/** - * Test for certificate validation utils. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest(KeyStore.class) -public class CertificateValidationUtilsTest extends PowerMockTestCase { - - String path = "src/test/resources"; - File file = new File(path); - String absolutePathForTestResources = file.getAbsolutePath(); - - @Test(description = "when valid certificate, then return java.security.cert.X509Certificate") - public void testConvertWithValidCert() { - javax.security.cert.X509Certificate testCert = TestValidationUtil - .getCertFromStr(TestValidationUtil.TEST_CLIENT_CERT); - - Assert.assertNotNull(CertificateValidationUtils.convert(testCert)); - Assert.assertTrue(CertificateValidationUtils.convert(testCert).orElse(null) instanceof X509Certificate); - } - - @Test(description = "when null certificate, then return Optional empty") - public void testConvertWithNullCert() { - Assert.assertFalse(CertificateValidationUtils.convert(null) - .isPresent()); - } - - @Test(description = "when valid certificate, then return java.security.cert.X509Certificate") - public void testConvertCertWithValidCert() throws CertificateException { - javax.security.cert.X509Certificate testCert = TestValidationUtil - .getCertFromStr(TestValidationUtil.TEST_CLIENT_CERT); - - Assert.assertNotNull(CertificateValidationUtils.convertCert(testCert)); - Assert.assertTrue(CertificateValidationUtils.convertCert(testCert) - .orElse(null) instanceof X509Certificate); - } - - @Test(description = "when null certificate, then return Optional empty") - public void testConvertCertWithNullCert() throws CertificateException { - Assert.assertFalse(CertificateValidationUtils.convertCert(null) - .isPresent()); - } - - @Test(description = "when client store is null, then throw CertificateValidationException", - expectedExceptions = CertificateValidationException.class) - public void testGetIssuerCertificateFromNullTruststore() throws CertificateValidationException, - OpenBankingException { - - X509Certificate peerCertificate = CertificateUtils.parseCertificate(TestValidationUtil.TEST_CLIENT_CERT); - - X509Certificate issuerCertificate = CertificateValidationUtils - .getIssuerCertificateFromTruststore(peerCertificate); - - Assert.assertNotNull(issuerCertificate); - } - - @Test(description = "when valid peer certificate, then issuer certificate should return") - public void testGetIssuerCertificateFromTruststore() throws CertificateException, NoSuchAlgorithmException, - KeyStoreException, IOException, CertificateValidationException, OpenBankingException { - - X509Certificate peerCertificate = CertificateUtils.parseCertificate(TestValidationUtil.TEST_CLIENT_CERT); - - CertificateValidationUtils.loadTrustStore(absolutePathForTestResources + "/client-truststore.jks", - "wso2carbon".toCharArray()); - X509Certificate issuerCertificate = CertificateValidationUtils - .getIssuerCertificateFromTruststore(peerCertificate); - - Assert.assertNotNull(issuerCertificate); - Assert.assertEquals(issuerCertificate.getSubjectDN().getName(), peerCertificate.getIssuerDN().getName()); - } - - @Test(description = "when error occured, then should set error true in OBAPIRequestContext object") - public void testHandleExecutorErrors() { - OBAPIRequestContext obapiRequestContext = Mockito.mock(OBAPIRequestContext.class); - CertificateValidationException exception = new CertificateValidationException("dummy exception"); - - CertificateValidationUtils.handleExecutorErrors(exception, obapiRequestContext); - - Mockito.verify(obapiRequestContext, Mockito.times(1)).setError(true); - Mockito.verify(obapiRequestContext, Mockito.times(1)) - .setErrors(Mockito.any(ArrayList.class)); - } - - @Test(description = "should return current date") - public void testGetNewDate() { - Instant actual = CertificateValidationUtils.getNewDate().toInstant().truncatedTo(ChronoUnit.DAYS); - Instant expected = new Date().toInstant().truncatedTo(ChronoUnit.DAYS); - Assert.assertEquals(actual, expected); - } - - @Test(description = "when uninitialized keystore, then throw CertificateValidationException", - expectedExceptions = CertificateValidationException.class) - public void testRetrieveCertificateFromTruststore() throws KeyStoreException, CertificateValidationException { - PowerMockito.mockStatic(KeyStore.class); - KeyStore keyStoreMock = PowerMockito.mock(KeyStore.class); - - CertificateValidationUtils.retrieveCertificateFromTruststore(null, keyStoreMock); - } - - @Test(description = "when invalid encoded certificate, then return empty", - expectedExceptions = CertificateException.class) - public void testConvertWithInvalidCertEncoding() throws CertificateEncodingException, CertificateException { - javax.security.cert.X509Certificate x509CertificateMock = Mockito - .mock(javax.security.cert.X509Certificate.class); - Mockito.doThrow(CertificateEncodingException.class).when(x509CertificateMock).getEncoded(); - - Optional convertedCert = CertificateValidationUtils.convertCert(x509CertificateMock); - Assert.assertFalse(convertedCert.isPresent()); - } - - @Test(description = "when error occurred while converting, then return empty", - expectedExceptions = CertificateException.class) - public void testConvertWithInvalidCert() throws CertificateEncodingException, CertificateException { - javax.security.cert.X509Certificate x509CertificateMock = Mockito - .mock(javax.security.cert.X509Certificate.class); - Mockito.doThrow(CertificateException.class).when(x509CertificateMock).getEncoded(); - - Optional convertedCert = CertificateValidationUtils.convertCert(x509CertificateMock); - Assert.assertFalse(convertedCert.isPresent()); - } - - @Test(description = "when valid java.security.cert.Certificate is provided, " + - "then return java.security.cert.X509Certificate") - public void testConvertCertToX509CertWithValidCert() throws OpenBankingException, CertificateException { - Certificate testCert = TestValidationUtil.getTestClientCertificate(); - - Assert.assertTrue(CertificateValidationUtils.convertCertToX509Cert(testCert).isPresent()); - Assert.assertTrue(CertificateValidationUtils.convertCertToX509Cert(testCert).get() instanceof X509Certificate); - } - - @Test(expectedExceptions = CertificateException.class, description = "When an invalid certificate is provided, " + - "throw a CertificateException") - public void testConvertCertToX509CertException() throws CertificateException { - - CertificateValidationUtils.convertCertToX509Cert(TestValidationUtil.getEmptyTestCertificate()); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/util/TestValidationUtil.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/util/TestValidationUtil.java deleted file mode 100644 index cbc787ff..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/executor/util/TestValidationUtil.java +++ /dev/null @@ -1,384 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.executor.util; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.CertificateUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.io.ByteArrayInputStream; -import java.io.InputStream; -import java.security.PublicKey; -import java.security.cert.Certificate; -import java.util.Base64; - -import javax.security.cert.CertificateException; -import javax.security.cert.X509Certificate; - -/** - * Test for validation utils. - */ -public class TestValidationUtil { - - public static final String TEST_CLIENT_CERT = "-----BEGIN CERTIFICATE-----" + - "MIIFODCCBCCgAwIBAgIEWcZEPjANBgkqhkiG9w0BAQsFADBTMQswCQYDVQQGEwJH" + - "QjEUMBIGA1UEChMLT3BlbkJhbmtpbmcxLjAsBgNVBAMTJU9wZW5CYW5raW5nIFBy" + - "ZS1Qcm9kdWN0aW9uIElzc3VpbmcgQ0EwHhcNMjEwOTA4MDUyODEyWhcNMjIxMDA4" + - "MDU1ODEyWjBhMQswCQYDVQQGEwJHQjEUMBIGA1UEChMLT3BlbkJhbmtpbmcxGzAZ" + - "BgNVBAsTEjAwMTU4MDAwMDFIUVFyWkFBWDEfMB0GA1UEAxMWdTNaV2xmOVl0NDJk" + - "eVpnSXZ6a3ZxYjCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEBALqlz2yg" + - "mP4yqmWfvSkus6LrSvB1kknauQAnU3MgL7Eg+ZrlGljtgL0PJ3gPR9kkRG4fts2v" + - "sxbnrART4YTs/AVagSxahnXrVnj/GlFVbO+cWpMadnYLl+pe7k4n1IdtD7m3WIpV" + - "Zlwwgj/LQSD+b57Te+MkpCRoKFIWQMW0Eh5M6Mftb1MIN5h3zR/QLmEuREUzPshB" + - "3CIMHv9LX2St8mA6n5sH/gIJOQW7breP7N7QAsOjKhgOhy4vEWx+Ig7VjCH4EU7I" + - "AIHKSYhLICTBPKF5c1yTp/gMCE086VyMDu7i52jNKz2VsVX13qNr/7t2wVKaoQ2Z" + - "frUA3uq7HX0vEe8CAwEAAaOCAgQwggIAMA4GA1UdDwEB/wQEAwIHgDAgBgNVHSUB" + - "Af8EFjAUBggrBgEFBQcDAQYIKwYBBQUHAwIwgeAGA1UdIASB2DCB1TCB0gYLKwYB" + - "BAGodYEGAWQwgcIwKgYIKwYBBQUHAgEWHmh0dHA6Ly9vYi50cnVzdGlzLmNvbS9w" + - "b2xpY2llczCBkwYIKwYBBQUHAgIwgYYMgYNVc2Ugb2YgdGhpcyBDZXJ0aWZpY2F0" + - "ZSBjb25zdGl0dXRlcyBhY2NlcHRhbmNlIG9mIHRoZSBPcGVuQmFua2luZyBSb290" + - "IENBIENlcnRpZmljYXRpb24gUG9saWNpZXMgYW5kIENlcnRpZmljYXRlIFByYWN0" + - "aWNlIFN0YXRlbWVudDBtBggrBgEFBQcBAQRhMF8wJgYIKwYBBQUHMAGGGmh0dHA6" + - "Ly9vYi50cnVzdGlzLmNvbS9vY3NwMDUGCCsGAQUFBzAChilodHRwOi8vb2IudHJ1" + - "c3Rpcy5jb20vb2JfcHBfaXNzdWluZ2NhLmNydDA6BgNVHR8EMzAxMC+gLaArhilo" + - "dHRwOi8vb2IudHJ1c3Rpcy5jb20vb2JfcHBfaXNzdWluZ2NhLmNybDAfBgNVHSME" + - "GDAWgBRQc5HGIXLTd/T+ABIGgVx5eW4/UDAdBgNVHQ4EFgQUt/iQ/+ksD95pZUol" + - "YF8R+2838bgwDQYJKoZIhvcNAQELBQADggEBAH7/dvG7jm6xN1G0nziOHN/GSdJt" + - "6wxodmRr/nDGBiHjONS2qq6wSSaN/QfUfe5OPbICi6dDNDgJpk1ZJKWXpdBW3K0e" + - "3mjOvEjMSC6V/iu8T6NT4PWF9IGc10I93z/NbVYFahjfLtuBzBKwr7DbASYawzVF" + - "rUa7CGbzk+nUGoqoMV/0eF+UtjDx2NYoGov7WK07XDFxsJJOjq0lA7SB3/3BqttW" + - "J+iX9CafGYP2v9hjjOz1y7Jbr66Kd9tBK9C0+5bHvO84VoupUl8iateeBiFPqd+p" + - "gLzORyiwIa7lsLvx273Fz3iOvX2Ksg9I/qhWABZ4adm//G45+GDGKFebzLo=" + - "-----END CERTIFICATE-----"; - - public static final String TEST_CLIENT_CERT_ISSUER = "-----BEGIN CERTIFICATE-----" + - "MIIDtzCCAp+gAwIBAgIQDOfg5RfYRv6P5WD8G/AwOTANBgkqhkiG9w0BAQUFADBl" + - "MQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYDVQQLExB3" + - "d3cuZGlnaWNlcnQuY29tMSQwIgYDVQQDExtEaWdpQ2VydCBBc3N1cmVkIElEIFJv" + - "b3QgQ0EwHhcNMDYxMTEwMDAwMDAwWhcNMzExMTEwMDAwMDAwWjBlMQswCQYDVQQG" + - "EwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYDVQQLExB3d3cuZGlnaWNl" + - "cnQuY29tMSQwIgYDVQQDExtEaWdpQ2VydCBBc3N1cmVkIElEIFJvb3QgQ0EwggEi" + - "MA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQCtDhXO5EOAXLGH87dg+XESpa7c" + - "JpSIqvTO9SA5KFhgDPiA2qkVlTJhPLWxKISKityfCgyDF3qPkKyK53lTXDGEKvYP" + - "mDI2dsze3Tyoou9q+yHyUmHfnyDXH+Kx2f4YZNISW1/5WBg1vEfNoTb5a3/UsDg+" + - "wRvDjDPZ2C8Y/igPs6eD1sNuRMBhNZYW/lmci3Zt1/GiSw0r/wty2p5g0I6QNcZ4" + - "VYcgoc/lbQrISXwxmDNsIumH0DJaoroTghHtORedmTpyoeb6pNnVFzF1roV9Iq4/" + - "AUaG9ih5yLHa5FcXxH4cDrC0kqZWs72yl+2qp/C3xag/lRbQ/6GW6whfGHdPAgMB" + - "AAGjYzBhMA4GA1UdDwEB/wQEAwIBhjAPBgNVHRMBAf8EBTADAQH/MB0GA1UdDgQW" + - "BBRF66Kv9JLLgjEtUYunpyGd823IDzAfBgNVHSMEGDAWgBRF66Kv9JLLgjEtUYun" + - "pyGd823IDzANBgkqhkiG9w0BAQUFAAOCAQEAog683+Lt8ONyc3pklL/3cmbYMuRC" + - "dWKuh+vy1dneVrOfzM4UKLkNl2BcEkxY5NM9g0lFWJc1aRqoR+pWxnmrEthngYTf" + - "fwk8lOa4JiwgvT2zKIn3X/8i4peEH+ll74fg38FnSbNd67IJKusm7Xi+fT8r87cm" + - "NW1fiQG2SVufAQWbqz0lwcy2f8Lxb4bG+mRo64EtlOtCt/qMHt1i8b5QZ7dsvfPx" + - "H2sMNgcWfzd8qVttevESRmCD1ycEvkvOl77DZypoEd+A5wwzZr8TDRRu838fYxAe" + - "+o0bJW1sj6W3YQGx0qMmoRBxna3iw/nDmVG3KwcIzi7mULKn+gpFL6Lw8g==" + - "-----END CERTIFICATE-----"; - - public static final String EXPIRED_SELF_CERT = "-----BEGIN CERTIFICATE-----" + - "MIIDiTCCAnGgAwIBAgIENx3SZjANBgkqhkiG9w0BAQsFADB1MQswCQYDVQQGEwJs" + - "azEQMA4GA1UECBMHd2VzdGVybjEQMA4GA1UEBxMHY29sb21ibzENMAsGA1UEChME" + - "d3NvMjEUMBIGA1UECxMLb3BlbmJhbmtpbmcxHTAbBgNVBAMTFG9wZW5iYW5raW5n" + - "LndzbzIuY29tMB4XDTIwMDMwMTEyMjE1MVoXDTIwMDUzMDEyMjE1MVowdTELMAkG" + - "A1UEBhMCbGsxEDAOBgNVBAgTB3dlc3Rlcm4xEDAOBgNVBAcTB2NvbG9tYm8xDTAL" + - "BgNVBAoTBHdzbzIxFDASBgNVBAsTC29wZW5iYW5raW5nMR0wGwYDVQQDExRvcGVu" + - "YmFua2luZy53c28yLmNvbTCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEB" + - "AKWMb1mhSthxi5vmQcvEnt0rauYv8uFWjGyiuCkk5wQbArybGXyC8rrZf5qNNY4s" + - "RG2+Yimxph2Z8MWWPFBebTIABPuRcVDquX7fL4+8FZJTH3JLwfT+slunAA4473mZ" + - "9s2fAVu6CmQf1V09+fEbMGI9WWh53g19wg5WdlToOX4g5lh4QtGRpbWpEWaYrKzS" + - "B5EWOUI7lroFtv6s9OpEO59VAkXWKUbT98T8TCYqiDH+nMy3k+GbVawxXeHYHQr+" + - "XlbcChPaCwhMXspqKG49xaJmrOuRMoAWCBGUW8r2RDhQ+FP5V/sTRMqKmBv9gTe6" + - "RJwoKPlDt+0aX9vaFjKpjPcCAwEAAaMhMB8wHQYDVR0OBBYEFGH0gyeHIz1+ONGI" + - "PuGnAhrS3apoMA0GCSqGSIb3DQEBCwUAA4IBAQCVEakh1SLnZOz2IK0ISbAV5UBb" + - "nerLNDl+X+YSYsCQM1SBcXDjlkSAeP3ErJEO3RW3wdRQjLRRHomwSCSRE84SUfSL" + - "VPIbeR7jm4sS9x5rnlGF6iqhYh2MlZD/hFxdrGoYv8g/JN4FFFMXRmmaQ8ouYJwc" + - "4ZoxRdCXszeI5Zp2+b14cs/nf4geYliHtcDr/w7fkvQ0hn+c1lTihbW0/eE32aUK" + - "SULAmjx0sCDfDAQItP79CC7jCW0TFN0CMORw/+fzp/dnVboSZ2MgcuRIH1Ez+6/1" + - "1QJD2SrkkaRSEaXI6fe9jgHVhnqK9V3y3WAuzEKjaKw6jV8BjkXAA4dQj1Re" + - "-----END CERTIFICATE-----"; - public static final String EIDAS_CERT = "-----BEGIN CERTIFICATE-----" + - "MIIF2DCCBMCgAwIBAgIEWcYJGDANBgkqhkiG9w0BAQsFADBTMQswCQYDVQQGEwJH" + - "QjEUMBIGA1UEChMLT3BlbkJhbmtpbmcxLjAsBgNVBAMTJU9wZW5CYW5raW5nIFBy" + - "ZS1Qcm9kdWN0aW9uIElzc3VpbmcgQ0EwHhcNMjAxMjE1MDY1ODMxWhcNMjIwMTE1" + - "MDcyODMxWjBzMQswCQYDVQQGEwJHQjEaMBgGA1UEChMRV1NPMiAoVUspIExJTUlU" + - "RUQxKzApBgNVBGETIlBTREdCLU9CLVVua25vd24wMDE1ODAwMDAxSFFRclpBQVgx" + - "GzAZBgNVBAMTEjAwMTU4MDAwMDFIUVFyWkFBWDCCASIwDQYJKoZIhvcNAQEBBQAD" + - "ggEPADCCAQoCggEBAN4RybsCYch4OAzJz3bfVAsz04lcuGYz1DE21l6PKkrABU3k" + - "AYWUw9YtLWDVfA4nemSd5vb9dNJJoY6bvLTBbWBpWqOmq+lzXB4WrGuF5v4BaE8U" + - "OeuVoIxKg9sV2mHAOaflVX8cz0dZSAbf1h+lvRRzIlX4TgN2ApZACIdtcBZfooOj" + - "1F070MM9gyLw2A3cOew4MXaaZZFHP0CzQWlRyftaw0mYrx7m2iUK+4d4zEgEjC05" + - "kdEpkdTtXvuTla/ER9O7DSnx++qKoRcEkqloOF/Rz7uhRhGfQHy6JwrNrZOr9khS" + - "90pEejBnr8Is9BLqaRwE6COAPq/C+w5ZQ4pd9oMCAwEAAaOCApIwggKOMA4GA1Ud" + - "DwEB/wQEAwIHgDCBiwYIKwYBBQUHAQMEfzB9MBMGBgQAjkYBBjAJBgcEAI5GAQYD" + - "MGYGBgQAgZgnAjBcMDUwMwYHBACBmCcBAgwGUFNQX1BJBgcEAIGYJwEDDAZQU1Bf" + - "QUkGBwQAgZgnAQQMBlBTUF9JQwwbRmluYW5jaWFsIENvbmR1Y3QgQXV0aG9yaXR5" + - "DAZHQi1GQ0EwIAYDVR0lAQH/BBYwFAYIKwYBBQUHAwEGCCsGAQUFBwMCMIHgBgNV" + - "HSAEgdgwgdUwgdIGCysGAQQBqHWBBgFkMIHCMCoGCCsGAQUFBwIBFh5odHRwOi8v" + - "b2IudHJ1c3Rpcy5jb20vcG9saWNpZXMwgZMGCCsGAQUFBwICMIGGDIGDVXNlIG9m" + - "IHRoaXMgQ2VydGlmaWNhdGUgY29uc3RpdHV0ZXMgYWNjZXB0YW5jZSBvZiB0aGUg" + - "T3BlbkJhbmtpbmcgUm9vdCBDQSBDZXJ0aWZpY2F0aW9uIFBvbGljaWVzIGFuZCBD" + - "ZXJ0aWZpY2F0ZSBQcmFjdGljZSBTdGF0ZW1lbnQwbQYIKwYBBQUHAQEEYTBfMCYG" + - "CCsGAQUFBzABhhpodHRwOi8vb2IudHJ1c3Rpcy5jb20vb2NzcDA1BggrBgEFBQcw" + - "AoYpaHR0cDovL29iLnRydXN0aXMuY29tL29iX3BwX2lzc3VpbmdjYS5jcnQwOgYD" + - "VR0fBDMwMTAvoC2gK4YpaHR0cDovL29iLnRydXN0aXMuY29tL29iX3BwX2lzc3Vp" + - "bmdjYS5jcmwwHwYDVR0jBBgwFoAUUHORxiFy03f0/gASBoFceXluP1AwHQYDVR0O" + - "BBYEFN0LLFBaqNtl17Ds7a+4EwedY69oMA0GCSqGSIb3DQEBCwUAA4IBAQBpyV93" + - "NoWNDg8PhcTWrxQFRLSvNCaDfKQw7MVzK7pl9cFnugZPXUg67KmLiJ+GzI9HHym/" + - "yfd3Vwx5SNtfQVACmStKsLGv6kRGJcUAIgICV8ZGVlbsWpKam2ck7wR2138QD8s1" + - "igAIaSWzHyHlkPjy44hRDbLpEYhRf9c2bUYGYnkMUBhmhI3ZhbopR3Zac/1/VBlA" + - "VR7G0VQiloTHoQUL6OkaTnfdOEjU9Eeo8lQgrGjob5aCWrrPe4ExCyAZdn0NgE69" + - "womfyrqwLoQpiUGmOSZCuOgWmPe8OrbpGIaodZz2Wk5qgR5xrVkNDfvgM/nXm1r8" + - "HxriBi5shkweEW6g" + - "-----END CERTIFICATE-----"; - - public static final String EIDAS_CERT_ISSUER = "-----BEGIN CERTIFICATE-----" + - "MIIGEzCCA/ugAwIBAgIEWcT9RzANBgkqhkiG9w0BAQsFADBQMQswCQYDVQQGEwJH" + - "QjEUMBIGA1UEChMLT3BlbkJhbmtpbmcxKzApBgNVBAMTIk9wZW5CYW5raW5nIFBy" + - "ZS1Qcm9kdWN0aW9uIFJvb3QgQ0EwHhcNMTcwOTIyMTI0NjU3WhcNMjcwOTIyMTMx" + - "NjU3WjBTMQswCQYDVQQGEwJHQjEUMBIGA1UEChMLT3BlbkJhbmtpbmcxLjAsBgNV" + - "BAMTJU9wZW5CYW5raW5nIFByZS1Qcm9kdWN0aW9uIElzc3VpbmcgQ0EwggEiMA0G" + - "CSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQCyyrRg2jF01jXhX3IR44p338ZBozn8" + - "WkZaCN8MB+AlBfuXHD6mC/0v+N/Z4XI6E5pzArmTho8D6a6JDpAHmmefqGSqOXVb" + - "clYv1tHFjmC1FtKqkFHTTMyhl41nEMo0dnvWA45bMsGm0yMi/tEM5Vb5dSY4Zr/2" + - "LWgUTDFUisgUbyIIHT+L6qxPUPCpNuEd+AWVc9K0SlmhaC+UIfVO83gE1+9ar2dO" + - "NSFaK/a445Us6MnqgKvfkvKdaR06Ok/EhGgiAZORcyZ61EYFVVzJewy5NrFSF3mw" + - "iPYvMxoT5bxcwAEvxqBXpTDv8njQfR+cgZDeloeK1UqmW/DpR+jj3KNHAgMBAAGj" + - "ggHwMIIB7DAOBgNVHQ8BAf8EBAMCAQYwEgYDVR0TAQH/BAgwBgEB/wIBADCB4AYD" + - "VR0gBIHYMIHVMIHSBgsrBgEEAah1gQYBZDCBwjAqBggrBgEFBQcCARYeaHR0cDov" + - "L29iLnRydXN0aXMuY29tL3BvbGljaWVzMIGTBggrBgEFBQcCAjCBhgyBg1VzZSBv" + - "ZiB0aGlzIENlcnRpZmljYXRlIGNvbnN0aXR1dGVzIGFjY2VwdGFuY2Ugb2YgdGhl" + - "IE9wZW5CYW5raW5nIFJvb3QgQ0EgQ2VydGlmaWNhdGlvbiBQb2xpY2llcyBhbmQg" + - "Q2VydGlmaWNhdGUgUHJhY3RpY2UgU3RhdGVtZW50MGoGCCsGAQUFBwEBBF4wXDAy" + - "BggrBgEFBQcwAoYmaHR0cDovL29iLnRydXN0aXMuY29tL29idGVzdHJvb3RjYS5j" + - "cnQwJgYIKwYBBQUHMAGGGmh0dHA6Ly9vYi50cnVzdGlzLmNvbS9vY3NwMDcGA1Ud" + - "HwQwMC4wLKAqoCiGJmh0dHA6Ly9vYi50cnVzdGlzLmNvbS9vYl9wcF9yb290Y2Eu" + - "Y3JsMB8GA1UdIwQYMBaAFOw4jgva8/k3PpDefV9q5mDNeUKDMB0GA1UdDgQWBBRQ" + - "c5HGIXLTd/T+ABIGgVx5eW4/UDANBgkqhkiG9w0BAQsFAAOCAgEAdRg2H9uLwzlG" + - "qvHGjIz0ydM1tElujEcWJp5MeiorikK0rMOlxVU6ZFBlXPfO1APu0cZXxfHwWs91" + - "zoNCpGXebC6tiDFQ3+mI4qywtippjBqb6Sft37NlkXDzQETomsY7wETuUJ31xFA0" + - "FccI8WlAUzUOBE8OAGo5kAZ4FTa/nkd8c2wmuwSp+9/s+gQe0K9BkxywoP1WAEdU" + - "AaKW3RE9yuTbHA/ZF/zz4/Rpw/FB/hYhOxvDV6qInl5B7ErSH4r4v4D2jiE6apAc" + - "n5LT+e0aBa/EgGAxgyAgrYpw1s+TCUJot+227xRvXxeeZzXa2igsd+C845BGiSlt" + - "hzr0mqYDYEWJMfApZ+BlMtxa7K9T3D2l6XMv12RoNnEWe6H5xazTvBLiTibW3c5i" + - "j8WWKJNtQbgmooRPaKJIl+0rm54MFH0FDxJ+P4mAR6qa8JS911nS26iCsE9FQVK5" + - "1djuct349FYBOVM595/GkkTz9k1vXw1BdD71lNjI00Yjf73AAtvL/X4CpRz92Nag" + - "shS2Ia5a3qjjFrjx7z4h7QtMJGjuUsjTI/c+yjIYwAZ5gelF5gz7l2dn3g6B40pu" + - "7y1EewlfIQh/HVMF0ZpF29XL6+7siYQCGhP5cNJ04fotzqDPaT2XlOhE3yNkjp82" + - "uzCWvhLUJgE3D9V9PL0XD/ykNEP0Fio=" + - "-----END CERTIFICATE-----"; - - public static final String REQUEST_BODY_WITH_SSA = "eyJ0eXAiOiJKV1QiLCJhbGciOiJQUzI1NiIsImtpZCI6IkR3TUtk" + - "V01tajdQV2ludm9xZlF5WFZ6eVo2USJ9.eyJpc3MiOiI5YjV1c0RwYk50bXhEY1R6czdHektwIiwiaWF0IjoxNjAxOTgy" + - "MDQyLCJleHAiOjE2MDcyNTI0NDIsImp0aSI6IjE2MDE5ODIwNDYiLCJhdWQiOiJodHRwczovL2xvY2FsaG9zdDo4MjQzL" + - "3Rva2VuIiwic2NvcGUiOiJhY2NvdW50cyBwYXltZW50cyIsInRva2VuX2VuZHBvaW50X2F1dGhfbWV0aG9kIjoicHJpdm" + - "F0ZV9rZXlfand0IiwiZ3JhbnRfdHlwZXMiOlsiYXV0aG9yaXphdGlvbl9jb2RlIiwicmVmcmVzaF90b2tlbiJdLCJyZXN" + - "wb25zZV90eXBlcyI6WyJjb2RlIGlkX3Rva2VuIl0sImlkX3Rva2VuX3NpZ25lZF9yZXNwb25zZV9hbGciOiJQUzI1NiIs" + - "InJlcXVlc3Rfb2JqZWN0X3NpZ25pbmdfYWxnIjoiUFMyNTYiLCJzb2Z0d2FyZV9pZCI6IjliNXVzRHBiTnRteERjVHpzN" + - "0d6S3AiLCJhcHBsaWNhdGlvbl90eXBlIjoid2ViIiwicmVkaXJlY3RfdXJpcyI6WyJodHRwczovL3dzbzIuY29tIl0sIn" + - "Rva2VuX2VuZHBvaW50X2F1dGhfc2lnbmluZ19hbGciOiJQUzI1NiIsInNvZnR3YXJlX3N0YXRlbWVudCI6ImV5SmhiR2N" + - "pT2lKUVV6STFOaUlzSW10cFpDSTZJa2g2WVRsMk5XSm5SRXBqVDI1b1kxVmFOMEpOZDJKVFRGODBUbFl3WjFOR2RrbHFZ" + - "Vk5ZWkVNdE1XTTlJaXdpZEhsd0lqb2lTbGRVSW4wLmV5SnBjM01pT2lKUGNHVnVRbUZ1YTJsdVp5Qk1kR1FpTENKcFlYU" + - "WlPakUxT1RJek5qUTFOamdzSW1wMGFTSTZJak5rTVdJek5UazFaV1poWXpSbE16WWlMQ0p6YjJaMGQyRnlaVjlsYm5acG" + - "NtOXViV1Z1ZENJNkluTmhibVJpYjNnaUxDSnpiMlowZDJGeVpWOXRiMlJsSWpvaVZHVnpkQ0lzSW5OdlpuUjNZWEpsWDJ" + - "sa0lqb2lPV0kxZFhORWNHSk9kRzE0UkdOVWVuTTNSM3BMY0NJc0luTnZablIzWVhKbFgyTnNhV1Z1ZEY5cFpDSTZJamxp" + - "TlhWelJIQmlUblJ0ZUVSalZIcHpOMGQ2UzNBaUxDSnpiMlowZDJGeVpWOWpiR2xsYm5SZmJtRnRaU0k2SWxkVFR6SWdUM" + - "0JsYmlCQ1lXNXJhVzVuSUZSUVVDQW9VMkZ1WkdKdmVDa2lMQ0p6YjJaMGQyRnlaVjlqYkdsbGJuUmZaR1Z6WTNKcGNIUn" + - "BiMjRpT2lKVWFHbHpJRlJRVUNCSmN5QmpjbVZoZEdWa0lHWnZjaUIwWlhOMGFXNW5JSEIxY25CdmMyVnpMaUFpTENKemI" + - "yWjBkMkZ5WlY5MlpYSnphVzl1SWpveExqVXNJbk52Wm5SM1lYSmxYMk5zYVdWdWRGOTFjbWtpT2lKb2RIUndjem92TDNk" + - "emJ6SXVZMjl0SWl3aWMyOW1kSGRoY21WZmNtVmthWEpsWTNSZmRYSnBjeUk2V3lKb2RIUndjem92TDNkemJ6SXVZMjl0S" + - "Wwwc0luTnZablIzWVhKbFgzSnZiR1Z6SWpwYklrRkpVMUFpTENKUVNWTlFJbDBzSW05eVoyRnVhWE5oZEdsdmJsOWpiMj" + - "F3WlhSbGJuUmZZWFYwYUc5eWFYUjVYMk5zWVdsdGN5STZleUpoZFhSb2IzSnBkSGxmYVdRaU9pSlBRa2RDVWlJc0luSmx" + - "aMmx6ZEhKaGRHbHZibDlwWkNJNklsVnVhMjV2ZDI0d01ERTFPREF3TURBeFNGRlJjbHBCUVZnaUxDSnpkR0YwZFhNaU9p" + - "SkJZM1JwZG1VaUxDSmhkWFJvYjNKcGMyRjBhVzl1Y3lJNlczc2liV1Z0WW1WeVgzTjBZWFJsSWpvaVIwSWlMQ0p5YjJ4b" + - "GN5STZXeUpCU1ZOUUlpd2lVRWxUVUNKZGZTeDdJbTFsYldKbGNsOXpkR0YwWlNJNklrbEZJaXdpY205c1pYTWlPbHNpUV" + - "VsVFVDSXNJbEJKVTFBaVhYMHNleUp0WlcxaVpYSmZjM1JoZEdVaU9pSk9UQ0lzSW5KdmJHVnpJanBiSWtGSlUxQWlMQ0p" + - "RU1ZOUUlsMTlYWDBzSW5OdlpuUjNZWEpsWDJ4dloyOWZkWEpwSWpvaWFIUjBjSE02THk5M2MyOHlMbU52YlM5M2MyOHlM" + - "bXB3WnlJc0ltOXlaMTl6ZEdGMGRYTWlPaUpCWTNScGRtVWlMQ0p2Y21kZmFXUWlPaUl3TURFMU9EQXdNREF4U0ZGUmNsc" + - "EJRVmdpTENKdmNtZGZibUZ0WlNJNklsZFRUeklnS0ZWTEtTQk1TVTFKVkVWRUlpd2liM0puWDJOdmJuUmhZM1J6SWpwYm" + - "V5SnVZVzFsSWpvaVZHVmphRzVwWTJGc0lpd2laVzFoYVd3aU9pSnpZV05vYVc1cGMwQjNjMjh5TG1OdmJTSXNJbkJvYjI" + - "1bElqb2lLemswTnpjME1qYzBNemMwSWl3aWRIbHdaU0k2SWxSbFkyaHVhV05oYkNKOUxIc2libUZ0WlNJNklrSjFjMmx1" + - "WlhOeklpd2laVzFoYVd3aU9pSnpZV05vYVc1cGMwQjNjMjh5TG1OdmJTSXNJbkJvYjI1bElqb2lLemswTnpjME1qYzBNe" + - "mMwSWl3aWRIbHdaU0k2SWtKMWMybHVaWE56SW4xZExDSnZjbWRmYW5kcmMxOWxibVJ3YjJsdWRDSTZJbWgwZEhCek9pOH" + - "ZhMlY1YzNSdmNtVXViM0JsYm1KaGJtdHBibWQwWlhOMExtOXlaeTUxYXk4d01ERTFPREF3TURBeFNGRlJjbHBCUVZndk1" + - "EQXhOVGd3TURBd01VaFJVWEphUVVGWUxtcDNhM01pTENKdmNtZGZhbmRyYzE5eVpYWnZhMlZrWDJWdVpIQnZhVzUwSWpv" + - "aWFIUjBjSE02THk5clpYbHpkRzl5WlM1dmNHVnVZbUZ1YTJsdVozUmxjM1F1YjNKbkxuVnJMekF3TVRVNE1EQXdNREZJV" + - "VZGeVdrRkJXQzl5WlhadmEyVmtMekF3TVRVNE1EQXdNREZJVVZGeVdrRkJXQzVxZDJ0eklpd2ljMjltZEhkaGNtVmZhbm" + - "RyYzE5bGJtUndiMmx1ZENJNkltaDBkSEJ6T2k4dmEyVjVjM1J2Y21VdWIzQmxibUpoYm10cGJtZDBaWE4wTG05eVp5NTF" + - "heTh3TURFMU9EQXdNREF4U0ZGUmNscEJRVmd2T1dJMWRYTkVjR0pPZEcxNFJHTlVlbk0zUjNwTGNDNXFkMnR6SWl3aWMy" + - "OW1kSGRoY21WZmFuZHJjMTl5WlhadmEyVmtYMlZ1WkhCdmFXNTBJam9pYUhSMGNITTZMeTlyWlhsemRHOXlaUzV2Y0dWd" + - "VltRnVhMmx1WjNSbGMzUXViM0puTG5Wckx6QXdNVFU0TURBd01ERklVVkZ5V2tGQldDOXlaWFp2YTJWa0x6bGlOWFZ6Uk" + - "hCaVRuUnRlRVJqVkhwek4wZDZTM0F1YW5kcmN5SXNJbk52Wm5SM1lYSmxYM0J2YkdsamVWOTFjbWtpT2lKb2RIUndjem9" + - "2TDNkemJ6SXVZMjl0SWl3aWMyOW1kSGRoY21WZmRHOXpYM1Z5YVNJNkltaDBkSEJ6T2k4dmQzTnZNaTVqYjIwaUxDSnpi" + - "MlowZDJGeVpWOXZibDlpWldoaGJHWmZiMlpmYjNKbklqb2lWMU5QTWlCUGNHVnVJRUpoYm10cGJtY2lmUS5DQTE0b2dkY" + - "3BOd29IaUlKb3o2bVR4TnBNMndScnFpWkFjYm1LMFJuRHgyR0ROM0JIWW5aRzBFcTZWZ3lQYlByY1J5ZldsOGpRczJFU3" + - "NXYzVKU0J3ZWpIYnZwbng3a1ZCeVlrRzQ0ZGhvemFQQU5FWmx0Tmo0TTkxMkNnSGVLUGRfZDB1SUQ4ZElVcThfczJrWU1" + - "zb0NjY0JxR3lGVEl5bVZLMDFIWF9YXy1UN25wR19vdkU4Q0xnaWxNRmtpank1UGlGQzgzaG9weGl4ZVFmUmdkbUhDUl8x" + - "Ym9rc2JGREszUlBJRWU1UGlPRHZYOHZsV0I4aVVHeTdQR3paMGlrWEJEMGx4OXAxQUpFeVlGM3gxcENqc1NIOHRKQzVFN" + - "UNHMHhaTFFQUGtUM0FfU3BqaVVoNUVsTmROY21UUG93MkxWU3hQOVF1c040dldwRU1VTmQ5cHcifQ.kq8UsDUcb6Ee55w" + - "4U4JhiifyUB0sSiTAnobLV1bwujfS2msdUfxDHqVjyrvx4NvPd54sXg3_k1EIRHLT4vT-zUkojqtWiB_v2ndo5UqvPUrI" + - "FoqY0IQznKBfD6cLlGQ0laYqxm_GJWAEdEv_O8Ggw_z1DMiZZRHF9Oln9zZtT95JcGeJ8JCQVDkaX_AM-fZrVaixfD4iB" + - "fy-n4H6LHCy94c1DrCM9wEGr7XfHLAVNdZe2Qbyjf1sVEPukK_ccw4AYcWUo3UJQ2WIKxZL4fBmb_3Z0ez9k31k6in86H" + - "g4tHO9itXSVJvvzn8oAaYXXQrxfk4N1CojV3zk1bkhy6In3Q"; - - public static final String REQUEST_BODY_WITH_SSA_SINGLE_ROLE = "eyJ0eXAiOiJKV1QiLCJhbGciOiJQUzI1NiIsIm" + - "tpZCI6IkR3TUtkV01tajdQV2ludm9xZlF5WFZ6eVo2USJ9.eyJpc3MiOiI5YjV1c0RwYk50bXhEY1R6czdHektwIiwiaW" + - "F0IjoxNjAxOTgyMDQyLCJleHAiOjE2MDcyNTI0NDIsImp0aSI6IjE2MDE5ODIwNDYiLCJhdWQiOiJodHRwczovL2xvY2F" + - "saG9zdDo4MjQzL3Rva2VuIiwic2NvcGUiOiJhY2NvdW50cyBwYXltZW50cyIsInRva2VuX2VuZHBvaW50X2F1dGhfbWV0" + - "aG9kIjoicHJpdmF0ZV9rZXlfand0IiwiZ3JhbnRfdHlwZXMiOlsiYXV0aG9yaXphdGlvbl9jb2RlIiwicmVmcmVzaF90b" + - "2tlbiJdLCJyZXNwb25zZV90eXBlcyI6WyJjb2RlIGlkX3Rva2VuIl0sImlkX3Rva2VuX3NpZ25lZF9yZXNwb25zZV9hbG" + - "ciOiJQUzI1NiIsInJlcXVlc3Rfb2JqZWN0X3NpZ25pbmdfYWxnIjoiUFMyNTYiLCJzb2Z0d2FyZV9pZCI6IjliNXVzRHB" + - "iTnRteERjVHpzN0d6S3AiLCJhcHBsaWNhdGlvbl90eXBlIjoid2ViIiwicmVkaXJlY3RfdXJpcyI6WyJodHRwczovL3dz" + - "bzIuY29tIl0sInRva2VuX2VuZHBvaW50X2F1dGhfc2lnbmluZ19hbGciOiJQUzI1NiIsInNvZnR3YXJlX3N0YXRlbWVud" + - "CI6ImV5SmhiR2NpT2lKUVV6STFOaUlzSW10cFpDSTZJa2g2WVRsMk5XSm5SRXBqVDI1b1kxVmFOMEpOZDJKVFRGODBUbF" + - "l3WjFOR2RrbHFZVk5ZWkVNdE1XTTlJaXdpZEhsd0lqb2lTbGRVSW4wLmV5SnBjM01pT2lKUGNHVnVRbUZ1YTJsdVp5Qk1" + - "kR1FpTENKcFlYUWlPakUxT1RJek5qUTFOamdzSW1wMGFTSTZJak5rTVdJek5UazFaV1poWXpSbE16WWlMQ0p6YjJaMGQy" + - "RnlaVjlsYm5acGNtOXViV1Z1ZENJNkluTmhibVJpYjNnaUxDSnpiMlowZDJGeVpWOXRiMlJsSWpvaVZHVnpkQ0lzSW5Od" + - "lpuUjNZWEpsWDJsa0lqb2lPV0kxZFhORWNHSk9kRzE0UkdOVWVuTTNSM3BMY0NJc0luTnZablIzWVhKbFgyTnNhV1Z1ZE" + - "Y5cFpDSTZJamxpTlhWelJIQmlUblJ0ZUVSalZIcHpOMGQ2UzNBaUxDSnpiMlowZDJGeVpWOWpiR2xsYm5SZmJtRnRaU0k" + - "2SWxkVFR6SWdUM0JsYmlCQ1lXNXJhVzVuSUZSUVVDQW9VMkZ1WkdKdmVDa2lMQ0p6YjJaMGQyRnlaVjlqYkdsbGJuUmZa" + - "R1Z6WTNKcGNIUnBiMjRpT2lKVWFHbHpJRlJRVUNCSmN5QmpjbVZoZEdWa0lHWnZjaUIwWlhOMGFXNW5JSEIxY25CdmMyV" + - "npMaUFpTENKemIyWjBkMkZ5WlY5MlpYSnphVzl1SWpveExqVXNJbk52Wm5SM1lYSmxYMk5zYVdWdWRGOTFjbWtpT2lKb2" + - "RIUndjem92TDNkemJ6SXVZMjl0SWl3aWMyOW1kSGRoY21WZmNtVmthWEpsWTNSZmRYSnBjeUk2V3lKb2RIUndjem92TDN" + - "kemJ6SXVZMjl0SWwwc0luTnZablIzWVhKbFgzSnZiR1Z6SWpvaVFVbFRVQ0lzSW05eVoyRnVhWE5oZEdsdmJsOWpiMjF3" + - "WlhSbGJuUmZZWFYwYUc5eWFYUjVYMk5zWVdsdGN5STZleUpoZFhSb2IzSnBkSGxmYVdRaU9pSlBRa2RDVWlJc0luSmxaM" + - "mx6ZEhKaGRHbHZibDlwWkNJNklsVnVhMjV2ZDI0d01ERTFPREF3TURBeFNGRlJjbHBCUVZnaUxDSnpkR0YwZFhNaU9pSk" + - "JZM1JwZG1VaUxDSmhkWFJvYjNKcGMyRjBhVzl1Y3lJNlczc2liV1Z0WW1WeVgzTjBZWFJsSWpvaVIwSWlMQ0p5YjJ4bGN" + - "5STZXeUpCU1ZOUUlpd2lVRWxUVUNKZGZTeDdJbTFsYldKbGNsOXpkR0YwWlNJNklrbEZJaXdpY205c1pYTWlPbHNpUVVs" + - "VFVDSXNJbEJKVTFBaVhYMHNleUp0WlcxaVpYSmZjM1JoZEdVaU9pSk9UQ0lzSW5KdmJHVnpJanBiSWtGSlUxQWlMQ0pRU" + - "1ZOUUlsMTlYWDBzSW5OdlpuUjNZWEpsWDJ4dloyOWZkWEpwSWpvaWFIUjBjSE02THk5M2MyOHlMbU52YlM5M2MyOHlMbX" + - "B3WnlJc0ltOXlaMTl6ZEdGMGRYTWlPaUpCWTNScGRtVWlMQ0p2Y21kZmFXUWlPaUl3TURFMU9EQXdNREF4U0ZGUmNscEJ" + - "RVmdpTENKdmNtZGZibUZ0WlNJNklsZFRUeklnS0ZWTEtTQk1TVTFKVkVWRUlpd2liM0puWDJOdmJuUmhZM1J6SWpwYmV5" + - "SnVZVzFsSWpvaVZHVmphRzVwWTJGc0lpd2laVzFoYVd3aU9pSnpZV05vYVc1cGMwQjNjMjh5TG1OdmJTSXNJbkJvYjI1b" + - "Elqb2lLemswTnpjME1qYzBNemMwSWl3aWRIbHdaU0k2SWxSbFkyaHVhV05oYkNKOUxIc2libUZ0WlNJNklrSjFjMmx1Wl" + - "hOeklpd2laVzFoYVd3aU9pSnpZV05vYVc1cGMwQjNjMjh5TG1OdmJTSXNJbkJvYjI1bElqb2lLemswTnpjME1qYzBNemM" + - "wSWl3aWRIbHdaU0k2SWtKMWMybHVaWE56SW4xZExDSnZjbWRmYW5kcmMxOWxibVJ3YjJsdWRDSTZJbWgwZEhCek9pOHZh" + - "MlY1YzNSdmNtVXViM0JsYm1KaGJtdHBibWQwWlhOMExtOXlaeTUxYXk4d01ERTFPREF3TURBeFNGRlJjbHBCUVZndk1EQ" + - "XhOVGd3TURBd01VaFJVWEphUVVGWUxtcDNhM01pTENKdmNtZGZhbmRyYzE5eVpYWnZhMlZrWDJWdVpIQnZhVzUwSWpvaW" + - "FIUjBjSE02THk5clpYbHpkRzl5WlM1dmNHVnVZbUZ1YTJsdVozUmxjM1F1YjNKbkxuVnJMekF3TVRVNE1EQXdNREZJVVZ" + - "GeVdrRkJXQzl5WlhadmEyVmtMekF3TVRVNE1EQXdNREZJVVZGeVdrRkJXQzVxZDJ0eklpd2ljMjltZEhkaGNtVmZhbmRy" + - "YzE5bGJtUndiMmx1ZENJNkltaDBkSEJ6T2k4dmEyVjVjM1J2Y21VdWIzQmxibUpoYm10cGJtZDBaWE4wTG05eVp5NTFhe" + - "Th3TURFMU9EQXdNREF4U0ZGUmNscEJRVmd2T1dJMWRYTkVjR0pPZEcxNFJHTlVlbk0zUjNwTGNDNXFkMnR6SWl3aWMyOW" + - "1kSGRoY21WZmFuZHJjMTl5WlhadmEyVmtYMlZ1WkhCdmFXNTBJam9pYUhSMGNITTZMeTlyWlhsemRHOXlaUzV2Y0dWdVl" + - "tRnVhMmx1WjNSbGMzUXViM0puTG5Wckx6QXdNVFU0TURBd01ERklVVkZ5V2tGQldDOXlaWFp2YTJWa0x6bGlOWFZ6UkhC" + - "aVRuUnRlRVJqVkhwek4wZDZTM0F1YW5kcmN5SXNJbk52Wm5SM1lYSmxYM0J2YkdsamVWOTFjbWtpT2lKb2RIUndjem92T" + - "DNkemJ6SXVZMjl0SWl3aWMyOW1kSGRoY21WZmRHOXpYM1Z5YVNJNkltaDBkSEJ6T2k4dmQzTnZNaTVqYjIwaUxDSnpiMl" + - "owZDJGeVpWOXZibDlpWldoaGJHWmZiMlpmYjNKbklqb2lWMU5QTWlCUGNHVnVJRUpoYm10cGJtY2lmUS5vc1RWRlBqa1h" + - "CNnh3SDNJVE04OTVpRXNOamgyMnJjR1pSdnFQOWh6cV9TX2l6UENQNTJLQ3U0ZWJkZ1o5WnpXTmdBVkp4X3hKY2dZajhs" + - "TXV6S3VWa3RFX1M0bnl2REhsSU1sblBvRGd0UlNZcElDM05LcDVWY3QyaVd0bzhkNVpWVXZJQUhDTXpXZjllVDdGSURZQ" + - "2syUEhhWHNmblNLSVM3NWE4WEJXbUo1ckgzT0xoZ0RudFhHalduNW0wdVdYUDhmOWZ0TmNvSUdMa050bEp0cXV3S2dYcF" + - "JULWJIQVNSWUE4c0lsMnFETTUtay00TEY3S1hhZDBWR2ZfcmJxclVjWnlMbEt6bldaRmM4X1Q3bHZEbnEwQVJIYWtLaUU" + - "yZGhkUlRHTHhuX21yR2lKRXg0dVFsMTlrNWF0T0FzRlJ0X0liMjdCM1lLS3V6VGxOcDRjQ2cifQ.Z0J-sLdpWmvk5MEOE" + - "We-MwO_-pve2zo0OA-7JfG5cFrsQ1WPuvb-mjjxqSjxtER3IRONRWULmTIDnuu9FCX1oQ4e_0HAmh3AQMa2sp9n8fMwHc" + - "RxmXQcOuAfYajHV5i318xcSveOrapD6jFiqLgLiLtTK3-tJB7wDg-sdxuwSI9HSldR3sBjQDTPLOLFGEJ5jJ4bl8JkKQ6" + - "zHcFkYyxMpJw7zT13hyiWvR--0WVgP8g4yYeCobPXUNmBEI7zZMuYVlO5C6l-RKFPstaasqX81zWG3ES8TVZM8rMFKqN_" + - "QoctY4HMshtk58h2W4NP4UJZyYgsGNNb2hVI5IffCVxqIw"; - - private static final Log log = LogFactory.getLog(TestValidationUtil.class); - private static java.security.cert.X509Certificate testClientCertificate = null; - private static java.security.cert.X509Certificate testClientCertificateIssuer = null; - private static java.security.cert.X509Certificate testEidasCertificate = null; - private static java.security.cert.X509Certificate testEidasCertificateIssuer = null; - private static java.security.cert.X509Certificate expiredSelfCertificate = null; - - protected static X509Certificate getCertFromStr(String pemEncodedCert) { - byte[] decodedTransportCert = Base64.getDecoder().decode(pemEncodedCert - .replace(CertificateValidationUtils.BEGIN_CERT, "") - .replace(CertificateValidationUtils.END_CERT, "")); - - InputStream inputStream = new ByteArrayInputStream(decodedTransportCert); - X509Certificate x509Certificate = null; - try { - x509Certificate = X509Certificate.getInstance(inputStream); - } catch (CertificateException e) { - log.error("Exception occured while parsing test certificate. Caused by, ", e); - } - return x509Certificate; - } - - public static synchronized java.security.cert.X509Certificate getTestClientCertificate() - throws OpenBankingException { - if (testClientCertificate == null) { - testClientCertificate = CertificateUtils.parseCertificate(TEST_CLIENT_CERT); - } - return testClientCertificate; - } - - public static synchronized java.security.cert.X509Certificate getTestClientCertificateIssuer() - throws OpenBankingException { - if (testClientCertificateIssuer == null) { - testClientCertificateIssuer = CertificateUtils.parseCertificate(TEST_CLIENT_CERT_ISSUER); - } - return testClientCertificateIssuer; - } - - public static synchronized java.security.cert.X509Certificate getTestEidasCertificate() - throws OpenBankingException { - if (testEidasCertificate == null) { - testEidasCertificate = CertificateUtils.parseCertificate(EIDAS_CERT); - } - return testEidasCertificate; - } - - public static synchronized java.security.cert.X509Certificate getTestEidasCertificateIssuer() - throws OpenBankingException { - if (testEidasCertificateIssuer == null) { - testEidasCertificateIssuer = CertificateUtils.parseCertificate(EIDAS_CERT_ISSUER); - } - return testEidasCertificateIssuer; - } - - public static synchronized java.security.cert.X509Certificate getExpiredSelfCertificate() - throws OpenBankingException { - if (expiredSelfCertificate == null) { - expiredSelfCertificate = CertificateUtils.parseCertificate(EXPIRED_SELF_CERT); - } - return expiredSelfCertificate; - } - - public static Certificate getEmptyTestCertificate() { - - return new Certificate("X.509") { - @Override - public byte[] getEncoded() { - return new byte[0]; - } - - @Override - public void verify(PublicKey key) { - - } - - @Override - public void verify(PublicKey key, String sigProvider) { - - } - - @Override - public String toString() { - return null; - } - - @Override - public PublicKey getPublicKey() { - return null; - } - }; - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/handler/JwsResponseSignatureHandlerTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/handler/JwsResponseSignatureHandlerTest.java deleted file mode 100644 index b202442a..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/handler/JwsResponseSignatureHandlerTest.java +++ /dev/null @@ -1,163 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.handler; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.gateway.executor.exception.OpenBankingExecutorException; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import org.apache.axiom.soap.SOAPBody; -import org.apache.axiom.soap.SOAPEnvelope; -import org.apache.axis2.context.MessageContext; -import org.apache.commons.io.IOUtils; -import org.apache.synapse.commons.json.JsonUtil; -import org.apache.synapse.core.axis2.Axis2MessageContext; -import org.apache.synapse.core.axis2.Axis2Sender; -import org.apache.synapse.transport.passthru.util.RelayUtils; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -import java.util.HashMap; -import java.util.Optional; - -import static org.mockito.Mockito.doReturn; - -/** - * Test Handler class for Signing Responses. - */ -@PrepareForTest({OpenBankingConfigParser.class, GatewayUtils.class, JsonUtil.class, IOUtils.class, - RelayUtils.class, Axis2Sender.class}) -@PowerMockIgnore({"jdk.internal.reflect.*"}) -public class JwsResponseSignatureHandlerTest extends PowerMockTestCase { - - private MessageContext messageContext; - private HashMap headers = new HashMap<>(); - OpenBankingConfigParser openBankingConfigParserMock; - - @BeforeClass - public void init() { - - messageContext = Mockito.mock(MessageContext.class); - openBankingConfigParserMock = Mockito.mock(OpenBankingConfigParser.class); - headers = new HashMap<>(); - doReturn(headers).when(messageContext).getProperty(org.apache.axis2.context.MessageContext.TRANSPORT_HEADERS); - } - - @Test(priority = 1) - public void testHandleResponseOutFlow() { - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - Mockito.when(openBankingConfigParserMock.isJwsResponseSigningEnabled()).thenReturn(true); - Axis2MessageContext msgCtx = Mockito.mock(Axis2MessageContext.class); - doReturn(messageContext).when(msgCtx).getAxis2MessageContext(); - SOAPEnvelope soapEnvelope = Mockito.spy(SOAPEnvelope.class); - doReturn(soapEnvelope).when(msgCtx).getEnvelope(); - SOAPBody soapBody = Mockito.spy(SOAPBody.class); - doReturn(soapBody).when(soapEnvelope).getBody(); - doReturn("Schema validation failed").when(soapBody).toString(); - Assert.assertTrue(new JwsResponseSignatureHandler().handleResponseOutFlow(msgCtx)); - } - - @Test(priority = 1) - public void testHandleResponseOutFlowWithPayloadError() throws Exception { - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.mockStatic(GatewayUtils.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - Mockito.when(openBankingConfigParserMock.isJwsResponseSigningEnabled()).thenReturn(true); - Axis2MessageContext msgCtx = Mockito.mock(Axis2MessageContext.class); - doReturn(messageContext).when(msgCtx).getAxis2MessageContext(); - SOAPEnvelope soapEnvelope = Mockito.spy(SOAPEnvelope.class); - doReturn(soapEnvelope).when(msgCtx).getEnvelope(); - SOAPBody soapBody = Mockito.spy(SOAPBody.class); - doReturn(soapBody).when(soapEnvelope).getBody(); - doReturn("Schema validation failed").when(soapBody).toString(); - PowerMockito.doThrow(new OpenBankingException("")).when(GatewayUtils.class, - "buildMessagePayloadFromMessageContext", Mockito.any(), Mockito.anyMap()); - PowerMockito.doReturn("test").when(GatewayUtils.class, "constructJWSSignature", - "msgCtx", new HashMap<>()); - PowerMockito.doNothing().when(GatewayUtils.class, "returnSynapseHandlerJSONError", - Mockito.any(), Mockito.anyString(), Mockito.anyString()); - Assert.assertTrue(new JwsResponseSignatureHandler().handleResponseOutFlow(msgCtx)); - } - - @Test(priority = 1) - public void testHandleResponseOutFlowWithPayload() throws Exception { - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.mockStatic(GatewayUtils.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - Mockito.when(openBankingConfigParserMock.isJwsResponseSigningEnabled()).thenReturn(true); - Axis2MessageContext msgCtx = Mockito.mock(Axis2MessageContext.class); - doReturn(messageContext).when(msgCtx).getAxis2MessageContext(); - SOAPEnvelope soapEnvelope = Mockito.spy(SOAPEnvelope.class); - doReturn(soapEnvelope).when(msgCtx).getEnvelope(); - SOAPBody soapBody = Mockito.spy(SOAPBody.class); - doReturn(soapBody).when(soapEnvelope).getBody(); - doReturn("Schema validation failed").when(soapBody).toString(); - PowerMockito.doReturn(Optional.of("test")).when(GatewayUtils.class, "buildMessagePayloadFromMessageContext", - Mockito.any(), Mockito.anyMap()); - PowerMockito.doReturn("test").when(GatewayUtils.class, "constructJWSSignature", - "msgCtx", new HashMap<>()); - Assert.assertTrue(new JwsResponseSignatureHandler().handleResponseOutFlow(msgCtx)); - } - - @Test(priority = 1) - public void testGenerateJWSSignature() throws Exception { - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.mockStatic(GatewayUtils.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - Mockito.when(openBankingConfigParserMock.isJwsResponseSigningEnabled()).thenReturn(true); - PowerMockito.doReturn("test").when(GatewayUtils.class, "constructJWSSignature", - "msgCtx", new HashMap<>()); - Assert.assertNotNull(new JwsResponseSignatureHandler().generateJWSSignature(Optional.of("msgCtx"))); - } - - @Test(priority = 1) - public void testGenerateJWSSignatureWhenPayloadIsNull() throws Exception { - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.mockStatic(GatewayUtils.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - Mockito.when(openBankingConfigParserMock.isJwsResponseSigningEnabled()).thenReturn(true); - PowerMockito.doReturn("test").when(GatewayUtils.class, "constructJWSSignature", - "msgCtx", new HashMap<>()); - Assert.assertNull(new JwsResponseSignatureHandler().generateJWSSignature(Optional.of(""))); - } - - @Test(priority = 1) - public void testGenerateJWSSignatureForErrorCase() throws Exception { - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.mockStatic(GatewayUtils.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - Mockito.when(openBankingConfigParserMock.isJwsResponseSigningEnabled()).thenReturn(true); - PowerMockito.doThrow(new OpenBankingExecutorException("")).when(GatewayUtils.class, "constructJWSSignature", - "msgCtx", new HashMap<>()); - Assert.assertNull(new JwsResponseSignatureHandler().generateJWSSignature(Optional.of(""))); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/internal/TPPCertValidatorDataHolderTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/internal/TPPCertValidatorDataHolderTest.java deleted file mode 100644 index 435fa609..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/internal/TPPCertValidatorDataHolderTest.java +++ /dev/null @@ -1,71 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.internal; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import org.mockito.Mockito; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -import java.io.File; - -/** - * Test for TPP certificate validator data holder. - */ -public class TPPCertValidatorDataHolderTest { - - private TPPCertValidatorDataHolder tppCertValidatorDataHolder; - - @BeforeClass - public void init() { - System.setProperty("some.property", "property.value"); - System.setProperty("carbon.home", "."); - //CommonTestUtil.injectEnvironmentVariable("CARBON_HOME", "."); - - File file = new File("src/test/resources"); - String dummyConfigFile = file.getAbsolutePath() + "/open-banking.xml"; - OpenBankingConfigParser openBankingConfigParser = OpenBankingConfigParser.getInstance(dummyConfigFile); - - OpenBankingConfigurationService openBankingConfigurationServiceMock - = Mockito.mock(OpenBankingConfigurationService.class); - Mockito.doReturn(openBankingConfigParser.getConfiguration()) - .when(openBankingConfigurationServiceMock).getConfigurations(); - - tppCertValidatorDataHolder = TPPCertValidatorDataHolder.getInstance(); - tppCertValidatorDataHolder.setOpenBankingConfigurationService(openBankingConfigurationServiceMock); - tppCertValidatorDataHolder.initializeTPPValidationDataHolder(); - } - - @Test - public void testConfigs() { - Assert.assertEquals(tppCertValidatorDataHolder.getCertificateRevocationValidationRetryCount(), 3); - Assert.assertEquals(tppCertValidatorDataHolder.getCertificateRevocationProxyPort(), 8080); - Assert.assertEquals(tppCertValidatorDataHolder.getTppCertRevocationCacheExpiry(), 3600); - Assert.assertEquals(tppCertValidatorDataHolder.getTppValidationCacheExpiry(), 3600); - Assert.assertTrue(tppCertValidatorDataHolder.isCertificateRevocationValidationEnabled()); - Assert.assertTrue(tppCertValidatorDataHolder.isCertificateRevocationProxyEnabled()); - Assert.assertTrue(tppCertValidatorDataHolder.isTransportCertIssuerValidationEnabled()); - Assert.assertTrue(tppCertValidatorDataHolder.isPsd2RoleValidationEnabled()); - Assert.assertFalse(tppCertValidatorDataHolder.isTppValidationEnabled()); - Assert.assertNotNull(tppCertValidatorDataHolder.getCertificateRevocationValidationExcludedIssuers()); - Assert.assertNull(tppCertValidatorDataHolder.getTPPValidationServiceImpl()); - Assert.assertEquals(tppCertValidatorDataHolder.getCertificateRevocationProxyHost(), "PROXY_HOSTNAME"); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/mediator/BasicAuthMediatorTests.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/mediator/BasicAuthMediatorTests.java deleted file mode 100644 index 3991b923..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/mediator/BasicAuthMediatorTests.java +++ /dev/null @@ -1,387 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.mediator; - -import org.apache.axiom.om.OMElement; -import org.apache.axiom.soap.SOAPEnvelope; -import org.apache.axis2.AxisFault; -import org.apache.axis2.addressing.EndpointReference; -import org.apache.axis2.addressing.RelatesTo; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.synapse.ContinuationState; -import org.apache.synapse.FaultHandler; -import org.apache.synapse.Mediator; -import org.apache.synapse.MessageContext; -import org.apache.synapse.config.SynapseConfiguration; -import org.apache.synapse.core.SynapseEnvironment; -import org.apache.synapse.endpoints.Endpoint; -import org.mockito.Mockito; -import org.mockito.MockitoAnnotations; -import org.testng.Assert; -import org.testng.annotations.Test; - -import java.nio.charset.StandardCharsets; -import java.util.Base64; -import java.util.HashMap; -import java.util.Map; -import java.util.Set; -import java.util.Stack; - -import static org.mockito.Mockito.doReturn; - -/** - * Unit tests for Basic Auth Mediator. - */ -public class BasicAuthMediatorTests { - - @Test - public void testBasicAuthMediator() { - - MockitoAnnotations.initMocks(this); - BasicAuthMediator basicAuthMediator = Mockito.spy(BasicAuthMediator.class); - MessageContext messageContext = new MessageContextMock(); - doReturn("admin").when(basicAuthMediator).getAPIMConfigFromKey(Mockito.anyString()); - basicAuthMediator.mediate(messageContext); - Assert.assertTrue(StringUtils.equals((CharSequence) messageContext.getProperty("basicAuthentication"), - "Basic " + Base64.getEncoder().encodeToString("admin:admin".getBytes(StandardCharsets.UTF_8)))); - } - - // A mock class of MessageContext is created to mimic the behaviour - static class MessageContextMock implements MessageContext { - - Map properties = new HashMap<>(); - - @Override - public SynapseConfiguration getConfiguration() { - return null; - } - - @Override - public void setConfiguration(SynapseConfiguration synapseConfiguration) { - - } - - @Override - public SynapseEnvironment getEnvironment() { - return null; - } - - @Override - public void setEnvironment(SynapseEnvironment synapseEnvironment) { - - } - - @Override - public Map getContextEntries() { - return null; - } - - @Override - public void setContextEntries(Map map) { - - } - - @Override - public Mediator getMainSequence() { - return null; - } - - @Override - public Mediator getFaultSequence() { - return null; - } - - @Override - public Mediator getSequence(String s) { - return null; - } - - @Override - public OMElement getFormat(String s) { - return null; - } - - @Override - public Mediator getSequenceTemplate(String s) { - return null; - } - - @Override - public Endpoint getEndpoint(String s) { - return null; - } - - @Override - public Object getProperty(String s) { - return properties.get(s); - } - - @Override - public Object getEntry(String s) { - return null; - } - - @Override - public Object getLocalEntry(String s) { - return null; - } - - @Override - public void setProperty(String s, Object o) { - - this.properties.put(s, o); - } - - @Override - public Set getPropertyKeySet() { - return null; - } - - @Override - public SOAPEnvelope getEnvelope() { - return null; - } - - @Override - public void setEnvelope(SOAPEnvelope soapEnvelope) throws AxisFault { - - } - - @Override - public EndpointReference getFaultTo() { - return null; - } - - @Override - public void setFaultTo(EndpointReference endpointReference) { - - } - - @Override - public EndpointReference getFrom() { - return null; - } - - @Override - public void setFrom(EndpointReference endpointReference) { - - } - - @Override - public String getMessageID() { - return null; - } - - @Override - public void setMessageID(String s) { - - } - - @Override - public RelatesTo getRelatesTo() { - return null; - } - - @Override - public void setRelatesTo(RelatesTo[] relatesTos) { - - } - - @Override - public EndpointReference getReplyTo() { - return null; - } - - @Override - public void setReplyTo(EndpointReference endpointReference) { - - } - - @Override - public EndpointReference getTo() { - return null; - } - - @Override - public void setTo(EndpointReference endpointReference) { - - } - - @Override - public void setWSAAction(String s) { - - } - - @Override - public String getWSAAction() { - return null; - } - - @Override - public String getSoapAction() { - return null; - } - - @Override - public void setSoapAction(String s) { - - } - - @Override - public void setWSAMessageID(String s) { - - } - - @Override - public String getWSAMessageID() { - return null; - } - - @Override - public boolean isDoingMTOM() { - return false; - } - - @Override - public boolean isDoingSWA() { - return false; - } - - @Override - public void setDoingMTOM(boolean b) { - - } - - @Override - public void setDoingSWA(boolean b) { - - } - - @Override - public boolean isDoingPOX() { - return false; - } - - @Override - public void setDoingPOX(boolean b) { - - } - - @Override - public boolean isDoingGET() { - return false; - } - - @Override - public void setDoingGET(boolean b) { - - } - - @Override - public boolean isSOAP11() { - return false; - } - - @Override - public void setResponse(boolean b) { - - } - - @Override - public boolean isResponse() { - return false; - } - - @Override - public void setFaultResponse(boolean b) { - - } - - @Override - public boolean isFaultResponse() { - return false; - } - - @Override - public int getTracingState() { - return 0; - } - - @Override - public void setTracingState(int i) { - - } - - @Override - public Stack getFaultStack() { - return null; - } - - @Override - public void pushFaultHandler(FaultHandler faultHandler) { - - } - - @Override - public Stack getContinuationStateStack() { - return null; - } - - @Override - public void pushContinuationState(ContinuationState continuationState) { - - } - - @Override - public boolean isContinuationEnabled() { - return false; - } - - @Override - public void setContinuationEnabled(boolean b) { - - } - - @Override - public Log getServiceLog() { - return null; - } - - @Override - public Mediator getDefaultConfiguration(String s) { - return null; - } - - @Override - public String getMessageString() { - return null; - } - - @Override - public int getMessageFlowTracingState() { - return 0; - } - - @Override - public void setMessageFlowTracingState(int i) { - - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/reporter/OBAnalyticsMetricReporterTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/reporter/OBAnalyticsMetricReporterTest.java deleted file mode 100644 index 3c3b35a8..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/reporter/OBAnalyticsMetricReporterTest.java +++ /dev/null @@ -1,68 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.reporter; - -import com.wso2.openbanking.accelerator.gateway.internal.GatewayDataHolder; -import org.mockito.Mockito; -import org.testng.Assert; -import org.testng.annotations.Test; -import org.wso2.am.analytics.publisher.exception.MetricCreationException; -import org.wso2.am.analytics.publisher.exception.MetricReportingException; -import org.wso2.am.analytics.publisher.reporter.CounterMetric; - -import java.util.HashMap; -import java.util.Map; - -/** - * Test for Open Banking analytics metric reporter. - */ -public class OBAnalyticsMetricReporterTest { - - private OBAnalyticsMetricReporter obMetricReporter; - private GatewayDataHolder dataHolder; - - @Test - public void createOBMetricReporter() throws MetricCreationException { - - dataHolder = GatewayDataHolder.getInstance(); - dataHolder.setWorkerThreadCount("1"); - dataHolder.setOBDataPublishingEnabled("true"); - obMetricReporter = new OBAnalyticsMetricReporter(new HashMap<>()); - obMetricReporter.createTimer("testTimer"); - } - - @Test(dependsOnMethods = "createOBMetricReporter") - public void createCounterMetric() throws MetricCreationException, MetricReportingException { - - CounterMetric counterMetric = obMetricReporter.createCounter("testCounter", Mockito.any()); - Assert.assertEquals(counterMetric.getName(), "testCounter"); - counterMetric.incrementCount(Mockito.any()); - counterMetric.getSchema(); - } - - @Test(dependsOnMethods = "createOBMetricReporter") - public void createReporterForAPIMAnalytics() throws MetricCreationException { - - dataHolder.setAPIMAnalyticsEnabled("true"); - Map prop = new HashMap<>(); - prop.put("auth.api.token", "testToken"); - prop.put("auth.api.url", "testUrl"); - new OBAnalyticsMetricReporter(prop); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/reporter/TimestampPublishingTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/reporter/TimestampPublishingTest.java deleted file mode 100644 index 5bdc0a38..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/reporter/TimestampPublishingTest.java +++ /dev/null @@ -1,96 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.reporter; - -import org.testng.Assert; -import org.testng.annotations.Test; -import org.wso2.am.analytics.publisher.exception.MetricReportingException; -import org.wso2.am.analytics.publisher.reporter.MetricEventBuilder; -import org.wso2.am.analytics.publisher.reporter.cloud.DefaultResponseMetricEventBuilder; - -import java.sql.Timestamp; -import java.util.Date; -import java.util.Map; -import java.util.UUID; - -/** - * Test class for timestamp publishing. - */ -public class TimestampPublishingTest { - - private static final String CORRELATION_ID = "correlationId"; - private static final String REQUEST_TIMESTAMP = "requestTimestamp"; - private static final String BACKEND_LATENCY = "backendLatency"; - private static final String REQUEST_MEDIATION_LATENCY = "requestMediationLatency"; - private static final String RESPONSE_LATENCY = "responseLatency"; - private static final String RESPONSE_MEDIATION_LATENCY = "responseMediationLatency"; - private final Timestamp currentTimestamp = new Timestamp(new Date().getTime()); - - @Test - public void createMetricReporter() throws MetricReportingException { - - MetricEventBuilder metricEventBuilder = new MockEventBuilder(); - - metricEventBuilder.addAttribute(CORRELATION_ID, UUID.randomUUID()); - metricEventBuilder.addAttribute(REQUEST_TIMESTAMP, currentTimestamp); - metricEventBuilder.addAttribute(BACKEND_LATENCY, 10); - metricEventBuilder.addAttribute(REQUEST_MEDIATION_LATENCY, 20); - metricEventBuilder.addAttribute(RESPONSE_LATENCY, 30); - metricEventBuilder.addAttribute(RESPONSE_MEDIATION_LATENCY, 40); - - OBTimestampPublisher obTimestampPublisher = new MockOBTimestampPublisher(metricEventBuilder); - obTimestampPublisher.run(); - } - - private void validateData(Map analyticsData) { - Assert.assertTrue(analyticsData.get(REQUEST_TIMESTAMP).equals(currentTimestamp)); - Assert.assertTrue(analyticsData.get(BACKEND_LATENCY).equals(10)); - Assert.assertTrue(analyticsData.get(REQUEST_MEDIATION_LATENCY).equals(20)); - Assert.assertTrue(analyticsData.get(RESPONSE_LATENCY).equals(30)); - Assert.assertTrue(analyticsData.get(RESPONSE_MEDIATION_LATENCY).equals(40)); - } - - class MockEventBuilder extends DefaultResponseMetricEventBuilder { - @Override - public boolean validate() throws MetricReportingException { - return true; - } - - @Override - protected Map buildEvent() { - this.eventMap.put("eventType", "response"); - String userAgentHeader = (String) this.eventMap.remove("userAgentHeader"); - return this.eventMap; - } - } - - class MockOBTimestampPublisher extends OBTimestampPublisher { - - public MockOBTimestampPublisher(MetricEventBuilder builder) { - - super(builder); - } - - @Override - protected void publishLatencyData(Map analyticsData) { - // validate - validateData(analyticsData); - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/synapse/handler/DisputeResolutionSynapseHandlerTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/synapse/handler/DisputeResolutionSynapseHandlerTest.java deleted file mode 100644 index 8b025734..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/synapse/handler/DisputeResolutionSynapseHandlerTest.java +++ /dev/null @@ -1,193 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.synapse.handler; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.data.publisher.common.util.OBDataPublisherUtil; -import com.wso2.openbanking.accelerator.gateway.util.GatewayConstants; -import com.wso2.openbanking.accelerator.gateway.util.GatewayUtils; -import org.apache.axiom.om.OMElement; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.synapse.MessageContext; -import org.apache.synapse.commons.json.JsonUtil; -import org.apache.synapse.config.SynapseConfiguration; -import org.apache.synapse.core.SynapseEnvironment; -import org.apache.synapse.core.axis2.Axis2MessageContext; -import org.mockito.Mockito; -import org.mockito.MockitoAnnotations; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; -import org.wso2.carbon.apimgt.impl.APIConstants; - -import java.io.IOException; -import java.util.HashMap; -import java.util.Map; -import java.util.Optional; - -import javax.xml.parsers.ParserConfigurationException; - -import static org.powermock.api.mockito.PowerMockito.doNothing; -import static org.powermock.api.mockito.PowerMockito.mock; -import static org.powermock.api.mockito.PowerMockito.mockStatic; -import static org.powermock.api.mockito.PowerMockito.when; - - -/** - * Test for Dispute Resolution Synapse Handler. - */ -@PrepareForTest({OpenBankingConfigParser.class, OBDataPublisherUtil.class, JsonUtil.class, GatewayUtils.class}) -@PowerMockIgnore({"jdk.internal.reflect.*", "javax.management.*"}) -public class DisputeResolutionSynapseHandlerTest extends PowerMockTestCase { - private static final Log log = LogFactory.getLog(DisputeResolutionSynapseHandlerTest.class); - private static final String REQUEST_BODY = "requestBody"; - private Axis2MessageContext axis2MessageContextMock; - - Map sampleRequestHeaders = new HashMap<>(); - - DisputeResolutionSynapseHandler disputeResolutionSynapseHandler = new DisputeResolutionSynapseHandler(); - public DisputeResolutionSynapseHandlerTest() throws ParserConfigurationException, IOException { - - sampleRequestHeaders.put("Accept-Encoding", "gzip, deflate, br"); - sampleRequestHeaders.put("Access-Control-Allow-Headers", "authorization, Access-Control-Allow-Origin," + - " Content-Type, SOAPAction, apikey, Authorization"); - sampleRequestHeaders.put("Access-Control-Allow-Methods", "POST"); - sampleRequestHeaders.put("Access-Control-Allow-Origin", "*"); - sampleRequestHeaders.put("Access-Control-Expose-Headers", ""); - sampleRequestHeaders.put("Content-Type", "application/json"); - sampleRequestHeaders.put("WWW-Authenticate", "Internal API Key realm=\"WSO2 API Manager\"," + - " Bearer realm=\"WSO2 API Manager\", error=\"invalid_token\", error_description=\"The " + - "provided token is invalid\""); - } - - String sampleRequestBody = "{\"Data\":{\"Permissions\":[\"ReadAccountsDetail\",\"ReadTransactionsDetail\"," - + "\"ReadBalances\"]," - + "\"Risk\":{}}"; - String sampleResponseBody = "{\"code\": \"900902\"," + - "\"message\": \"Missing Credentials\"," + - "\"description\": \"Invalid Credentials." - + "Make sure your API invocation call has a header: " - + "'Authorization : Bearer ACCESS_TOKEN' or 'Authorization :" - + " Basic ACCESS_TOKEN' or 'apikey: API_KEY'\"}"; - - @BeforeClass - public void initClass() { - - MockitoAnnotations.initMocks(this); - - } - - @Test - public void testSynapseHandlerForResponseFlow() throws Exception { - - MessageContext messageContext = getResponseData(); - Assert.assertTrue(disputeResolutionSynapseHandler.handleResponseOutFlow(messageContext)); - } - @Test - public void testSynapseHandlerForRequestFlow() throws Exception { - - MessageContext messageContext = getRequestData(); - Assert.assertTrue(disputeResolutionSynapseHandler.handleRequestInFlow(messageContext)); - } - - - private MessageContext getResponseData() throws Exception { - - PowerMockito.mockStatic(GatewayUtils.class); - mockStatic(OpenBankingConfigParser.class); - OpenBankingConfigParser openBankingConfigParserMock = mock(OpenBankingConfigParser.class); - when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - Mockito.doReturn(true).when(openBankingConfigParserMock).isDisputeResolutionEnabled(); - Mockito.doReturn(true).when(openBankingConfigParserMock) - .isNonErrorDisputeDataPublishingEnabled(); - - SynapseConfiguration synapseConfigurationMock = mock(SynapseConfiguration.class); - SynapseEnvironment synapseEnvironmentMock = mock(SynapseEnvironment.class); - org.apache.axis2.context.MessageContext messageContextMock = - mock(org.apache.axis2.context.MessageContext.class); - MessageContext messageContext = new Axis2MessageContext(messageContextMock, synapseConfigurationMock, - synapseEnvironmentMock); - - messageContext.setProperty(APIConstants.API_ELECTED_RESOURCE, "/register"); - messageContext.setProperty(GatewayConstants.HTTP_METHOD, "POST"); - messageContext.setProperty(GatewayConstants.UNKNOWN, "unknown"); - - when(GatewayUtils.buildMessagePayloadFromMessageContext(Mockito.anyObject(), Mockito.anyMap())) - .thenReturn(Optional.of(sampleResponseBody)); - - Map contextEntries = new HashMap<>(); - contextEntries.put(REQUEST_BODY, sampleRequestBody); - messageContext.setContextEntries(contextEntries); - - org.apache.axis2.context.MessageContext axis2MessageContext = new org.apache.axis2.context.MessageContext(); - axis2MessageContext.setProperty(GatewayConstants.HTTP_SC, 401); - axis2MessageContext.setProperty(org.apache.axis2.context.MessageContext - .TRANSPORT_HEADERS, sampleRequestHeaders); - ((Axis2MessageContext) messageContext).setAxis2MessageContext(axis2MessageContext); - - mockStatic(OBDataPublisherUtil.class); - doNothing().when(OBDataPublisherUtil.class, "publishData", Mockito.anyString(), Mockito.anyString(), - Mockito.anyObject()); - - mockStatic(JsonUtil.class); - OMElement omElementMock = mock(OMElement.class); - when(JsonUtil.getNewJsonPayload(Mockito.anyObject(), Mockito.anyString(), Mockito.anyBoolean(), - Mockito.anyBoolean())).thenReturn(omElementMock); - - return messageContext; - } - - private MessageContext getRequestData() throws Exception { - - PowerMockito.mockStatic(GatewayUtils.class); - mockStatic(OpenBankingConfigParser.class); - OpenBankingConfigParser openBankingConfigParserMock = mock(OpenBankingConfigParser.class); - when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - Mockito.doReturn(true).when(openBankingConfigParserMock).isDisputeResolutionEnabled(); - - SynapseConfiguration synapseConfigurationMock = mock(SynapseConfiguration.class); - SynapseEnvironment synapseEnvironmentMock = mock(SynapseEnvironment.class); - org.apache.axis2.context.MessageContext messageContextMock = - mock(org.apache.axis2.context.MessageContext.class); - MessageContext messageContext = new Axis2MessageContext(messageContextMock, synapseConfigurationMock, - synapseEnvironmentMock); - - org.apache.axis2.context.MessageContext axis2MessageContext = new org.apache.axis2.context.MessageContext(); - ((Axis2MessageContext) messageContext).setAxis2MessageContext(axis2MessageContext); - axis2MessageContext.setProperty(org.apache.axis2.context.MessageContext - .TRANSPORT_HEADERS, sampleRequestHeaders); - - mockStatic(JsonUtil.class); - OMElement omElementMock = mock(OMElement.class); - when(JsonUtil.getNewJsonPayload(Mockito.anyObject(), Mockito.anyString(), Mockito.anyBoolean(), - Mockito.anyBoolean())).thenReturn(omElementMock); - - when(GatewayUtils.buildMessagePayloadFromMessageContext(Mockito.anyObject(), Mockito.anyMap())) - .thenReturn(Optional.of(sampleRequestBody)); - return messageContext; - } - - -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/throttling/OBThrottlingExtensionImplTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/throttling/OBThrottlingExtensionImplTest.java deleted file mode 100644 index 69bda5f4..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/throttling/OBThrottlingExtensionImplTest.java +++ /dev/null @@ -1,53 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.throttling; - -import com.wso2.openbanking.accelerator.gateway.internal.GatewayDataHolder; -import org.mockito.Mockito; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; -import org.wso2.carbon.apimgt.common.gateway.dto.ExtensionResponseDTO; -import org.wso2.carbon.apimgt.common.gateway.dto.RequestContextDTO; - -/** - * Test for Open Banking throttling extension implementation. - */ -public class OBThrottlingExtensionImplTest { - - OBThrottlingExtensionImpl obThrottlingExtension; - RequestContextDTO requestContextDTO; - - @BeforeClass - public void beforeClass() { - - GatewayDataHolder.getInstance().setThrottleDataPublisher(new ThrottleDataPublisherTestImpl()); - obThrottlingExtension = new OBThrottlingExtensionImpl(); - requestContextDTO = Mockito.mock(RequestContextDTO.class); - } - - @Test(priority = 1) - public void testAddCustomThrottlingKeys() { - - ExtensionResponseDTO extensionResponseDTO = obThrottlingExtension.preProcessRequest(requestContextDTO); - Assert.assertEquals(extensionResponseDTO.getCustomProperty().get("open"), "banking"); - Assert.assertEquals(extensionResponseDTO.getCustomProperty().get("wso2"), "ob"); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/throttling/ThrottleDataPublisherTestImpl.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/throttling/ThrottleDataPublisherTestImpl.java deleted file mode 100644 index bce43d59..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/throttling/ThrottleDataPublisherTestImpl.java +++ /dev/null @@ -1,39 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.throttling; - -import org.wso2.carbon.apimgt.common.gateway.dto.RequestContextDTO; - -import java.util.HashMap; -import java.util.Map; - -/** - * Test for throttle data publisher implementation. - */ -public class ThrottleDataPublisherTestImpl implements ThrottleDataPublisher { - - @Override - public Map getCustomProperties(RequestContextDTO requestContextDTO) { - - Map customPropertyMap = new HashMap<>(); - customPropertyMap.put("open", "banking"); - customPropertyMap.put("wso2", "OB"); - return customPropertyMap; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/util/GatewayUtilsTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/util/GatewayUtilsTest.java deleted file mode 100644 index 9b2b195b..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/java/com/wso2/openbanking/accelerator/gateway/util/GatewayUtilsTest.java +++ /dev/null @@ -1,195 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.gateway.util; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import org.apache.axiom.om.OMElement; -import org.apache.axiom.soap.SOAPBody; -import org.apache.axiom.soap.SOAPEnvelope; -import org.apache.axis2.AxisFault; -import org.apache.axis2.context.MessageContext; -import org.apache.commons.io.IOUtils; -import org.apache.http.HttpHeaders; -import org.apache.synapse.commons.json.JsonUtil; -import org.apache.synapse.core.axis2.Axis2MessageContext; -import org.apache.synapse.core.axis2.Axis2Sender; -import org.apache.synapse.transport.passthru.PassThroughConstants; -import org.apache.synapse.transport.passthru.util.RelayUtils; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -import java.io.ByteArrayInputStream; -import java.io.IOException; -import java.io.InputStream; -import java.nio.charset.StandardCharsets; -import java.util.HashMap; -import java.util.Optional; - -import static org.mockito.Mockito.doReturn; -import static org.mockito.Mockito.mock; -/** - * Utility methods used in gateway. - */ -@PrepareForTest({JsonUtil.class, IOUtils.class, RelayUtils.class, Axis2Sender.class}) -@PowerMockIgnore({"jdk.internal.reflect.*"}) -public class GatewayUtilsTest extends PowerMockTestCase { - - private MessageContext messageContext; - private HashMap headers = new HashMap<>(); - - @BeforeClass - public void init() { - - messageContext = Mockito.mock(MessageContext.class); - headers = new HashMap<>(); - doReturn(headers).when(messageContext).getProperty(org.apache.axis2.context.MessageContext.TRANSPORT_HEADERS); - - } - - @Test(priority = 1) - public void testBuildMessagePayloadFromMessageContextForXML() throws OpenBankingException { - - doReturn(false).when(messageContext).getProperty(PassThroughConstants.MESSAGE_BUILDER_INVOKED); - SOAPEnvelope soapEnvelope = Mockito.spy(SOAPEnvelope.class); - doReturn(soapEnvelope).when(messageContext).getEnvelope(); - SOAPBody soapBody = Mockito.spy(SOAPBody.class); - doReturn(soapBody).when(soapEnvelope).getBody(); - OMElement element = Mockito.spy(OMElement.class); - doReturn(element).when(soapBody).getFirstElement(); - headers.put(HttpHeaders.CONTENT_TYPE, GatewayConstants.APPLICATION_XML_CONTENT_TYPE); - Optional payload = GatewayUtils.buildMessagePayloadFromMessageContext(messageContext, headers); - Assert.assertTrue(payload.isPresent()); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testBuildMessageErrorPayloadFromMessageContextForXML() throws Exception { - - PowerMockito.mockStatic(RelayUtils.class); - PowerMockito.doThrow(new IOException()).when(RelayUtils.class, "buildMessage", messageContext); - - doReturn(false).when(messageContext).getProperty(PassThroughConstants.MESSAGE_BUILDER_INVOKED); - SOAPEnvelope soapEnvelope = Mockito.spy(SOAPEnvelope.class); - doReturn(soapEnvelope).when(messageContext).getEnvelope(); - SOAPBody soapBody = Mockito.spy(SOAPBody.class); - doReturn(soapBody).when(soapEnvelope).getBody(); - OMElement element = Mockito.spy(OMElement.class); - doReturn(element).when(soapBody).getFirstElement(); - headers.put(HttpHeaders.CONTENT_TYPE, GatewayConstants.APPLICATION_XML_CONTENT_TYPE); - Optional payload = GatewayUtils.buildMessagePayloadFromMessageContext(messageContext, headers); - Assert.assertTrue(payload.isPresent()); - } - - @Test(priority = 1) - public void testBuildMessagePayloadFromMessageContextForJWT() throws OpenBankingException { - - doReturn(false).when(messageContext).getProperty(PassThroughConstants.MESSAGE_BUILDER_INVOKED); - SOAPEnvelope soapEnvelope = Mockito.spy(SOAPEnvelope.class); - doReturn(soapEnvelope).when(messageContext).getEnvelope(); - SOAPBody soapBody = Mockito.spy(SOAPBody.class); - doReturn(soapBody).when(soapEnvelope).getBody(); - OMElement element = Mockito.spy(OMElement.class); - doReturn(element).when(soapBody).getFirstElement(); - headers.put(HttpHeaders.CONTENT_TYPE, GatewayConstants.JWT_CONTENT_TYPE); - Optional payload = GatewayUtils.buildMessagePayloadFromMessageContext(messageContext, headers); - Assert.assertTrue(payload.isPresent()); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testNegativeBuildMessagePayloadFromMessageContextForJSON() throws Exception { - - PowerMockito.mockStatic(JsonUtil.class); - doReturn(false).when(messageContext).getProperty(PassThroughConstants.MESSAGE_BUILDER_INVOKED); - SOAPEnvelope soapEnvelope = Mockito.spy(SOAPEnvelope.class); - doReturn(soapEnvelope).when(messageContext).getEnvelope(); - SOAPBody soapBody = Mockito.spy(SOAPBody.class); - doReturn(soapBody).when(soapEnvelope).getBody(); - OMElement element = Mockito.spy(OMElement.class); - doReturn(element).when(soapBody).getFirstElement(); - headers.put(HttpHeaders.CONTENT_TYPE, GatewayConstants.JSON_CONTENT_TYPE); - PowerMockito.when(JsonUtil.getJsonPayload(messageContext)).thenReturn(Mockito.mock(InputStream.class)); - Optional payload = GatewayUtils.buildMessagePayloadFromMessageContext(messageContext, headers); - } - - @Test(priority = 1) - public void testBuildMessagePayloadFromMessageContextForJSON() throws Exception { - - PowerMockito.mockStatic(JsonUtil.class); - PowerMockito.mockStatic(IOUtils.class); - doReturn(false).when(messageContext).getProperty(PassThroughConstants.MESSAGE_BUILDER_INVOKED); - SOAPEnvelope soapEnvelope = Mockito.spy(SOAPEnvelope.class); - doReturn(soapEnvelope).when(messageContext).getEnvelope(); - SOAPBody soapBody = Mockito.spy(SOAPBody.class); - doReturn(soapBody).when(soapEnvelope).getBody(); - OMElement element = Mockito.spy(OMElement.class); - doReturn(element).when(soapBody).getFirstElement(); - headers.put(HttpHeaders.CONTENT_TYPE, GatewayConstants.JSON_CONTENT_TYPE); - PowerMockito.when(JsonUtil.getJsonPayload(messageContext)).thenReturn( - new ByteArrayInputStream("".getBytes(StandardCharsets.UTF_8))); - Optional payload = GatewayUtils.buildMessagePayloadFromMessageContext(messageContext, headers); - Assert.assertFalse(payload.isPresent()); - } - - - @Test(priority = 1) - public void testReturnSynapseHandlerJSONError() throws Exception { - - Axis2MessageContext msgCtx = Mockito.mock(Axis2MessageContext.class); - PowerMockito.mockStatic(Axis2Sender.class); - PowerMockito.mockStatic(JsonUtil.class); - PowerMockito.doReturn(mock(OMElement.class)).when(JsonUtil.class, "getNewJsonPayload", Mockito.any(), - Mockito.anyString(), Mockito.anyBoolean(), Mockito.anyBoolean()); - PowerMockito.doNothing().when(Axis2Sender.class, "sendBack", msgCtx); - doReturn(messageContext).when(msgCtx).getAxis2MessageContext(); - GatewayUtils.returnSynapseHandlerJSONError(msgCtx, "", ""); - } - - @Test(priority = 1) - public void testReturnSynapseHandlerJSONErrorWithAxisFault() throws Exception { - - PowerMockito.mockStatic(RelayUtils.class); - PowerMockito.doThrow(new AxisFault("")).when(RelayUtils.class, "discardRequestMessage", messageContext); - PowerMockito.mockStatic(JsonUtil.class); - PowerMockito.doReturn(mock(OMElement.class)).when(JsonUtil.class, "getNewJsonPayload", Mockito.any(), - Mockito.anyString(), Mockito.anyBoolean(), Mockito.anyBoolean()); - Axis2MessageContext msgCtx = Mockito.mock(Axis2MessageContext.class); - PowerMockito.mockStatic(Axis2Sender.class); - PowerMockito.doNothing().when(Axis2Sender.class, "sendBack", msgCtx); - doReturn(messageContext).when(msgCtx).getAxis2MessageContext(); - GatewayUtils.returnSynapseHandlerJSONError(msgCtx, "", ""); - } - - @Test(priority = 1) - public void testReturnSynapseHandlerJSONErrorWithAxisFaultForJsonPayload() throws Exception { - - PowerMockito.mockStatic(JsonUtil.class); - PowerMockito.doThrow(new AxisFault("")).when(JsonUtil.class, "getNewJsonPayload", Mockito.any(), - Mockito.anyString(), Mockito.anyBoolean(), Mockito.anyBoolean()); - Axis2MessageContext msgCtx = Mockito.mock(Axis2MessageContext.class); - PowerMockito.mockStatic(Axis2Sender.class); - PowerMockito.doNothing().when(Axis2Sender.class, "sendBack", msgCtx); - doReturn(messageContext).when(msgCtx).getAxis2MessageContext(); - GatewayUtils.returnSynapseHandlerJSONError(msgCtx, "", ""); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/resources/client-truststore.jks b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/resources/client-truststore.jks deleted file mode 100644 index 4672372b6ece068c755fbe8357e9502128872692..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 104882 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 LHI;~nq)q>|2Al^3 diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/resources/open-banking.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/resources/open-banking.xml deleted file mode 100644 index 19c1d17c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/resources/open-banking.xml +++ /dev/null @@ -1,153 +0,0 @@ - - - - - - - - sampleDataSourceName - - - - - - - sampleRequestObjectValidator - - - - - - DummyValue - ${some.property} - - ${carbon.home} - - - - Nothing - Everything - Anything - - - - - - - - - - - - - - - - - - - - - - - - - false - - - - - true - - - - - - - - - - 3600 - 3600 - 86400 - - true - 3 - - - true - PROXY_HOSTNAME - 8080 - - - - - - - - - CN=Test Pre-Production Issuing CA, O=Test, C=GB - - - true - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - true - - Sample - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/resources/test_crl_entries.pem b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/resources/test_crl_entries.pem deleted file mode 100644 index 846a2b17..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/resources/test_crl_entries.pem +++ /dev/null @@ -1,1225 +0,0 @@ ------BEGIN X509 CRL----- -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 -MDExNDA3MzhaMAwwCgYDVR0VBAMKAQUwIwIEWcYYCBcNMjEwMjIzMjE0MDU3WjAM -MAoGA1UdFQQDCgEFMCMCBFnGGAcXDTIxMDIyMzIxNDAwNlowDDAKBgNVHRUEAwoB -BTAjAgRZxhgGFw0yMTAyMjUxMTIwMzdaMAwwCgYDVR0VBAMKAQUwIwIEWcYYBRcN -MjEwMjIzMjE0MjE0WjAMMAoGA1UdFQQDCgEFMCMCBFnGGAMXDTIxMDIyMzIxMjcw -OVowDDAKBgNVHRUEAwoBBTAjAgRZxhgCFw0yMTAyMjMyMTI2MTVaMAwwCgYDVR0V -BAMKAQUwIwIEWcYYARcNMjEwMjI1MTExNzA5WjAMMAoGA1UdFQQDCgEFMCMCBFnG -GAAXDTIxMDIyNTExMTcxMVowDDAKBgNVHRUEAwoBBTAjAgRZxhf+Fw0yMTAyMjUx -MTA5MDlaMAwwCgYDVR0VBAMKAQUwIwIEWcYX/RcNMjEwMjI1MTEwOTA3WjAMMAoG -A1UdFQQDCgEFMCMCBFnGF/wXDTIxMDIyNTExMTcwMFowDDAKBgNVHRUEAwoBBTAj -AgRZxhf7Fw0yMTAyMjUxMTE3MDZaMAwwCgYDVR0VBAMKAQUwIwIEWcYX+BcNMjEw -MjIzMjEwMDI2WjAMMAoGA1UdFQQDCgEFMCMCBFnGF/cXDTIxMDIyMzIwNTkzMFow -DDAKBgNVHRUEAwoBBTAjAgRZxhf2Fw0yMTAyMjUxMTI3MDhaMAwwCgYDVR0VBAMK -AQUwIwIEWcYX9RcNMjEwMjI1MTEyNzAwWjAMMAoGA1UdFQQDCgEFMCMCBFnGF/MX -DTIxMDIyMzIwNDM0MVowDDAKBgNVHRUEAwoBBTAjAgRZxhfyFw0yMTAyMjMyMDQy -NDNaMAwwCgYDVR0VBAMKAQUwIwIEWcYX8RcNMjEwMjI1MTExNjQ3WjAMMAoGA1Ud -FQQDCgEFMCMCBFnGF/AXDTIxMDIyMzIwNDQ1OVowDDAKBgNVHRUEAwoBBTAjAgRZ -xhfuFw0yMTAyMjMyMDI2MDNaMAwwCgYDVR0VBAMKAQUwIwIEWcYX7RcNMjEwMjIz -MjAyNTEwWjAMMAoGA1UdFQQDCgEFMCMCBFnGF+wXDTIxMDIyNTExMTY0MlowDDAK -BgNVHRUEAwoBBTAjAgRZxhfrFw0yMTAyMjMyMDI3MjVaMAwwCgYDVR0VBAMKAQUw -IwIEWcYXvhcNMjEwMjI1MTExNjM4WjAMMAoGA1UdFQQDCgEFMCMCBFnGF70XDTIx -MDIyNTExMTYzNFowDDAKBgNVHRUEAwoBBTAjAgRZxhdlFw0yMTAyMjUxMTE2MzFa -MAwwCgYDVR0VBAMKAQUwIwIEWcYXZBcNMjEwMjI1MTExNjI4WjAMMAoGA1UdFQQD -CgEFMCMCBFnGF14XDTIxMDIyMjA3MjUxOVowDDAKBgNVHRUEAwoBBTAjAgRZxhbT -Fw0yMTAyMTkwNjMxMjZaMAwwCgYDVR0VBAMKAQUwIwIEWcYWqRcNMjEwMjE4MTQ1 -ODA1WjAMMAoGA1UdFQQDCgEFMCMCBFnGFmcXDTIxMDIxNzEwMTEwN1owDDAKBgNV -HRUEAwoBBTAjAgRZxhWZFw0yMTAzMDExNjM3MjdaMAwwCgYDVR0VBAMKAQUwIwIE -WcYVkRcNMjEwMjEyMTE1MjAyWjAMMAoGA1UdFQQDCgEFMCMCBFnGFZAXDTIxMDIx -MjExNTAzNFowDDAKBgNVHRUEAwoBBTAjAgRZxhWOFw0yMTAyMTIxMTIyNTFaMAww -CgYDVR0VBAMKAQUwIwIEWcYVjRcNMjEwMjEyMTEyMTMwWjAMMAoGA1UdFQQDCgEF -MCMCBFnGFYwXDTIxMDIxMjExMjYyMFowDDAKBgNVHRUEAwoBBTAjAgRZxhWLFw0y -MTAyMTIxMTI0NTVaMAwwCgYDVR0VBAMKAQUwIwIEWcYVihcNMjEwMjI1MTExNjI0 -WjAMMAoGA1UdFQQDCgEFMCMCBFnGFYkXDTIxMDIyNTExMTYyMFowDDAKBgNVHRUE -AwoBBTAjAgRZxhWHFw0yMTAyMTIxMTAwMjZaMAwwCgYDVR0VBAMKAQUwIwIEWcYV -hhcNMjEwMjEyMTEwMDIzWjAMMAoGA1UdFQQDCgEFMCMCBFnGFYQXDTIxMDIyNTEx -Mzk0OFowDDAKBgNVHRUEAwoBBTAjAgRZxhWDFw0yMTAyMTIxMTAwMzJaMAwwCgYD -VR0VBAMKAQUwIwIEWcYVghcNMjEwMjEyMTEwMDI4WjAMMAoGA1UdFQQDCgEFMCMC -BFnGFYEXDTIxMDIyNTExMzk0NlowDDAKBgNVHRUEAwoBBTAjAgRZxhWAFw0yMTAy -MjUxMTE2MTZaMAwwCgYDVR0VBAMKAQUwIwIEWcYVfxcNMjEwMjI1MTExNjEzWjAM -MAoGA1UdFQQDCgEFMCMCBFnGFX4XDTIxMDIxMjEwNTM1MVowDDAKBgNVHRUEAwoB -BTAjAgRZxhV8Fw0yMTAyMTIxMDUzNDZaMAwwCgYDVR0VBAMKAQUwIwIEWcYVexcN -MjEwMjEyMTA1MzU1WjAMMAoGA1UdFQQDCgEFMCMCBFnGFXoXDTIxMDIyNTExMTYw -OFowDDAKBgNVHRUEAwoBBTAjAgRZxhV5Fw0yMTAyMTIxMDUzNTBaMAwwCgYDVR0V -BAMKAQUwIwIEWcYVeBcNMjEwMjI1MTExNjA1WjAMMAoGA1UdFQQDCgEFMCMCBFnG -FXYXDTIxMDIyNTExMzk0M1owDDAKBgNVHRUEAwoBBTAjAgRZxhV1Fw0yMTAyMTIx -MDUzNDNaMAwwCgYDVR0VBAMKAQUwIwIEWcYVdBcNMjEwMjI1MTEzOTQxWjAMMAoG -A1UdFQQDCgEFMCMCBFnGFXMXDTIxMDIyNTExMTYwMVowDDAKBgNVHRUEAwoBBTAj -AgRZxhVyFw0yMTAyMjUxMTE1NThaMAwwCgYDVR0VBAMKAQUwIwIEWcYVcRcNMjEw -MjEyMTA0MzU5WjAMMAoGA1UdFQQDCgEFMCMCBFnGFXAXDTIxMDIxMjEwNDM1NVow -DDAKBgNVHRUEAwoBBTAjAgRZxhVuFw0yMTAyMjUxMTA4NTdaMAwwCgYDVR0VBAMK -AQUwIwIEWcYVbRcNMjEwMjI1MTEwODU1WjAMMAoGA1UdFQQDCgEFMCMCBFnGFWwX -DTIxMDIyNTExMTU1NFowDDAKBgNVHRUEAwoBBTAjAgRZxhVrFw0yMTAyMjUxMTE1 -NTFaMAwwCgYDVR0VBAMKAQUwIwIEWcYVahcNMjEwMjE4MTAxMzM0WjAMMAoGA1Ud -FQQDCgEFMCMCBFnGFWkXDTIxMDIxODEwMTM0M1owDDAKBgNVHRUEAwoBBTAjAgRZ -xhVoFw0yMTAyMjUxMTE1NDdaMAwwCgYDVR0VBAMKAQUwIwIEWcYVZxcNMjEwMjI1 -MTExNTQxWjAMMAoGA1UdFQQDCgEFMCMCBFnGFWYXDTIxMDIyNTExMTUzN1owDDAK -BgNVHRUEAwoBBTAjAgRZxhUDFw0yMTAyMTAwMDU5NTFaMAwwCgYDVR0VBAMKAQUw -IwIEWcYVAhcNMjEwMjEwMDA1OTQ4WjAMMAoGA1UdFQQDCgEFMCMCBFnGFQEXDTIx -MDIxMDAxMDAwMlowDDAKBgNVHRUEAwoBBTAjAgRZxhUAFw0yMTAyMTAwMDU5NTda -MAwwCgYDVR0VBAMKAQUwIwIEWcYU/xcNMjEwMjI1MTIwMTU0WjAMMAoGA1UdFQQD -CgEFMCMCBFnGFP4XDTIxMDIyNTEyMDIwNFowDDAKBgNVHRUEAwoBBTAjAgRZxhT9 -Fw0yMTAyMTAwMDU2MjJaMAwwCgYDVR0VBAMKAQUwIwIEWcYU/BcNMjEwMjEwMDA1 -NjE2WjAMMAoGA1UdFQQDCgEFMCMCBFnGFPoXDTIxMDIxMDAwNDQyOFowDDAKBgNV -HRUEAwoBBTAjAgRZxhT5Fw0yMTAyMTAwMDQ0MjVaMAwwCgYDVR0VBAMKAQUwIwIE -WcYU+BcNMjEwMjEwMDA0NDM2WjAMMAoGA1UdFQQDCgEFMCMCBFnGFPcXDTIxMDIx -MDAwNDQzMlowDDAKBgNVHRUEAwoBBTAjAgRZxhT1Fw0yMTAyMTAwMDM3MTRaMAww -CgYDVR0VBAMKAQUwIwIEWcYU9BcNMjEwMjEwMDAzNzEyWjAMMAoGA1UdFQQDCgEF -MCMCBFnGFPMXDTIxMDIxODEwMTM1NlowDDAKBgNVHRUEAwoBBTAjAgRZxhTyFw0y -MTAyMTgxMDEzNTlaMAwwCgYDVR0VBAMKAQUwIwIEWcYU8BcNMjEwMjEwMDAyNjUx -WjAMMAoGA1UdFQQDCgEFMCMCBFnGFO8XDTIxMDIxMDAwMjY0OVowDDAKBgNVHRUE -AwoBBTAjAgRZxhTtFw0yMTAyMTcxMDE2MDFaMAwwCgYDVR0VBAMKAQUwIwIEWcYU -7BcNMjEwMjE3MTAxNTU4WjAMMAoGA1UdFQQDCgEFMCMCBFnGFOsXDTIxMDIxNzEw -MTYxMVowDDAKBgNVHRUEAwoBBTAjAgRZxhTqFw0yMTAyMTAwMDEwMjhaMAwwCgYD -VR0VBAMKAQUwIwIEWcYU6BcNMjEwMjEwMDAxMDI0WjAMMAoGA1UdFQQDCgEFMCMC -BFnGFOcXDTIxMDIxMDAwMTAzNVowDDAKBgNVHRUEAwoBBTAjAgRZxhTmFw0yMTAy -MTAwMDEwMzNaMAwwCgYDVR0VBAMKAQUwIwIEWcYU5RcNMjEwMjI1MTEwOTM1WjAM -MAoGA1UdFQQDCgEFMCMCBFnGFOQXDTIxMDIyNTExMTUzM1owDDAKBgNVHRUEAwoB -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 -IQkOUlYPYnTFIzwNzTljmNFs2bNOD+WY ------END X509 CRL----- diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/resources/testng.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/resources/testng.xml deleted file mode 100644 index 49169106..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/resources/testng.xml +++ /dev/null @@ -1,76 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/resources/wso2carbon.jks b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.gateway/src/test/resources/wso2carbon.jks deleted file mode 100644 index c8775783fb3555d6e88255c1236576e7a4a7df1f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 98874 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 o1{wExgs!Xzw`yH% - - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../pom.xml - - 4.0.0 - - com.wso2.openbanking.accelerator.identity - bundle - WSO2 Open Banking - Identity Extensions - - - org.wso2.eclipse.osgi - org.eclipse.osgi.services - - - org.eclipse.osgi - org.eclipse.osgi - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - provided - - - javax.validation - validation-api - provided - - - org.apache.tomcat - tomcat-catalina - - - org.wso2.carbon - org.wso2.carbon.logging - - - javax.ws.rs - javax.ws.rs-api - - - org.wso2.carbon - org.wso2.carbon.core - - - org.wso2.carbon.identity.framework - org.wso2.carbon.identity.application.common - - - org.wso2.carbon.identity.framework - org.wso2.carbon.identity.application.mgt - - - org.wso2.carbon.identity.framework - org.wso2.carbon.identity.core - - - org.wso2.carbon.extension.identity.oauth.addons - org.wso2.carbon.identity.oauth2.token.handler.clientauth.mutualtls - - - org.wso2.carbon.identity.inbound.auth.oauth2 - org.wso2.carbon.identity.oauth - - - org.wso2.carbon.identity.inbound.auth.oauth2 - org.wso2.carbon.identity.oauth.ciba - - - org.wso2.carbon.identity.framework - org.wso2.carbon.identity.application.authentication.framework - - - org.wso2.carbon.identity.application.auth.basic - org.wso2.carbon.identity.application.authenticator.basicauth - - - org.wso2.carbon.apimgt - org.wso2.carbon.apimgt.impl - - - org.wso2.orbit.org.apache.oltu.oauth2 - oltu - - - org.mockito - mockito-all - - - org.powermock - powermock-module-testng - - - org.powermock - powermock-api-mockito - - - org.springframework - spring-test - - - org.springframework - spring-core - - - org.hibernate - hibernate-validator - provided - - - com.fasterxml.jackson.core - jackson-databind - provided - - - org.testng - testng - test - - - org.jacoco - org.jacoco.agent - runtime - test - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.data.publisher.common - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.consent.service - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.consent.dao - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.throttler.dao - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.throttler.service - - - org.wso2.carbon.identity.outbound.auth.push - org.wso2.carbon.identity.application.authenticator.push.device.handler - - - org.wso2.carbon.identity.outbound.auth.push - org.wso2.carbon.identity.application.authenticator.push - - - org.wso2.carbon.identity.outbound.auth.push - org.wso2.carbon.identity.application.authenticator.push.common - - - - - - - org.apache.maven.plugins - maven-compiler-plugin - - - org.jacoco - jacoco-maven-plugin - - - - **/*Constants.class - **/*Component.class - **/*Enum.class - **/*Exception.class - **/*DataHolder.class - **/*ServiceComponent.class - **/*IdentityCommonUtil.class - **/*DefaultTokenFilter.class - **/object/validator/** - **/com.wso2.openbanking.accelerator.identity.dcr/validation/annotation/* - - **/validator/annotations/**/* - **/*Constants.class - **/*RegistrationValidator.class - **/*SignatureValidator.class - **/*RegistrationError.class - **/*RegistrationRequest.class - **/*RegistrationResponse.class - **/*SoftwareStatementBody.class - **/*Wrapper.class - **/*ClaimProvider.class - **/*KeyIDProvider.class - **/*IntrospectionDataProvider.class - **/*GrantHandler.class - **/*Cache* - **/*IdentityCache.class - **/*IdentityCacheKey.class - **/*OBDefaultResponseTypeHandlerImpl.class - **/*PushAuthErrorResponse.class - **/*DefaultSPMetadataFilter.class - **/*IdentityServiceExporter.class - **/*DeviceVerificationToken.class - **/wrapper/* - - - - - default-prepare-agent - - prepare-agent - - - - default-prepare-agent-integration - - prepare-agent-integration - - - - default-report - - report - - - - default-report-integration - - report-integration - - - - default-check - - check - - - - - BUNDLE - - - INSTRUCTION - COVEREDRATIO - 0.8 - - - - - - - - - - org.apache.maven.plugins - maven-surefire-plugin - - - src/test/resources/testng.xml - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - false - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-exclude.xml - ${project.basedir}/src/main/resources/findbugs-include.xml - false - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - org.apache.felix - maven-bundle-plugin - true - - - - ${project.artifactId} - - - com.wso2.openbanking.accelerator.identity.internal - - - javax.servlet;version="${imp.pkg.version.javax.servlet}", - javax.servlet.descriptor; version="${imp.pkg.version.javax.servlet}", - javax.servlet.http; version="${imp.pkg.version.javax.servlet}", - javax.validation.*;version="0", - - com.nimbusds.jwt.*;version="${nimbusds.osgi.version.range}", - com.nimbusds.jose.*;version="${nimbusds.osgi.version.range}", - net.minidev.json.*;version="${json-smart}", - com.google.gson.*;version="${gson.version}", - org.json.*;version="${org.json.version}", - org.wso2.carbon.core.*;version="${carbon.kernel.version}", - org.wso2.carbon.user.core.service.*;version="${carbon.kernel.version}", - - org.wso2.carbon.identity.application.authentication.framework.*; - version="${carbon.identity.framework.version.range}", - org.wso2.carbon.identity.application.common.*;version="${carbon.identity.framework.version.range}", - org.wso2.carbon.identity.application.mgt.*;version="${carbon.identity.framework.version.range}", - org.wso2.carbon.identity.core.*;version="${carbon.identity.framework.version.range}", - org.wso2.carbon.identity.oauth2.keyidprovider;version="${identity.inbound.auth.oauth.version.range}", - org.wso2.carbon.identity.oauth2.token.handler.clientauth.mutualtls.*;version="${carbon.identity.clientauth.mutualtls.version}", - org.wso2.carbon.identity.openidconnect.*;version="${identity.inbound.auth.oauth.version.range}", - org.wso2.carbon.identity.oauth2.*;version="${identity.inbound.auth.oauth.version.range}", - org.wso2.carbon.identity.oauth2.model.*;version="${identity.inbound.auth.oauth.version.range}", - org.wso2.carbon.identity.oauth.*;version="${identity.inbound.auth.oauth.version.range}", - - com.wso2.openbanking.accelerator.common.*;version="${project.version}", - com.wso2.openbanking.accelerator.data.publisher.common.*;version="${project.version}", - com.wso2.openbanking.accelerator.consent.mgt.service.impl.*;version="${project.version}", - com.wso2.openbanking.accelerator.consent.mgt.dao.models.*, - com.wso2.openbanking.accelerator.throttler.service.*;version="${project.version}", - - org.apache.commons.lang3;version="${commons-lang.version}", - org.apache.commons.beanutils.*;version="0", - org.apache.commons.logging.*;version="${commons.logging.version}", - org.osgi.framework;version="${osgi.framework.imp.pkg.version.range}", - org.osgi.service.component;version="${osgi.service.component.imp.pkg.version.range}" - - - !com.wso2.openbanking.accelerator.identity.internal, - com.wso2.openbanking.accelerator.identity.*;version="${project.version}", - com.wso2.openbanking.accelerator.identity.authenticator.*;version="${project.version}", - com.wso2.openbanking.accelerator.identity.authenticator.OBIdentifierAuthenticator;version="${project.version}", - - * - <_dsannotations>* - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthenticator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthenticator.java deleted file mode 100644 index beb348c7..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthenticator.java +++ /dev/null @@ -1,248 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app; - -import com.nimbusds.jwt.SignedJWT; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.identity.app2app.exception.JWTValidationException; -import com.wso2.openbanking.accelerator.identity.app2app.model.DeviceVerificationToken; -import com.wso2.openbanking.accelerator.identity.app2app.utils.App2AppAuthUtils; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.application.authentication.framework.AbstractApplicationAuthenticator; -import org.wso2.carbon.identity.application.authentication.framework.FederatedApplicationAuthenticator; -import org.wso2.carbon.identity.application.authentication.framework.context.AuthenticationContext; -import org.wso2.carbon.identity.application.authentication.framework.exception.AuthenticationFailedException; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedUser; -import org.wso2.carbon.identity.application.authenticator.push.device.handler.DeviceHandler; -import org.wso2.carbon.identity.application.authenticator.push.device.handler.exception.PushDeviceHandlerClientException; -import org.wso2.carbon.identity.application.authenticator.push.device.handler.exception.PushDeviceHandlerServerException; -import org.wso2.carbon.identity.application.authenticator.push.device.handler.impl.DeviceHandlerImpl; -import org.wso2.carbon.user.api.UserRealm; -import org.wso2.carbon.user.api.UserStoreException; - -import java.text.ParseException; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; - -/** - * App2App authenticator for authenticating users from native auth attempt. - */ -public class App2AppAuthenticator extends AbstractApplicationAuthenticator - implements FederatedApplicationAuthenticator { - - private static final Log log = LogFactory.getLog(App2AppAuthenticator.class); - private static final long serialVersionUID = -5439464372188473141L; - private static DeviceHandler deviceHandler; - - /** - * Constructor for the App2AppAuthenticator. - */ - public App2AppAuthenticator() { - - if (deviceHandler == null) { - deviceHandler = new DeviceHandlerImpl(); - } - } - - /** - * This method is used to get authenticator name. - * - * @return String Authenticator name. - */ - @Override - public String getName() { - - return App2AppAuthenticatorConstants.AUTHENTICATOR_NAME; - } - - /** - * This method is used to get the friendly name of the authenticator. - * - * @return String Friendly name of the authenticator - */ - @Override - public String getFriendlyName() { - - return App2AppAuthenticatorConstants.AUTHENTICATOR_FRIENDLY_NAME; - } - - /** - * This method processes the authentication response received from the client. - * It verifies the authenticity of the received JWT token, extracts necessary information, - * and performs validations before authenticating the user. - * - * @param httpServletRequest The HTTP servlet request object containing the authentication response. - * @param httpServletResponse The HTTP servlet response object for sending responses. - * @param authenticationContext The authentication context containing information related to the authentication - * process. - * @throws AuthenticationFailedException If authentication fails due to various reasons such as missing parameters, - * parsing errors, JWT validation errors, or exceptions during authentication process. - */ - @Override - protected void processAuthenticationResponse(HttpServletRequest httpServletRequest, - HttpServletResponse httpServletResponse, - AuthenticationContext authenticationContext) - throws AuthenticationFailedException { - - authenticationContext.setCurrentAuthenticator(App2AppAuthenticatorConstants.AUTHENTICATOR_NAME); - String jwtString = - httpServletRequest.getParameter(App2AppAuthenticatorConstants.DEVICE_VERIFICATION_TOKEN_IDENTIFIER); - String request = - httpServletRequest.getParameter(App2AppAuthenticatorConstants.REQUEST); - - try { - SignedJWT signedJWT = JWTUtils.getSignedJWT(jwtString); - DeviceVerificationToken deviceVerificationToken = new DeviceVerificationToken(signedJWT); - //Extracting deviceId and loginHint is necessary to retrieve the public key - String loginHint = deviceVerificationToken.getLoginHint(); - String deviceID = deviceVerificationToken.getDeviceId(); - - //Checking whether deviceId and loginHint present in passed jwt - if (StringUtils.isBlank(loginHint) || StringUtils.isBlank(deviceID)) { - if (log.isDebugEnabled()) { - log.debug(App2AppAuthenticatorConstants.REQUIRED_PARAMS_MISSING_MESSAGE); - } - throw new AuthenticationFailedException(App2AppAuthenticatorConstants.REQUIRED_PARAMS_MISSING_MESSAGE); - } - - AuthenticatedUser userToBeAuthenticated = - App2AppAuthUtils.getAuthenticatedUserFromSubjectIdentifier(loginHint); - String publicKey = getPublicKeyByDeviceID(deviceID, userToBeAuthenticated); - deviceVerificationToken.setPublicKey(publicKey); - deviceVerificationToken.setRequestObject(request); - // setting the user is mandatory for data publishing purposes - //If exception is thrown before setting a user data publishing will encounter exceptions - authenticationContext.setSubject(userToBeAuthenticated); - /* - if validations are failed it will throw a JWTValidationException and flow will be interrupted. - Hence, user Authentication will fail. - */ - App2AppAuthUtils.validateToken(deviceVerificationToken); - //If the flow is not interrupted user will be authenticated. - if (log.isDebugEnabled()) { - log.debug(String.format(App2AppAuthenticatorConstants.USER_AUTHENTICATED_MSG, - userToBeAuthenticated.getUserName())); - } - } catch (ParseException e) { - log.error(e.getMessage()); - throw new AuthenticationFailedException(App2AppAuthenticatorConstants.PARSE_EXCEPTION_MESSAGE, e); - } catch (JWTValidationException e) { - log.error(e.getMessage()); - throw new AuthenticationFailedException - (App2AppAuthenticatorConstants.APP_AUTH_IDENTIFIER_VALIDATION_EXCEPTION_MESSAGE, e); - } catch (OpenBankingException e) { - log.error(e.getMessage()); - throw new AuthenticationFailedException(App2AppAuthenticatorConstants.OPEN_BANKING_EXCEPTION_MESSAGE, e); - } catch (PushDeviceHandlerServerException e) { - log.error(e.getMessage()); - throw new AuthenticationFailedException - (App2AppAuthenticatorConstants.PUSH_DEVICE_HANDLER_SERVER_EXCEPTION_MESSAGE, e); - } catch (UserStoreException e) { - log.error(e.getMessage()); - throw new AuthenticationFailedException(App2AppAuthenticatorConstants.USER_STORE_EXCEPTION_MESSAGE, e); - } catch (PushDeviceHandlerClientException e) { - log.error(e.getMessage()); - throw new AuthenticationFailedException - (App2AppAuthenticatorConstants.PUSH_DEVICE_HANDLER_CLIENT_EXCEPTION_MESSAGE, e); - } catch (IllegalArgumentException e) { - log.error(e.getMessage()); - throw new - AuthenticationFailedException(App2AppAuthenticatorConstants.ILLEGAL_ARGUMENT_EXCEPTION_MESSAGE, e); - } - } - - /** - * Determines whether this authenticator can handle the incoming HTTP servlet request. - * This method checks if the request contains the necessary parameter for App2App authentication, - * which is the device verification token identifier. - * - * @param httpServletRequest The HTTP servlet request object to be checked for handling. - * @return True if this authenticator can handle the request, false otherwise. - */ - @Override - public boolean canHandle(HttpServletRequest httpServletRequest) { - - /* - App2App authenticates the user in one step depending on the app_auth_key, - Hence it's mandatory to have the required parameter app_auth_key. - */ - return StringUtils.isNotBlank(httpServletRequest.getParameter( - App2AppAuthenticatorConstants.DEVICE_VERIFICATION_TOKEN_IDENTIFIER)); - } - - /** - * Retrieves the context identifier(sessionDataKey in this case) from the HTTP servlet request. - * - * @param request The HTTP servlet request object from which to retrieve the context identifier. - * @return The context identifier extracted from the request, typically representing session data key. - */ - @Override - public String getContextIdentifier(HttpServletRequest request) { - - return request.getParameter(App2AppAuthenticatorConstants.SESSION_DATA_KEY); - } - - /** - * Initiates the authentication request, but App2App authenticator does not support this operation. - * Therefore, this method terminates the authentication process and throws an AuthenticationFailedException. - * - * @param request The HTTP servlet request object. - * @param response The HTTP servlet response object. - * @param context The authentication context. - * @throws AuthenticationFailedException if this method is called - */ - @Override - protected void initiateAuthenticationRequest(HttpServletRequest request, - HttpServletResponse response, - AuthenticationContext context) - throws AuthenticationFailedException { - - /* - App2App authenticator does not support initiating authentication request, - Hence authentication process will be terminated. - */ - log.error(App2AppAuthenticatorConstants.INITIALIZATION_ERROR_MESSAGE); - throw new AuthenticationFailedException( - App2AppAuthenticatorConstants.DEVICE_VERIFICATION_TOKEN_MISSING_ERROR_MESSAGE); - } - - /** - * Retrieves the public key associated with a device and user. - * - * @param deviceID The identifier of the device for which the public key is requested. - * @param authenticatedUser the authenticated user for this request - * @return The public key associated with the specified device and user. - * @throws UserStoreException If an error occurs while accessing user store. - * @throws PushDeviceHandlerServerException If an error occurs on the server side of the push device handler. - * @throws PushDeviceHandlerClientException If an error occurs on the client side of the push device handler. - */ - private String getPublicKeyByDeviceID(String deviceID, AuthenticatedUser authenticatedUser) - throws UserStoreException, PushDeviceHandlerServerException, PushDeviceHandlerClientException, - OpenBankingException { - - UserRealm userRealm = App2AppAuthUtils.getUserRealm(authenticatedUser); - String userID = App2AppAuthUtils.getUserIdFromUsername(authenticatedUser.getUserName(), userRealm); - return App2AppAuthUtils.getPublicKey(deviceID, userID, deviceHandler); - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthenticatorConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthenticatorConstants.java deleted file mode 100644 index b00f9c5a..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthenticatorConstants.java +++ /dev/null @@ -1,51 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app; - -/** - * Constants related with App2App Authenticator process. - */ -public class App2AppAuthenticatorConstants { - - public static final String AUTHENTICATOR_NAME = "app2app"; - public static final String AUTHENTICATOR_FRIENDLY_NAME = "App2App Authenticator"; - public static final String REQUEST = "request"; - public static final String DEVICE_VERIFICATION_TOKEN_IDENTIFIER = "deviceVerificationToken"; - public static final String SESSION_DATA_KEY = "sessionDataKey"; - public static final String APP_AUTH_IDENTIFIER_VALIDATION_EXCEPTION_MESSAGE - = "Error while validating device verification token."; - public static final String ILLEGAL_ARGUMENT_EXCEPTION_MESSAGE - = "Error while creating user for provided login_hint."; - public static final String PARSE_EXCEPTION_MESSAGE = "Error while parsing the provided JWT."; - public static final String PUSH_DEVICE_HANDLER_SERVER_EXCEPTION_MESSAGE - = "Error occurred in push device handler service."; - public static final String USER_STORE_EXCEPTION_MESSAGE = "Error while creating authenticated user."; - public static final String PUSH_DEVICE_HANDLER_CLIENT_EXCEPTION_MESSAGE - = "Error occurred in Push Device handler client."; - public static final String INITIALIZATION_ERROR_MESSAGE = "Initializing App2App authenticator is not supported."; - public static final String DEVICE_VERIFICATION_TOKEN_MISSING_ERROR_MESSAGE - = "Device verification token null or empty in request."; - public static final String USER_AUTHENTICATED_MSG - = "User {%s} authenticated by app2app authenticator successfully."; - public static final String OPEN_BANKING_EXCEPTION_MESSAGE - = "Error while retrieving user."; - public static final String REQUIRED_PARAMS_MISSING_MESSAGE - = "Required Parameters did or loginHint null or empty."; -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/cache/JTICache.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/cache/JTICache.java deleted file mode 100644 index b32a8796..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/cache/JTICache.java +++ /dev/null @@ -1,73 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app.cache; - -import com.wso2.openbanking.accelerator.identity.cache.IdentityCache; -import com.wso2.openbanking.accelerator.identity.cache.IdentityCacheKey; - -/** - * Class for maintaining JTI cache. - */ -public class JTICache { - - private static volatile IdentityCache jtiCacheInstance; - - /** - * Get JTI cache instance. - * - * @return IdentityCache instance as JTICache - */ - public static IdentityCache getInstance() { - - //Outer null check avoids entering synchronized block when jtiCache is not null. - if (jtiCacheInstance == null) { - // Synchronize access to ensure thread safety - synchronized (JTICache.class) { - // Avoids race condition within threads - if (jtiCacheInstance == null) { - jtiCacheInstance = new IdentityCache(); - } - } - } - - return jtiCacheInstance; - } - - /** - * Adds the provided JTI (JSON Web Token ID) to the cache for efficient retrieval and management. - * - * @param jti The JTI (JSON Web Token ID) to be added to the cache. - */ - public static void addJtiDataToCache(String jti) { - - JTICache.getInstance().addToCache(IdentityCacheKey.of(jti), jti); - } - - /** - * Retrieves the data associated with the provided JTI (JSON Web Token ID) from the cache. - * - * @param jti The JTI (JSON Web Token ID) for which data is to be retrieved from the cache. - * @return The data associated with the provided JTI if found in the cache, otherwise null. - */ - public static Object getJtiDataFromCache(String jti) { - - return JTICache.getInstance().getFromCache(IdentityCacheKey.of(jti)); - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/exception/JWTValidationException.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/exception/JWTValidationException.java deleted file mode 100644 index e11dea70..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/exception/JWTValidationException.java +++ /dev/null @@ -1,38 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app.exception; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; - -/** - * DeviceVerificationToken Object Validation Exception. - */ -public class JWTValidationException extends OpenBankingException { - - private static final long serialVersionUID = -2572459527308720228L; - - public JWTValidationException(String message) { - super(message); - } - - public JWTValidationException(String message, Throwable e) { - super(message, e); - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/model/DeviceVerificationToken.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/model/DeviceVerificationToken.java deleted file mode 100644 index f977d161..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/model/DeviceVerificationToken.java +++ /dev/null @@ -1,165 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app.model; - -import com.google.gson.annotations.SerializedName; -import com.nimbusds.jwt.JWTClaimsSet; -import com.nimbusds.jwt.SignedJWT; -import com.wso2.openbanking.accelerator.identity.app2app.validations.annotations.ValidateDigest; -import com.wso2.openbanking.accelerator.identity.app2app.validations.annotations.ValidateExpiry; -import com.wso2.openbanking.accelerator.identity.app2app.validations.annotations.ValidateJTI; -import com.wso2.openbanking.accelerator.identity.app2app.validations.annotations.ValidateNBF; -import com.wso2.openbanking.accelerator.identity.app2app.validations.annotations.ValidateSignature; -import com.wso2.openbanking.accelerator.identity.common.annotations.validationgroups.MandatoryChecks; -import com.wso2.openbanking.accelerator.identity.common.annotations.validationgroups.SignatureCheck; -import com.wso2.openbanking.accelerator.identity.common.annotations.validationgroups.ValidityChecks; - -import java.text.ParseException; -import java.util.Date; - -import javax.validation.constraints.NotBlank; -import javax.validation.constraints.NotNull; - -/** - * Model class for App2App Auth DeviceVerificationToken. - */ -@ValidateJTI(groups = ValidityChecks.class) -@ValidateSignature(groups = SignatureCheck.class) -@ValidateExpiry(groups = ValidityChecks.class) -@ValidateNBF(groups = ValidityChecks.class) -@ValidateDigest(groups = ValidityChecks.class) -public class DeviceVerificationToken { - - @SerializedName(DeviceVerificationTokenConstants.DEVICE_IDENTIFIER) - private String deviceId; - @SerializedName(DeviceVerificationTokenConstants.LOGIN_HINT) - private String loginHint; - @SerializedName(DeviceVerificationTokenConstants.EXPIRY_TIME) - private Date expirationTime; - @SerializedName(DeviceVerificationTokenConstants.NOT_VALID_BEFORE) - private Date notValidBefore; - @SerializedName(DeviceVerificationTokenConstants.JWT_ID) - private String jti; - @SerializedName(DeviceVerificationTokenConstants.ISSUED_TIME) - private Date issuedTime; - @SerializedName(DeviceVerificationTokenConstants.DIGEST) - private String digest; - private SignedJWT signedJWT; - private String publicKey; - private String requestObject; - - public DeviceVerificationToken(SignedJWT signedJWT) - throws ParseException { - - this.signedJWT = signedJWT; - JWTClaimsSet jwtClaimsSet = signedJWT.getJWTClaimsSet(); - this.expirationTime = jwtClaimsSet.getExpirationTime(); - this.notValidBefore = jwtClaimsSet.getNotBeforeTime(); - this.issuedTime = jwtClaimsSet.getIssueTime(); - this.jti = jwtClaimsSet.getJWTID(); - this.deviceId = getClaim(jwtClaimsSet, DeviceVerificationTokenConstants.DEVICE_IDENTIFIER); - this.loginHint = getClaim(jwtClaimsSet, DeviceVerificationTokenConstants.LOGIN_HINT); - this.digest = getClaim(jwtClaimsSet, DeviceVerificationTokenConstants.DIGEST); - } - - @NotBlank(message = "Required parameter did cannot be null or empty.", groups = MandatoryChecks.class) - public String getDeviceId() { - - return deviceId; - } - - @NotBlank(message = "Required parameter loginHint cannot be null or empty.", groups = MandatoryChecks.class) - public String getLoginHint() { - - return loginHint; - } - - @NotNull(message = "Required parameter exp cannot be null.", groups = MandatoryChecks.class) - public Date getExpirationTime() { - - return expirationTime; - } - - @NotNull(message = "Required parameter nbf cannot be null.", groups = MandatoryChecks.class) - public Date getNotValidBefore() { - - return notValidBefore; - } - - @NotBlank(message = "Required parameter jti cannot be null or empty.", groups = MandatoryChecks.class) - public String getJti() { - - return jti; - } - - @NotNull(message = "Required parameter iat cannot be null.", groups = MandatoryChecks.class) - public Date getIssuedTime() { - - return issuedTime; - } - - @NotNull(message = "Required parameter signedJWT cannot be null.", groups = MandatoryChecks.class) - public SignedJWT getSignedJWT() { - - return signedJWT; - } - - public void setSignedJWT(SignedJWT signedJWT) { - - this.signedJWT = signedJWT; - } - - @NotBlank(message = "Required parameter public key cannot be null or empty.", groups = MandatoryChecks.class) - public String getPublicKey() { - - return publicKey; - } - - public void setPublicKey(String publicKey) { - - this.publicKey = publicKey; - } - - public String getDigest() { - - return this.digest; - } - - /** - * Retrieves the value of the specified claim from the provided JWTClaimsSet. - * - * @param jwtClaimsSet the JWTClaimsSet from which to retrieve the claim value - * @param claim the name of the claim to retrieve - * @return the value of the specified claim, or null if the claim is not present - */ - private String getClaim(JWTClaimsSet jwtClaimsSet , String claim) { - - Object claimObj = jwtClaimsSet.getClaim(claim); - return (String) claimObj; - } - - public String getRequestObject() { - return requestObject; - } - - public void setRequestObject(String requestObject) { - this.requestObject = requestObject; - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/model/DeviceVerificationTokenConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/model/DeviceVerificationTokenConstants.java deleted file mode 100644 index c5c81664..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/model/DeviceVerificationTokenConstants.java +++ /dev/null @@ -1,34 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app.model; - -/** - * Constants for DeviceVerificationToken. - */ -public class DeviceVerificationTokenConstants { - - public static final String EXPIRY_TIME = "exp"; - public static final String NOT_VALID_BEFORE = "nbf"; - public static final String LOGIN_HINT = "login_hint"; - public static final String ISSUED_TIME = "ist"; - public static final String DEVICE_IDENTIFIER = "did"; - public static final String JWT_ID = "jti"; - public static final String DIGEST = "digest"; -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/utils/App2AppAuthUtils.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/utils/App2AppAuthUtils.java deleted file mode 100644 index ec1efd4f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/utils/App2AppAuthUtils.java +++ /dev/null @@ -1,149 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app.utils; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.validator.OpenBankingValidator; -import com.wso2.openbanking.accelerator.identity.app2app.exception.JWTValidationException; -import com.wso2.openbanking.accelerator.identity.app2app.model.DeviceVerificationToken; -import com.wso2.openbanking.accelerator.identity.app2app.validations.validationorder.App2AppValidationOrder; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import org.apache.commons.lang.StringUtils; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedUser; -import org.wso2.carbon.identity.application.authenticator.push.device.handler.DeviceHandler; -import org.wso2.carbon.identity.application.authenticator.push.device.handler.exception.PushDeviceHandlerClientException; -import org.wso2.carbon.identity.application.authenticator.push.device.handler.exception.PushDeviceHandlerServerException; -import org.wso2.carbon.identity.application.authenticator.push.device.handler.model.Device; -import org.wso2.carbon.identity.core.util.IdentityTenantUtil; -import org.wso2.carbon.user.api.UserRealm; -import org.wso2.carbon.user.api.UserStoreException; -import org.wso2.carbon.user.core.common.AbstractUserStoreManager; -import org.wso2.carbon.user.core.service.RealmService; - -import java.util.List; - -/** - * Utils class for Authentication related logic implementations. - */ -public class App2AppAuthUtils { - - /** - * Retrieves an authenticated user object based on the provided subject identifier. - * - * @param subjectIdentifier the subject identifier used to retrieve the authenticated user - * @return an AuthenticatedUser object representing the authenticated user - */ - public static AuthenticatedUser getAuthenticatedUserFromSubjectIdentifier(String subjectIdentifier) { - - return AuthenticatedUser.createLocalAuthenticatedUserFromSubjectIdentifier(subjectIdentifier); - } - - /** - * Retrieves the user realm associated with the provided authenticated user. - * - * @param authenticatedUser the authenticated user for whom to retrieve the user realm - * @return the user realm associated with the authenticated user, or null if the user is not authenticated - * @throws UserStoreException if an error occurs while retrieving the user realm - */ - public static UserRealm getUserRealm(AuthenticatedUser authenticatedUser) throws UserStoreException { - - UserRealm userRealm = null; - - if (authenticatedUser != null) { - String tenantDomain = authenticatedUser.getTenantDomain(); - int tenantId = IdentityTenantUtil.getTenantId(tenantDomain); - RealmService realmService = IdentityExtensionsDataHolder.getInstance().getRealmService(); - userRealm = realmService.getTenantUserRealm(tenantId); - } - - return userRealm; - } - - /** - * Retrieves the user ID associated with the provided username from the specified user realm. - * - * @param username the username for which to retrieve the user ID - * @param userRealm the user realm from which to retrieve the user ID - * @return the user ID associated with the username - * @throws UserStoreException if an error occurs while retrieving the user ID - */ - public static String getUserIdFromUsername(String username, UserRealm userRealm) throws UserStoreException, - OpenBankingException { - - if (userRealm != null) { - AbstractUserStoreManager userStoreManager = (AbstractUserStoreManager) userRealm.getUserStoreManager(); - return userStoreManager.getUserIDFromUserName(username); - } else { - throw new OpenBankingException("UserRealm service can not be null."); - } - } - - /** - * Retrieve Public key of the device specified if it is registered under specified user. - * TODO: Optimise this code to retrieve device by did and validate userID. - * Github issue :{...} - * - * @param deviceId deviceId of the device where the public key is required - * @param userId userId of the user - * @return the public key of the intended device. - * @throws PushDeviceHandlerServerException if an error occurs on the server side while handling the device - * @throws IllegalArgumentException if the provided device identifier does not exist - * @throws PushDeviceHandlerClientException if an error occurs on the client side while handling the device - */ - public static String getPublicKey(String deviceId, String userId, DeviceHandler deviceHandler) - throws PushDeviceHandlerServerException, IllegalArgumentException, PushDeviceHandlerClientException, - OpenBankingException { - - /* - It is important to verify the device is registered under the given user - as public key is associated with device not the user. - */ - List deviceList = deviceHandler.listDevices(userId); - //If none of the devices registered under the given user matches the specified deviceId then throw a exception - deviceList.stream() - .filter(registredDevice -> StringUtils.equals(registredDevice.getDeviceId(), deviceId)) - .findFirst() - .orElseThrow(() -> - new OpenBankingException("Provided Device ID doesn't match any device registered under user.")); - //If a device is found retrieve and return the public key - return deviceHandler.getPublicKey(deviceId); - } - - /** - * Validator util to validate DeviceVerificationToken model for given validationOrder. - * - * @param deviceVerificationToken DeviceVerificationToken object that needs to be validated - * @throws JWTValidationException if validation failed - */ - public static void validateToken(DeviceVerificationToken deviceVerificationToken) throws JWTValidationException { - /* - App2AppValidationOrder validation order - 1.Required Params validation - 2.Validity Validations - Signature, JTI, Timeliness, Digest will be validated. - */ - String error = OpenBankingValidator.getInstance() - .getFirstViolation(deviceVerificationToken, App2AppValidationOrder.class); - - //if there is a validation violation convert it to JWTValidationException - if (error != null) { - throw new JWTValidationException(error); - } - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/DigestValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/DigestValidator.java deleted file mode 100644 index 6b9c0e4c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/DigestValidator.java +++ /dev/null @@ -1,105 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app.validations; - -import com.wso2.openbanking.accelerator.identity.app2app.model.DeviceVerificationToken; -import com.wso2.openbanking.accelerator.identity.app2app.validations.annotations.ValidateDigest; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.nio.charset.StandardCharsets; -import java.security.MessageDigest; -import java.security.NoSuchAlgorithmException; -import java.util.Base64; - -import javax.validation.ConstraintValidator; -import javax.validation.ConstraintValidatorContext; - -/** - * Validator class for validating digest of a device verification token. - * Digest here is expected to be the hash of the request object if it is present. - */ -public class DigestValidator implements ConstraintValidator { - - private static final Log log = LogFactory.getLog(DigestValidator.class); - - /** - * Checks if the given device verification token is valid based on its digest. - * - * @param deviceVerificationToken The device verification token to be validated. - * @param constraintValidatorContext The context in which the validation is performed. - * @return true if the token is valid, false otherwise. - */ - @Override - public boolean isValid(DeviceVerificationToken deviceVerificationToken, - ConstraintValidatorContext constraintValidatorContext) { - - String requestObject = deviceVerificationToken.getRequestObject(); - String digest = deviceVerificationToken.getDigest(); - return isDigestValid(digest, requestObject); - } - - /** - * Validating the digest of the requestObject. - * Digest is expected to be the hash of requestObject if request Object is not null. - * - * @param digest digest sent in the device verification token - * @param requestObject JWT String of the request object - * @return return true if the digest validation is a success, false otherwise - */ - protected boolean isDigestValid(String digest, String requestObject) { - - if (StringUtils.isBlank(requestObject)) { - //If the request is null nothing to validate. - return true; - } else if (StringUtils.isBlank(digest)) { - //If request is not empty and digest us empty validation fails. - return false; - } - - try { - // Example : SHA-256=EkH8fPgZ2TY2XGns8c5Vvce8h3DB83V+w47zHiyYfiQ= - String[] digestAttribute = digest.split("=", 2); - - if (digestAttribute.length != 2) { - log.error("Invalid digest."); - return false; - } - // Example : SHA-256 - String digestAlgorithm = digestAttribute[0].trim(); - String digestValue = digestAttribute[1].trim(); - MessageDigest messageDigest = MessageDigest.getInstance(digestAlgorithm); - byte[] digestHash = messageDigest.digest(requestObject.getBytes(StandardCharsets.UTF_8)); - String generatedDigest = Base64.getEncoder() - .encodeToString(digestHash); - - if (generatedDigest.equals(digestValue)) { - return true; - } - - } catch (NoSuchAlgorithmException e) { - log.error("Invalid algorithm.", e); - return false; - } - - return false; - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/ExpiryValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/ExpiryValidator.java deleted file mode 100644 index 078fb847..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/ExpiryValidator.java +++ /dev/null @@ -1,52 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app.validations; - -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.identity.app2app.model.DeviceVerificationToken; -import com.wso2.openbanking.accelerator.identity.app2app.validations.annotations.ValidateExpiry; - -import java.util.Date; - -import javax.validation.ConstraintValidator; -import javax.validation.ConstraintValidatorContext; - -/** - * Validator class for validating expiry of a device verification token. - */ -public class ExpiryValidator implements ConstraintValidator { - - private static final long DEFAULT_TIME_SKEW_IN_SECONDS = 300L; - - /** - * Checks if the given device verification token is valid based on its expiration time. - * - * @param deviceVerificationToken The device verification token to be validated. - * @param constraintValidatorContext The context in which the validation is performed. - * @return true if the token is valid, false otherwise. - */ - @Override - public boolean isValid(DeviceVerificationToken deviceVerificationToken, - ConstraintValidatorContext constraintValidatorContext) { - - Date expiryTime = deviceVerificationToken.getExpirationTime(); - return JWTUtils.isValidExpiryTime(expiryTime, DEFAULT_TIME_SKEW_IN_SECONDS); - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/JTIValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/JTIValidator.java deleted file mode 100644 index 960d67fd..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/JTIValidator.java +++ /dev/null @@ -1,69 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app.validations; - -import com.wso2.openbanking.accelerator.identity.app2app.cache.JTICache; -import com.wso2.openbanking.accelerator.identity.app2app.model.DeviceVerificationToken; -import com.wso2.openbanking.accelerator.identity.app2app.validations.annotations.ValidateJTI; - -import javax.validation.ConstraintValidator; -import javax.validation.ConstraintValidatorContext; - -/** - * Validator class for validating the JWT ID of a device verification token. - */ -public class JTIValidator implements ConstraintValidator { - - /** - * Checks if the given device verification token is valid based on its JTI value. - * - * @param deviceVerificationToken The device verification token to be validated. - * @param constraintValidatorContext The context in which the validation is performed. - * @return true if the token is valid, false otherwise. - */ - @Override - public boolean isValid(DeviceVerificationToken deviceVerificationToken, - ConstraintValidatorContext constraintValidatorContext) { - - String jti = deviceVerificationToken.getJti(); - return validateJTI(jti); - } - - private boolean validateJTI(String jti) { - - if (getFromCache(jti) != null) { - return false; - } - - //adding to cache to prevent the value from being replayed again - addToCache(jti); - return true; - } - - private Object getFromCache(String jti) { - - return JTICache.getJtiDataFromCache(jti); - } - - private void addToCache(String jti) { - - JTICache.addJtiDataToCache(jti); - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/NBFValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/NBFValidator.java deleted file mode 100644 index ff7b6528..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/NBFValidator.java +++ /dev/null @@ -1,55 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app.validations; - -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.identity.app2app.model.DeviceVerificationToken; -import com.wso2.openbanking.accelerator.identity.app2app.validations.annotations.ValidateNBF; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.Date; - -import javax.validation.ConstraintValidator; -import javax.validation.ConstraintValidatorContext; - -/** - * Validation class for validating NBF of a device verification token. - */ -public class NBFValidator implements ConstraintValidator { - - private static final long DEFAULT_TIME_SKEW_IN_SECONDS = 300L; - private static final Log log = LogFactory.getLog(NBFValidator.class); - - /** - * Checks if the given device verification token is valid based on its nbf time. - * - * @param deviceVerificationToken The device verification token to be validated. - * @param constraintValidatorContext The context in which the validation is performed. - * @return true if the token is valid, false otherwise. - */ - @Override - public boolean isValid(DeviceVerificationToken deviceVerificationToken, - ConstraintValidatorContext constraintValidatorContext) { - - Date notValidBefore = deviceVerificationToken.getNotValidBefore(); - return JWTUtils.isValidNotValidBeforeTime(notValidBefore, DEFAULT_TIME_SKEW_IN_SECONDS); - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/PublicKeySignatureValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/PublicKeySignatureValidator.java deleted file mode 100644 index 6ea59823..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/PublicKeySignatureValidator.java +++ /dev/null @@ -1,77 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app.validations; - -import com.nimbusds.jose.JOSEException; -import com.nimbusds.jwt.SignedJWT; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.identity.app2app.model.DeviceVerificationToken; -import com.wso2.openbanking.accelerator.identity.app2app.validations.annotations.ValidateSignature; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.security.NoSuchAlgorithmException; -import java.security.spec.InvalidKeySpecException; - -import javax.validation.ConstraintValidator; -import javax.validation.ConstraintValidatorContext; - -/** - * Validator class for validating the signature of a device verification token. - */ -public class PublicKeySignatureValidator implements ConstraintValidator { - - private static final Log log = LogFactory.getLog(PublicKeySignatureValidator.class); - - /** - * Checks if the given device verification token is valid based on its signature. - * - * @param deviceVerificationToken The device verification token to be validated. - * @param constraintValidatorContext The context in which the validation is performed. - * @return true if the token is valid, false otherwise. - */ - @Override - public boolean isValid(DeviceVerificationToken deviceVerificationToken, - ConstraintValidatorContext constraintValidatorContext) { - - SignedJWT signedJWT = deviceVerificationToken.getSignedJWT(); - String publicKey = deviceVerificationToken.getPublicKey(); - - try { - if (!JWTUtils.isValidSignature(signedJWT, publicKey)) { - log.error("Signature can't be verified with registered public key."); - return false; - } - } catch (NoSuchAlgorithmException e) { - log.error("No such algorithm found.", e); - return false; - } catch (InvalidKeySpecException e) { - log.error("Invalid key spec.", e); - return false; - } catch (JOSEException e) { - log.error("JOSE exception.", e); - return false; - } catch (OpenBankingException e) { - log.error("Algorithm not supported yet.", e); - } - return true; - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateDigest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateDigest.java deleted file mode 100644 index d50243c8..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateDigest.java +++ /dev/null @@ -1,48 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app.validations.annotations; - -import com.wso2.openbanking.accelerator.identity.app2app.validations.DigestValidator; - -import java.lang.annotation.Documented; -import java.lang.annotation.Retention; -import java.lang.annotation.Target; - -import javax.validation.Constraint; -import javax.validation.Payload; - -import static java.lang.annotation.ElementType.TYPE; -import static java.lang.annotation.RetentionPolicy.RUNTIME; - -/** - * Annotation class for validating digest. - */ -@Target(TYPE) -@Retention(RUNTIME) -@Documented -@Constraint(validatedBy = {DigestValidator.class}) -public @interface ValidateDigest { - - String message() default "Digest validation failed."; - - Class[] groups() default {}; - - Class[] payload() default {}; -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateExpiry.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateExpiry.java deleted file mode 100644 index 44134d4e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateExpiry.java +++ /dev/null @@ -1,48 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app.validations.annotations; - -import com.wso2.openbanking.accelerator.identity.app2app.validations.ExpiryValidator; - -import java.lang.annotation.Documented; -import java.lang.annotation.Retention; -import java.lang.annotation.Target; - -import javax.validation.Constraint; -import javax.validation.Payload; - -import static java.lang.annotation.ElementType.TYPE; -import static java.lang.annotation.RetentionPolicy.RUNTIME; - -/** - * Annotation class for validating expiry of a device verification token. - */ -@Target(TYPE) -@Retention(RUNTIME) -@Documented -@Constraint(validatedBy = {ExpiryValidator.class}) -public @interface ValidateExpiry { - - String message() default "JWT token is expired."; - - Class[] groups() default {}; - - Class[] payload() default {}; -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateJTI.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateJTI.java deleted file mode 100644 index 24e33d67..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateJTI.java +++ /dev/null @@ -1,48 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app.validations.annotations; - -import com.wso2.openbanking.accelerator.identity.app2app.validations.JTIValidator; - -import java.lang.annotation.Documented; -import java.lang.annotation.Retention; -import java.lang.annotation.Target; - -import javax.validation.Constraint; -import javax.validation.Payload; - -import static java.lang.annotation.ElementType.TYPE; -import static java.lang.annotation.RetentionPolicy.RUNTIME; - -/** - * Annotation class for validating JWT ID of a device verification token. - */ -@Target(TYPE) -@Retention(RUNTIME) -@Documented -@Constraint(validatedBy = {JTIValidator.class}) -public @interface ValidateJTI { - - String message() default "JTI has been replayed"; - - Class[] groups() default {}; - - Class[] payload() default {}; -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateNBF.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateNBF.java deleted file mode 100644 index c5c6ad02..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateNBF.java +++ /dev/null @@ -1,48 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app.validations.annotations; - -import com.wso2.openbanking.accelerator.identity.app2app.validations.NBFValidator; - -import java.lang.annotation.Documented; -import java.lang.annotation.Retention; -import java.lang.annotation.Target; - -import javax.validation.Constraint; -import javax.validation.Payload; - -import static java.lang.annotation.ElementType.TYPE; -import static java.lang.annotation.RetentionPolicy.RUNTIME; - -/** - * Annotation class for validating NBF of a device verification token. - */ -@Target(TYPE) -@Retention(RUNTIME) -@Documented -@Constraint(validatedBy = {NBFValidator.class}) -public @interface ValidateNBF { - - String message() default "JWT token is not active."; - - Class[] groups() default {}; - - Class[] payload() default {}; -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateSignature.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateSignature.java deleted file mode 100644 index beaeb7a8..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/annotations/ValidateSignature.java +++ /dev/null @@ -1,48 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app.validations.annotations; - -import com.wso2.openbanking.accelerator.identity.app2app.validations.PublicKeySignatureValidator; - -import java.lang.annotation.Documented; -import java.lang.annotation.Retention; -import java.lang.annotation.Target; - -import javax.validation.Constraint; -import javax.validation.Payload; - -import static java.lang.annotation.ElementType.TYPE; -import static java.lang.annotation.RetentionPolicy.RUNTIME; - -/** - * Annotation class for validating JWT Signature of a device verification token. - */ -@Target(TYPE) -@Retention(RUNTIME) -@Documented -@Constraint(validatedBy = {PublicKeySignatureValidator.class}) -public @interface ValidateSignature { - - String message() default "Signature validation Failed."; - - Class[] groups() default {}; - - Class[] payload() default {}; - -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/validationorder/App2AppValidationOrder.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/validationorder/App2AppValidationOrder.java deleted file mode 100644 index 5619320b..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/app2app/validations/validationorder/App2AppValidationOrder.java +++ /dev/null @@ -1,34 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app.validations.validationorder; - -import com.wso2.openbanking.accelerator.identity.common.annotations.validationgroups.MandatoryChecks; -import com.wso2.openbanking.accelerator.identity.common.annotations.validationgroups.SignatureCheck; -import com.wso2.openbanking.accelerator.identity.common.annotations.validationgroups.ValidityChecks; - -import javax.validation.GroupSequence; - -/** - * Class to define the order of execution for the hibernate validation groups. - */ -@GroupSequence({SignatureCheck.class , MandatoryChecks.class, ValidityChecks.class}) -public interface App2AppValidationOrder { - -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/adaptive/function/OpenBankingAuthenticationWorker.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/adaptive/function/OpenBankingAuthenticationWorker.java deleted file mode 100644 index ee7ce39b..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/adaptive/function/OpenBankingAuthenticationWorker.java +++ /dev/null @@ -1,33 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.adaptive.function; - -import org.json.JSONObject; -import org.wso2.carbon.identity.application.authentication.framework.config.model.graph.js.JsAuthenticationContext; - -import java.util.Map; - -/** - * Interface for Open Banking Authentication Worker. - */ -@FunctionalInterface -public interface OpenBankingAuthenticationWorker { - - JSONObject handleRequest(JsAuthenticationContext context, Map properties); -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/adaptive/function/OpenBankingAuthenticationWorkerFunction.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/adaptive/function/OpenBankingAuthenticationWorkerFunction.java deleted file mode 100644 index c5559d39..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/adaptive/function/OpenBankingAuthenticationWorkerFunction.java +++ /dev/null @@ -1,34 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.adaptive.function; - - -import org.json.JSONObject; -import org.wso2.carbon.identity.application.authentication.framework.config.model.graph.js.JsAuthenticationContext; - -import java.util.Map; - -/** - * Functional Interface for OpenBankingAuthWorker. - */ -@FunctionalInterface -public interface OpenBankingAuthenticationWorkerFunction { - - JSONObject handleRequest(JsAuthenticationContext context, Map properties, String workerName); -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/adaptive/function/OpenBankingAuthenticationWorkerFunctionImpl.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/adaptive/function/OpenBankingAuthenticationWorkerFunctionImpl.java deleted file mode 100644 index fa882adb..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/adaptive/function/OpenBankingAuthenticationWorkerFunctionImpl.java +++ /dev/null @@ -1,50 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.adaptive.function; - -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.json.JSONObject; -import org.wso2.carbon.identity.application.authentication.framework.config.model.graph.js.JsAuthenticationContext; - -import java.util.Map; - -/** - * Implementation of OpenBankingAuthenticationWorkerFunction. - */ -public class OpenBankingAuthenticationWorkerFunctionImpl implements OpenBankingAuthenticationWorkerFunction { - - private static final Log log = LogFactory.getLog(OpenBankingAuthenticationWorkerFunctionImpl.class); - - - @Override - public JSONObject handleRequest(JsAuthenticationContext context, - Map properties, String workerName) { - - Map workers = - IdentityExtensionsDataHolder.getInstance().getWorkers(); - if (workers.containsKey(workerName)) { - return workers.get(workerName).handleRequest(context, properties); - } else { - log.error("Failed to find a worker for the requested name : " + workerName); - } - return new JSONObject().put("Error", "Failed to find a worker for the requested name : " + workerName); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/authz/request/OBOAuthAuthzRequest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/authz/request/OBOAuthAuthzRequest.java deleted file mode 100644 index 9e5b6699..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/authz/request/OBOAuthAuthzRequest.java +++ /dev/null @@ -1,148 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.authz.request; - -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.oltu.oauth2.as.request.OAuthAuthzRequest; -import org.apache.oltu.oauth2.common.OAuth; -import org.apache.oltu.oauth2.common.exception.OAuthProblemException; -import org.apache.oltu.oauth2.common.exception.OAuthSystemException; -import org.apache.oltu.oauth2.common.utils.OAuthUtils; -import org.apache.oltu.oauth2.common.validators.OAuthValidator; -import org.json.JSONObject; -import org.wso2.carbon.identity.oauth.config.OAuthServerConfiguration; - -import java.nio.charset.StandardCharsets; -import java.util.Base64; -import java.util.Set; - -import javax.servlet.http.HttpServletRequest; - -import static com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants.OAUTH2_INVALID_REQUEST_MESSAGE; -import static com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants.REQUEST; -import static com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants.REQUEST_URI; - -/** - * OB OAuth 2 authorization request for request_uri support. - */ -public class OBOAuthAuthzRequest extends OAuthAuthzRequest { - - private static final Log log = LogFactory.getLog(OBOAuthAuthzRequest.class); - - public OBOAuthAuthzRequest(HttpServletRequest request) throws OAuthSystemException, OAuthProblemException { - - super(request); - } - - @Override - protected OAuthValidator initValidator() throws OAuthProblemException, OAuthSystemException { - - String responseTypeValue = getParam(OAuth.OAUTH_RESPONSE_TYPE); - - // Check if request object reference is present. - if (OAuthUtils.isEmpty(responseTypeValue) && request.getParameterMap().containsKey(REQUEST_URI)) { - responseTypeValue = IdentityCommonUtil.decodeRequestObjectAndGetKey(request, OAuth.OAUTH_RESPONSE_TYPE); - } - if (OAuthUtils.isEmpty(responseTypeValue)) { - throw IdentityCommonUtil.handleOAuthProblemException(OAUTH2_INVALID_REQUEST_MESSAGE, - "Missing response_type parameter value", getState()); - } - - Class> oauthValidatorClass = OAuthServerConfiguration.getInstance() - .getSupportedResponseTypeValidators().get(responseTypeValue); - - if (oauthValidatorClass == null) { - if (log.isDebugEnabled()) { - - //Do not change this log format as these logs use by external applications - log.debug("Unsupported Response Type : " + responseTypeValue + - " for client id : " + getClientId()); - } - throw IdentityCommonUtil.handleOAuthProblemException(OAUTH2_INVALID_REQUEST_MESSAGE, - "Invalid response_type parameter value", getState()); - } - - return OAuthUtils.instantiateClass(oauthValidatorClass); - } - - @Override - public Set getScopes() { - - if (request.getParameterMap().containsKey(REQUEST_URI) && request.getParameter(REQUEST_URI) != null) { - try { - return OAuthUtils.decodeScopes(IdentityCommonUtil - .decodeRequestObjectAndGetKey(request, OAuth.OAUTH_SCOPE)); - } catch (OAuthProblemException e) { - log.error("Invalid request URI", e); - return null; - } - } else { - return super.getScopes(); - } - } - - @Override - public String getResponseType() { - - if (request.getParameterMap().containsKey(REQUEST_URI) && request.getParameter(REQUEST_URI) != null) { - try { - return IdentityCommonUtil.decodeRequestObjectAndGetKey(request, OAuth.OAUTH_RESPONSE_TYPE); - } catch (OAuthProblemException e) { - log.error("Invalid request URI", e); - return null; - } - } else { - return super.getResponseType(); - } - - } - - @Override - public String getState() { - - if (request.getParameterMap().containsKey(REQUEST_URI) && request.getParameter(REQUEST_URI) != null) { - try { - return IdentityCommonUtil.decodeRequestObjectAndGetKey(request, OAuth.OAUTH_STATE); - } catch (OAuthProblemException e) { - log.error("Invalid request URI", e); - return null; - } - } else { - - //retrieve state value if present inside request body. - if (StringUtils.isNotBlank(getParam(REQUEST))) { - byte[] requestObject; - try { - requestObject = Base64.getDecoder().decode(getParam(REQUEST).split("\\.")[1]); - } catch (IllegalArgumentException e) { - - // Decode if the requestObject is base64-url encoded. - requestObject = Base64.getUrlDecoder().decode(getParam(REQUEST).split("\\.")[1]); - } - JSONObject requestObjectJson = new JSONObject(new String(requestObject, StandardCharsets.UTF_8)); - return requestObjectJson.has(OAuth.OAUTH_STATE) ? requestObjectJson.getString(OAuth.OAUTH_STATE) : null; - } - return null; - } - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/DefaultOBRequestObjectValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/DefaultOBRequestObjectValidator.java deleted file mode 100644 index 52e52edc..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/DefaultOBRequestObjectValidator.java +++ /dev/null @@ -1,93 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator; - -import com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator.models.OBRequestObject; -import com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator.models.ValidationResponse; -import net.minidev.json.JSONObject; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.oauth2.RequestObjectException; - -import java.util.Map; - -/** - * The extension class for enforcing OB Request Object Validations. For Tool kits to extend. - */ -public class DefaultOBRequestObjectValidator extends OBRequestObjectValidator { - - private static final String CLAIMS = "claims"; - private static final String[] CLAIM_FIELDS = new String[]{"id_token", "userinfo"}; - private static final String OPENBANKING_INTENT_ID = "openbanking_intent_id"; - private static final String VALUE = "value"; - private static final String CLIENT_ID = "client_id"; - private static final String SCOPE = "scope"; - - private static final Log log = LogFactory.getLog(DefaultOBRequestObjectValidator.class); - - public DefaultOBRequestObjectValidator() { - } - - /** - * Extension point for tool kits. Perform validation and return the error message if any, else null. - * - * @param obRequestObject request object - * @param dataMap provides scope related data needed for validation from service provider meta data - * @return the response object with error message. - */ - @Override - public ValidationResponse validateOBConstraints(OBRequestObject obRequestObject, Map dataMap) { - - ValidationResponse superValidationResponse = super.validateOBConstraints(obRequestObject, dataMap); - - if (superValidationResponse.isValid()) { - try { - if (isClientIdAndScopePresent(obRequestObject)) { - // consent id and client id is matching - return new ValidationResponse(true); - } - return new ValidationResponse(false, - "Consent Id in the request does not match with the client Id"); - } catch (RequestObjectException e) { - return new ValidationResponse(false, e.getMessage()); - } - } else { - return superValidationResponse; - } - } - - /** - * Extract clientId and scope from ob request object and check whether it's present. - * - * @param obRequestObject - * @return result received from validateConsentIdWithClientId method - * @throws RequestObjectException if error occurred while validating - */ - private boolean isClientIdAndScopePresent(OBRequestObject obRequestObject) throws RequestObjectException { - JSONObject jsonObject = obRequestObject.getSignedJWT().getPayload().toJSONObject(); - final String clientId = jsonObject.getAsString(CLIENT_ID); - String scope = jsonObject.getAsString(SCOPE); - if (StringUtils.isBlank(clientId) || StringUtils.isBlank(scope)) { - log.error("Client id or scope cannot be empty"); - throw new RequestObjectException("Client id or scope cannot be empty"); - } - return true; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/OBRequestObjectValidationExtension.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/OBRequestObjectValidationExtension.java deleted file mode 100644 index bd6b4d54..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/OBRequestObjectValidationExtension.java +++ /dev/null @@ -1,151 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator.models.OBRequestObject; -import com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator.models.ValidationResponse; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonHelper; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.oauth2.RequestObjectException; -import org.wso2.carbon.identity.oauth2.model.OAuth2Parameters; -import org.wso2.carbon.identity.openidconnect.RequestObjectValidatorImpl; -import org.wso2.carbon.identity.openidconnect.model.RequestObject; - -import java.util.Arrays; -import java.util.HashMap; -import java.util.Map; - -/** - * The extension of RequestObjectValidatorImpl to enforce Open Banking specific validations of the - * request object. - */ -public class OBRequestObjectValidationExtension extends RequestObjectValidatorImpl { - - private static final Log log = LogFactory.getLog(OBRequestObjectValidationExtension.class); - // Get extension impl - static OBRequestObjectValidator obDefaultRequestObjectValidator = - IdentityExtensionsDataHolder.getInstance().getObRequestObjectValidator(); - - /** - * Validations related to clientId, response type, exp, redirect URL, mandatory params, - * issuer, audience are done. Called after signature validation. - * - * @param initialRequestObject request object - * @param oAuth2Parameters oAuth2Parameters - * @throws RequestObjectException - RequestObjectException - */ - @Override - public boolean validateRequestObject(RequestObject initialRequestObject, OAuth2Parameters oAuth2Parameters) - throws RequestObjectException { - - try { - if (isRegulatory(oAuth2Parameters)) { - - OBRequestObject obRequestObject = new OBRequestObject(initialRequestObject); - - Map dataMap = new HashMap<>(); - final String allowedScopes = getAllowedScopes(oAuth2Parameters); - if (StringUtils.isNotBlank(allowedScopes)) { - dataMap.put(IdentityCommonConstants.SCOPE, Arrays.asList(allowedScopes.split(" "))); - } - // perform OB customized validations - ValidationResponse validationResponse = obDefaultRequestObjectValidator - .validateOBConstraints(obRequestObject, dataMap); - - if (!validationResponse.isValid()) { - // Exception will be caught and converted to auth error by IS at endpoint. - throw new RequestObjectException(RequestObjectException.ERROR_CODE_INVALID_REQUEST, - validationResponse.getViolationMessage()); - } - } - return validateIAMConstraints(initialRequestObject, oAuth2Parameters); - - } catch (OpenBankingException e) { - log.error("Error while retrieving regulatory property from sp metadata", e); - throw new RequestObjectException(RequestObjectException.ERROR_CODE_INVALID_REQUEST, "Error while " + - "retrieving regulatory property from sp metadata"); - } - } - - /** - * Validate IAM related logic. - * @param requestObject - * @param oAuth2Parameters - * @return - * @throws RequestObjectException - */ - @Generated(message = "super methods cannot be mocked") - boolean validateIAMConstraints(RequestObject requestObject, - OAuth2Parameters oAuth2Parameters) throws RequestObjectException { - - return super.validateRequestObject(requestObject, oAuth2Parameters); - } - - - /** - * Called by validateRequestObject. - * - * @param requestObject - * @param oAuth2Parameters - * @return - */ - @Generated(message = "Empty method") - @Override - protected boolean isValidAudience(RequestObject requestObject, OAuth2Parameters oAuth2Parameters) { - - // converted to validation layer - return true; - } - - /** - * Called by validateRequestObject. - * - * @param oAuth2Parameters - * @return - */ - @Generated(message = "Excluding from code coverage since it requires a service call") - protected String getAllowedScopes(OAuth2Parameters oAuth2Parameters) throws RequestObjectException { - - try { - return new IdentityCommonHelper() - .getAppPropertyFromSPMetaData(oAuth2Parameters.getClientId(), IdentityCommonConstants.SCOPE); - } catch (OpenBankingException e) { - throw new RequestObjectException(e.getMessage(), e); - } - } - - /** - * Get regulatory property from sp metadata. - * - * @param oAuth2Parameters oAuthParameters - * @return - */ - @Generated(message = "Excluding from code coverage since it requires a service call") - protected boolean isRegulatory(OAuth2Parameters oAuth2Parameters) throws OpenBankingException { - - return IdentityCommonUtil.getRegulatoryFromSPMetaData(oAuth2Parameters.getClientId()); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/OBRequestObjectValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/OBRequestObjectValidator.java deleted file mode 100644 index 6576baf4..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/OBRequestObjectValidator.java +++ /dev/null @@ -1,51 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator; - -import com.wso2.openbanking.accelerator.common.validator.OpenBankingValidator; -import com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator.models.OBRequestObject; -import com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator.models.ValidationResponse; -import org.apache.commons.lang.StringUtils; - -import java.util.Map; - -/** - * The extension class for enforcing OB Request Object Validations. For Tool kits to extend. - */ -public class OBRequestObjectValidator { - - /** - * Extension point for tool kits. Perform validation and return the error message if any, else null. - * - * @param obRequestObject request object - * @param dataMap provides scope related data needed for validation from service provider meta data - * @return the response object with error message. - */ - public ValidationResponse validateOBConstraints(OBRequestObject obRequestObject, Map dataMap) { - - String violation = OpenBankingValidator.getInstance().getFirstViolation(obRequestObject); - - if (StringUtils.isEmpty(violation)) { - return new ValidationResponse(true); - } else { - return new ValidationResponse(false, violation); - } - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/annotations/SigningAlgorithmValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/annotations/SigningAlgorithmValidator.java deleted file mode 100644 index 50ebb847..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/annotations/SigningAlgorithmValidator.java +++ /dev/null @@ -1,88 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator.annotations; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonHelper; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.beanutils.BeanUtils; -import org.apache.commons.beanutils.NestedNullException; -import org.apache.commons.beanutils.PropertyUtilsBean; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.lang.reflect.InvocationTargetException; - -import javax.validation.ConstraintValidator; -import javax.validation.ConstraintValidatorContext; - - -/** - * To validate if the signing algorithm used to sign request object - * is the same as the algorithm given during registration. - */ -public class SigningAlgorithmValidator implements ConstraintValidator { - - private String algorithmXpath; - private String clientIdXPath; - private static Log log = LogFactory.getLog(SigningAlgorithmValidator.class); - - @Override - public void initialize(ValidSigningAlgorithm constraintAnnotation) { - - this.algorithmXpath = constraintAnnotation.algorithm(); - this.clientIdXPath = constraintAnnotation.clientId(); - } - - @Override - public boolean isValid(Object object, ConstraintValidatorContext constraintValidatorContext) { - - try { - final String algorithm = new PropertyUtilsBean().getProperty(object, algorithmXpath).toString(); - final String clientId = BeanUtils.getProperty(object, clientIdXPath); - - return algorithmValidate(algorithm, clientId); - - } catch (IllegalAccessException | InvocationTargetException | NoSuchMethodException | NestedNullException e) { - log.error("Error while resolving validation fields", e); - return false; - } - } - - boolean algorithmValidate(String requestedAlgo, String clientId) { - - try { - if (!(StringUtils.isNotEmpty(new IdentityCommonHelper().getCertificateContent(clientId)) - && IdentityCommonUtil.getRegulatoryFromSPMetaData(clientId))) { - String registeredAlgo = new IdentityCommonHelper().getAppPropertyFromSPMetaData( - clientId, IdentityCommonConstants.REQUEST_OBJECT_SIGNING_ALG); - - return requestedAlgo.equals(registeredAlgo); - } else { - return true; - } - } catch (OpenBankingException e) { - log.error("Error while getting signing SP metadata", e); - } - - return false; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/annotations/ValidSigningAlgorithm.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/annotations/ValidSigningAlgorithm.java deleted file mode 100644 index 0410400b..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/annotations/ValidSigningAlgorithm.java +++ /dev/null @@ -1,49 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator.annotations; - -import java.lang.annotation.Documented; -import java.lang.annotation.ElementType; -import java.lang.annotation.Retention; -import java.lang.annotation.Target; - -import javax.validation.Constraint; -import javax.validation.Payload; - -import static java.lang.annotation.RetentionPolicy.RUNTIME; - -/** - * An annotation to execute signing algorithm validation. - */ -@Target(ElementType.TYPE) -@Retention(RUNTIME) -@Documented -@Constraint(validatedBy = {SigningAlgorithmValidator.class}) -public @interface ValidSigningAlgorithm { - - String message() default "Invalid signing algorithm used"; - - Class[] groups() default {}; - - Class[] payload() default {}; - - String algorithm() default "alg"; - - String clientId() default "clientId"; -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/models/OBRequestObject.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/models/OBRequestObject.java deleted file mode 100644 index 2bd05183..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/models/OBRequestObject.java +++ /dev/null @@ -1,142 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator.models; - -import com.nimbusds.jwt.JWTClaimsSet; -import com.nimbusds.jwt.PlainJWT; -import com.nimbusds.jwt.SignedJWT; -import com.wso2.openbanking.accelerator.common.validator.annotation.RequiredParameter; -import com.wso2.openbanking.accelerator.common.validator.annotation.RequiredParameters; -import com.wso2.openbanking.accelerator.common.validator.annotation.ValidAudience; -import com.wso2.openbanking.accelerator.common.validator.annotation.ValidScopeFormat; -import com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator.annotations.ValidSigningAlgorithm; -import org.wso2.carbon.identity.oauth2.RequestObjectException; -import org.wso2.carbon.identity.openidconnect.model.RequestObject; -import org.wso2.carbon.identity.openidconnect.model.RequestedClaim; - -import java.util.List; -import java.util.Map; - -/** - * A decorator class of RequestObject to enforce validations. Can add delegate methods as required. - * - * @param Any child of this class. - */ -@RequiredParameters({ - @RequiredParameter(param = "signedJWT", - message = "Only Signed JWS is accepted for request object"), - @RequiredParameter(param = "claimsSet.claims.aud", - message = "aud parameter is missing in the request object") -}) -@ValidScopeFormat(scope = "claimsSet.claims.scope") -@ValidAudience(audience = "claimsSet.claims.aud", clientId = "claimsSet.claims.client_id") -@ValidSigningAlgorithm(algorithm = "signedJWT.header.algorithm.name", clientId = "claimsSet.claims.client_id") -public class OBRequestObject extends RequestObject { - - // decorator object - private RequestObject requestObject; - private static final long serialVersionUID = -568639546792395972L; - - public OBRequestObject(RequestObject requestObject) throws RequestObjectException { - if (requestObject == null) { - throw new RequestObjectException(RequestObjectException.ERROR_CODE_INVALID_REQUEST, - "Null request object passed"); - } - this.requestObject = requestObject; - } - - - // for tool kits to use - - /** - * Any child object of this class can create object of this class using this constructor. - * - * @param childObject Any class extending this class. - */ - public OBRequestObject(T childObject) { - this.requestObject = childObject; - } - - - // delegations - @Override - public SignedJWT getSignedJWT() { - return requestObject.getSignedJWT(); - } - - @Override - public JWTClaimsSet getClaimsSet() { - return requestObject.getClaimsSet(); - } - - @Override - public boolean isSignatureValid() { - return requestObject.isSignatureValid(); - } - - @Override - public void setIsSignatureValid(boolean isSignatureValid) { - requestObject.setIsSignatureValid(isSignatureValid); - } - - @Override - public boolean isSigned() { - return requestObject.isSigned(); - } - - @Override - public PlainJWT getPlainJWT() { - return requestObject.getPlainJWT(); - } - - @Override - public void setPlainJWT(PlainJWT plainJWT) throws RequestObjectException { - requestObject.setPlainJWT(plainJWT); - } - - @Override - public Map> getRequestedClaims() { - return requestObject.getRequestedClaims(); - } - - @Override - public void setRequestedClaims(Map> claimsforRequestParameter) { - requestObject.setRequestedClaims(claimsforRequestParameter); - } - - @Override - public void setSignedJWT(SignedJWT signedJWT) throws RequestObjectException { - requestObject.setSignedJWT(signedJWT); - } - - @Override - public void setClaimSet(JWTClaimsSet claimSet) { - requestObject.setClaimSet(claimSet); - } - - @Override - public String getClaimValue(String claimName) { - return requestObject.getClaimValue(claimName); - } - - @Override - public Object getClaim(String claimName) { - return requestObject.getClaim(claimName); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/models/ValidationResponse.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/models/ValidationResponse.java deleted file mode 100644 index 88e9b676..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/models/ValidationResponse.java +++ /dev/null @@ -1,52 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator.models; - -/** - * Holder of response in request object validation. - */ -public class ValidationResponse { - private boolean valid; - private String violationMessage; - - public ValidationResponse(boolean valid) { - this.valid = valid; - } - - public ValidationResponse(boolean valid, String violationMessage) { - this.valid = valid; - this.violationMessage = violationMessage; - } - - public boolean isValid() { - return valid; - } - - public void setValid(boolean valid) { - this.valid = valid; - } - - public String getViolationMessage() { - return violationMessage; - } - - public void setViolationMessage(String violationMessage) { - this.violationMessage = violationMessage; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/OBCodeResponseTypeHandlerExtension.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/OBCodeResponseTypeHandlerExtension.java deleted file mode 100644 index 5d3ddf6f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/OBCodeResponseTypeHandlerExtension.java +++ /dev/null @@ -1,87 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.response.handler; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.authz.OAuthAuthzReqMessageContext; -import org.wso2.carbon.identity.oauth2.authz.handlers.CodeResponseTypeHandler; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AuthorizeRespDTO; - -/** - * Extension to append scope with OB_ prefix at the end of auth flow, before offering auth code. - */ -public class OBCodeResponseTypeHandlerExtension extends CodeResponseTypeHandler { - - static OBResponseTypeHandler obResponseTypeHandler = - IdentityExtensionsDataHolder.getInstance().getObResponseTypeHandler(); - - /** - * Extension point to get updated scope and refresh token validity period. - * - * @param oauthAuthzMsgCtx - * @return - * @throws IdentityOAuth2Exception - */ - @Override - public OAuth2AuthorizeRespDTO issue(OAuthAuthzReqMessageContext oauthAuthzMsgCtx) throws IdentityOAuth2Exception { - - try { - if (!isRegulatory(oauthAuthzMsgCtx.getAuthorizationReqDTO().getConsumerKey())) { - return issueCode(oauthAuthzMsgCtx); - } - } catch (OpenBankingException e) { - throw new IdentityOAuth2Exception("Error while reading regulatory property"); - } - - // make oauthAuthzMsgCtx immutable - oauthAuthzMsgCtx.setRefreshTokenvalidityPeriod( - obResponseTypeHandler.updateRefreshTokenValidityPeriod(oauthAuthzMsgCtx)); - if (obResponseTypeHandler.updateApprovedScopes(oauthAuthzMsgCtx) != null) { - oauthAuthzMsgCtx.setApprovedScope(obResponseTypeHandler.updateApprovedScopes(oauthAuthzMsgCtx)); - } else { - throw new IdentityOAuth2Exception("Error while updating scopes"); - } - - return issueCode(oauthAuthzMsgCtx); - } - - /** - * Separated method to call parent issue. - * - * @param oAuthAuthzReqMessageContext - * @return - * @throws IdentityOAuth2Exception - */ - @Generated(message = "Cannot test super calls") - OAuth2AuthorizeRespDTO issueCode( - OAuthAuthzReqMessageContext oAuthAuthzReqMessageContext) throws IdentityOAuth2Exception { - - return super.issue(oAuthAuthzReqMessageContext); - } - - @Generated(message = "Ignoring because it requires a service call") - boolean isRegulatory(String clientId) throws OpenBankingException { - - return IdentityCommonUtil.getRegulatoryFromSPMetaData(clientId); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/OBHybridResponseTypeHandlerExtension.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/OBHybridResponseTypeHandlerExtension.java deleted file mode 100644 index 5fc608ad..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/OBHybridResponseTypeHandlerExtension.java +++ /dev/null @@ -1,85 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.response.handler; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.authz.OAuthAuthzReqMessageContext; -import org.wso2.carbon.identity.oauth2.authz.handlers.HybridResponseTypeHandler; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AuthorizeRespDTO; - -/** - * Extension to append scope with OB_ prefix at the end of auth flow, before offering auth code. - */ -public class OBHybridResponseTypeHandlerExtension extends HybridResponseTypeHandler { - - static OBResponseTypeHandler obResponseTypeHandler = - IdentityExtensionsDataHolder.getInstance().getObResponseTypeHandler(); - - /** - * Extension point to get updated scope and refresh token validity period. - * - * @param oauthAuthzMsgCtx - * @return - * @throws IdentityOAuth2Exception - */ - @Override - public OAuth2AuthorizeRespDTO issue(OAuthAuthzReqMessageContext oauthAuthzMsgCtx) throws IdentityOAuth2Exception { - - try { - if (!isRegulatory(oauthAuthzMsgCtx.getAuthorizationReqDTO().getConsumerKey())) { - return issueCode(oauthAuthzMsgCtx); - } - } catch (OpenBankingException e) { - throw new IdentityOAuth2Exception("Error while reading regulatory property"); - } - - oauthAuthzMsgCtx.setRefreshTokenvalidityPeriod( - obResponseTypeHandler.updateRefreshTokenValidityPeriod(oauthAuthzMsgCtx)); - if (obResponseTypeHandler.updateApprovedScopes(oauthAuthzMsgCtx) != null) { - oauthAuthzMsgCtx.setApprovedScope(obResponseTypeHandler.updateApprovedScopes(oauthAuthzMsgCtx)); - } else { - throw new IdentityOAuth2Exception("Error while updating scopes"); - } - return issueCode(oauthAuthzMsgCtx); - } - - /** - * Separated method to call parent issue. - * - * @param oAuthAuthzReqMessageContext - * @return - * @throws IdentityOAuth2Exception - */ - @Generated(message = "cant unit test super calls") - OAuth2AuthorizeRespDTO issueCode( - OAuthAuthzReqMessageContext oAuthAuthzReqMessageContext) throws IdentityOAuth2Exception { - - return super.issue(oAuthAuthzReqMessageContext); - } - - @Generated(message = "Ignoring because it requires a service call") - boolean isRegulatory(String clientId) throws OpenBankingException { - - return IdentityCommonUtil.getRegulatoryFromSPMetaData(clientId); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/OBResponseTypeHandler.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/OBResponseTypeHandler.java deleted file mode 100644 index dd305d7f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/OBResponseTypeHandler.java +++ /dev/null @@ -1,45 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.response.handler; - -import org.wso2.carbon.identity.oauth2.authz.OAuthAuthzReqMessageContext; - -/** - * Extension interface for setting values in response type handling. Toolkits have to implement this. - */ -public interface OBResponseTypeHandler { - - /** - * return the new refresh validity period. - * - * @param oAuthAuthzReqMessageContext - * @return - */ - public long updateRefreshTokenValidityPeriod(OAuthAuthzReqMessageContext oAuthAuthzReqMessageContext); - - /** - * return the new approved scope. - * - * @param oAuthAuthzReqMessageContext - * @return - */ - public String[] updateApprovedScopes(OAuthAuthzReqMessageContext oAuthAuthzReqMessageContext); - - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/impl/OBDefaultResponseTypeHandlerImpl.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/impl/OBDefaultResponseTypeHandlerImpl.java deleted file mode 100644 index 15d4de52..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/impl/OBDefaultResponseTypeHandlerImpl.java +++ /dev/null @@ -1,169 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.response.handler.impl; - -import com.wso2.openbanking.accelerator.identity.auth.extensions.response.handler.OBResponseTypeHandler; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import org.apache.commons.lang.ArrayUtils; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.oauth2.RequestObjectException; -import org.wso2.carbon.identity.oauth2.authz.OAuthAuthzReqMessageContext; -import org.wso2.carbon.identity.openidconnect.RequestObjectService; -import org.wso2.carbon.identity.openidconnect.model.RequestedClaim; - -import java.util.Arrays; -import java.util.List; - -/** - * Default extension implementation. Used to do the accelerator testing. Mimics a UK flow. - */ -public class OBDefaultResponseTypeHandlerImpl implements OBResponseTypeHandler { - - private static final String OPENBANKING_INTENT_ID = "openbanking_intent_id"; - private static final Log log = LogFactory.getLog(OBDefaultResponseTypeHandlerImpl.class); - - /** - * return the new refresh validity period. - * - * @param oAuthAuthzReqMessageContext - * @return - */ - public long updateRefreshTokenValidityPeriod(OAuthAuthzReqMessageContext oAuthAuthzReqMessageContext) { - - return oAuthAuthzReqMessageContext.getRefreshTokenvalidityPeriod(); - } - - /** - * return the new approved scope. - * - * @param oAuthAuthzReqMessageContext - * @return - */ - public String[] updateApprovedScopes(OAuthAuthzReqMessageContext oAuthAuthzReqMessageContext) { - - if (oAuthAuthzReqMessageContext != null && oAuthAuthzReqMessageContext.getAuthorizationReqDTO() != null) { - - String[] scopes = oAuthAuthzReqMessageContext.getApprovedScope(); - if (scopes != null && !Arrays.asList(scopes).contains("api_store")) { - - String sessionDataKey = oAuthAuthzReqMessageContext.getAuthorizationReqDTO().getSessionDataKey(); - String consentID = getConsentIDFromSessionData(sessionDataKey); - if (consentID.isEmpty()) { - log.error("Consent-ID retrieved from request object claims is empty"); - return scopes; - } - - String consentIdClaim = IdentityExtensionsDataHolder.getInstance().getConfigurationMap() - .get(IdentityCommonConstants.CONSENT_ID_CLAIM_NAME).toString(); - String consentScope = consentIdClaim + consentID; - if (!Arrays.asList(scopes).contains(consentScope)) { - String[] updatedScopes = (String[]) ArrayUtils.addAll(scopes, new String[]{consentScope}); - if (log.isDebugEnabled()) { - log.debug("Updated scopes: " + Arrays.toString(updatedScopes)); - } - return updatedScopes; - } - } - - } else { - return new String[0]; - } - - return oAuthAuthzReqMessageContext.getApprovedScope(); - } - - /** - * Call sessionDataAPI and retrieve request object, decode it and return consentID. - * - * @param sessionDataKey sessionDataKeyConsent parameter from authorize request - * @return consentID - */ - String getConsentIDFromSessionData(String sessionDataKey) { - - String consentID = StringUtils.EMPTY; - if (sessionDataKey != null && !sessionDataKey.isEmpty()) { - RequestObjectService requestObjectService = IdentityExtensionsDataHolder.getInstance() - .getRequestObjectService(); - if (requestObjectService != null) { - consentID = retrieveConsentIDFromReqObjService(requestObjectService, sessionDataKey); - if (consentID.isEmpty()) { - log.error("Failed to retrieve ConsentID from query parameters"); - } - } else { - log.error("Failed to retrieve Request Object Service"); - } - } else { - log.error("Invalid Session Data Key"); - } - return consentID; - } - - /** - * Call Request Object Service and retrieve consent id. - * - * @param service request object service - * @param sessionDataKey session data key - * @return consentID - */ - String retrieveConsentIDFromReqObjService(RequestObjectService service, String sessionDataKey) { - - String consentID = StringUtils.EMPTY; - try { - List requestedClaims = service.getRequestedClaimsForSessionDataKey(sessionDataKey, - false); - consentID = iterateClaims(requestedClaims); - if (consentID.isEmpty()) { - requestedClaims = service.getRequestedClaimsForSessionDataKey(sessionDataKey, true); - consentID = iterateClaims(requestedClaims); - } - - } catch (RequestObjectException ex) { - log.error("Exception occurred", ex); - } - return consentID; - } - - /** - * Iterate the claims list to identify the consent-ID. - * - * @param requestedClaims list of claims - * @return consent id - */ - String iterateClaims(List requestedClaims) { - - String consentID = StringUtils.EMPTY; - for (RequestedClaim claim : requestedClaims) { - if (log.isDebugEnabled()) { - log.debug("Claim: " + claim.getName() + ", value: " + claim.getValue()); - } - - if (OPENBANKING_INTENT_ID.equals(claim.getName())) { - consentID = claim.getValue(); - if (log.isDebugEnabled()) { - log.debug("Consent-ID retrieved: " + consentID); - } - break; - } - } - return consentID; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/validator/OBCodeResponseTypeValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/validator/OBCodeResponseTypeValidator.java deleted file mode 100644 index 7ba4aebc..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/validator/OBCodeResponseTypeValidator.java +++ /dev/null @@ -1,80 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.response.validator; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.oltu.oauth2.common.exception.OAuthProblemException; -import org.apache.oltu.oauth2.common.validators.AbstractValidator; - -import javax.servlet.http.HttpServletRequest; - -/** - * Validator to validate whether the response type in the request object is allowed. - * Validate whether the correct response type is sent for regulatory applications. By default, code response type - * is not allowed for regulatory apps. - */ -public class OBCodeResponseTypeValidator extends AbstractValidator { - - private static Log log = LogFactory.getLog(OBCodeResponseTypeValidator.class); - private static final String CODE = "code"; - - @Override - public void validateMethod(HttpServletRequest request) throws OAuthProblemException { - - } - - @Override - public void validateContentType(HttpServletRequest request) throws OAuthProblemException { - - } - - @Override - public void validateRequiredParameters(HttpServletRequest request) throws OAuthProblemException { - String responseType = request.getParameter("response_type"); - String clientId = request.getParameter("client_id"); - if (!isValidResponseType(clientId, responseType)) { - log.error("Unsupported Response Type"); - throw OAuthProblemException.error("Unsupported Response Type"); - } - } - - /** - * Validate whether the correct response type is sent for regulatory applications. By default code response type - * is not allowed for regulatory apps. - * - * @param clientId Client Id received from Request Object - * @param responseType Response Type received from Request Object - * @return - */ - private boolean isValidResponseType(String clientId, String responseType) { - - try { - if (IdentityCommonUtil.getRegulatoryFromSPMetaData(clientId) && CODE.equals(responseType)) { - return false; - } - } catch (OpenBankingException e) { - log.error("Error while retrieving service provider metadata", e); - return false; - } - return true; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/validator/OBHybridResponseTypeValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/validator/OBHybridResponseTypeValidator.java deleted file mode 100644 index 0b8cb1e5..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/validator/OBHybridResponseTypeValidator.java +++ /dev/null @@ -1,95 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.response.validator; - -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.oltu.oauth2.as.validator.TokenValidator; -import org.apache.oltu.oauth2.common.OAuth; -import org.apache.oltu.oauth2.common.error.OAuthError; -import org.apache.oltu.oauth2.common.exception.OAuthProblemException; -import org.wso2.carbon.identity.oauth.common.OAuthConstants; - -import java.util.ArrayList; -import java.util.Arrays; - -import javax.servlet.http.HttpServletRequest; - -/** - Validator for hybrid flow code token requests. - */ -public class OBHybridResponseTypeValidator extends TokenValidator { - - private static final Log log = LogFactory.getLog(OBHybridResponseTypeValidator.class); - - private static boolean isContainOIDCScope(String scope) { - - String[] scopeArray = scope.split("\\s+"); - for (String anyScope : scopeArray) { - if (anyScope.equals(OAuthConstants.Scope.OPENID)) { - return true; - } - } - return false; - } - - @Override - public void validateRequiredParameters(HttpServletRequest request) throws OAuthProblemException { - - String openIdScope; - if (StringUtils.isNotBlank(request.getParameter(IdentityCommonConstants.REQUEST_URI))) { - - this.requiredParams = new ArrayList(Arrays.asList(OAuth.OAUTH_CLIENT_ID, - IdentityCommonConstants.REQUEST_URI)); - this.notAllowedParams.add(IdentityCommonConstants.REQUEST); - openIdScope = IdentityCommonUtil.decodeRequestObjectAndGetKey(request, OAuth.OAUTH_SCOPE); - } else { - openIdScope = request.getParameter(OAuth.OAUTH_SCOPE); - } - - super.validateRequiredParameters(request); - - if (StringUtils.isBlank(openIdScope) || !isContainOIDCScope(openIdScope)) { - String clientID = request.getParameter(OAuth.OAUTH_CLIENT_ID); - throw OAuthProblemException.error(OAuthError.CodeResponse.INVALID_REQUEST) - .description("Request with \'client_id\' = \'" + clientID + - "\' has \'response_type\' for \'hybrid flow\'; but \'openid\' scope not found."); - } - - } - - @Override - public void validateMethod(HttpServletRequest request) throws OAuthProblemException { - - String method = request.getMethod(); - if (!OAuth.HttpMethod.GET.equals(method) && !OAuth.HttpMethod.POST.equals(method)) { - throw OAuthProblemException.error(OAuthError.CodeResponse.INVALID_REQUEST) - .description("Method not correct."); - } - } - - @Override - public void validateContentType(HttpServletRequest request) throws OAuthProblemException { - - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/authenticator/OBIdentifierAuthenticator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/authenticator/OBIdentifierAuthenticator.java deleted file mode 100644 index 4eb4682d..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/authenticator/OBIdentifierAuthenticator.java +++ /dev/null @@ -1,608 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.authenticator; - -import com.wso2.openbanking.accelerator.common.exception.OBThrottlerException; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.identity.authenticator.constants.IdentifierHandlerConstants; -import com.wso2.openbanking.accelerator.identity.authenticator.util.OBIdentifierAuthUtil; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.util.HTTPClientUtils; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonHelper; -import com.wso2.openbanking.accelerator.throttler.service.OBThrottleService; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpGet; -import org.apache.http.impl.client.CloseableHttpClient; -import org.json.JSONObject; -import org.wso2.carbon.identity.application.authentication.framework.AbstractApplicationAuthenticator; -import org.wso2.carbon.identity.application.authentication.framework.AuthenticatorFlowStatus; -import org.wso2.carbon.identity.application.authentication.framework.LocalApplicationAuthenticator; -import org.wso2.carbon.identity.application.authentication.framework.config.ConfigurationFacade; -import org.wso2.carbon.identity.application.authentication.framework.config.model.AuthenticatorConfig; -import org.wso2.carbon.identity.application.authentication.framework.context.AuthenticationContext; -import org.wso2.carbon.identity.application.authentication.framework.exception.AuthenticationFailedException; -import org.wso2.carbon.identity.application.authentication.framework.exception.InvalidCredentialsException; -import org.wso2.carbon.identity.application.authentication.framework.exception.LogoutFailedException; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedIdPData; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedUser; -import org.wso2.carbon.identity.application.authentication.framework.util.FrameworkConstants; -import org.wso2.carbon.identity.application.authentication.framework.util.FrameworkUtils; -import org.wso2.carbon.identity.application.authenticator.basicauth.BasicAuthenticator; -import org.wso2.carbon.identity.application.authenticator.basicauth.BasicAuthenticatorConstants; -import org.wso2.carbon.identity.application.common.model.User; -import org.wso2.carbon.identity.base.IdentityRuntimeException; -import org.wso2.carbon.identity.core.model.IdentityErrorMsgContext; -import org.wso2.carbon.identity.core.util.IdentityCoreConstants; -import org.wso2.carbon.identity.core.util.IdentityTenantUtil; -import org.wso2.carbon.identity.core.util.IdentityUtil; -import org.wso2.carbon.identity.oauth.cache.SessionDataCache; -import org.wso2.carbon.identity.oauth.cache.SessionDataCacheEntry; -import org.wso2.carbon.identity.oauth.cache.SessionDataCacheKey; -import org.wso2.carbon.user.api.RealmConfiguration; -import org.wso2.carbon.user.api.UserRealm; -import org.wso2.carbon.user.core.UserCoreConstants; -import org.wso2.carbon.user.core.UserStoreException; -import org.wso2.carbon.user.core.UserStoreManager; -import org.wso2.carbon.utils.multitenancy.MultitenantUtils; - -import java.io.BufferedReader; -import java.io.IOException; -import java.io.InputStreamReader; -import java.net.HttpURLConnection; -import java.net.URLEncoder; -import java.nio.charset.StandardCharsets; -import java.util.Base64; -import java.util.HashMap; -import java.util.Map; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; - -import static org.wso2.carbon.identity.application.authentication.framework.util.FrameworkConstants.RequestParams.IDENTIFIER_CONSENT; - -/** - * OB Identifier based authenticator. - */ -public class OBIdentifierAuthenticator extends AbstractApplicationAuthenticator - implements LocalApplicationAuthenticator { - - private static final long serialVersionUID = 1819664539416029785L; - private static final Log log = LogFactory.getLog(OBIdentifierAuthenticator.class); - private static final String PROMPT_CONFIRMATION_WINDOW = "promptConfirmationWindow"; - private static final String CONTINUE = "continue"; - private static final String RESET = "reset"; - private static final String RE_CAPTCHA_USER_DOMAIN = "user-domain-recaptcha"; - private static final String USER_TENANT_DOMAIN_MISMATCH = "UserTenantDomainMismatch"; - private static final String OB_IDENTIFIER_AUTHENTICATOR = "OBIdentifierAuthenticator"; - private static final String REDIRECT_URI = "redirect_uri"; - private static final String REQUEST_URI = "request_uri"; - - @Override - public boolean canHandle(HttpServletRequest request) { - - String userName = request.getParameter(IdentifierHandlerConstants.USER_NAME); - String identifierConsent = request.getParameter(IDENTIFIER_CONSENT); - return userName != null || identifierConsent != null; - } - - @Override - public AuthenticatorFlowStatus process(HttpServletRequest request, - HttpServletResponse response, AuthenticationContext context) - throws AuthenticationFailedException, LogoutFailedException { - - if (context.isLogoutRequest()) { - return AuthenticatorFlowStatus.SUCCESS_COMPLETED; - } else { - if (context.getPreviousAuthenticatedIdPs().get(BasicAuthenticatorConstants.LOCAL) != null) { - AuthenticatedIdPData local = - context.getPreviousAuthenticatedIdPs().get(BasicAuthenticatorConstants.LOCAL); - if (local.getAuthenticators().size() > 0) { - for (AuthenticatorConfig authenticatorConfig : local.getAuthenticators()) { - if (authenticatorConfig.getApplicationAuthenticator() instanceof BasicAuthenticator) { - boolean isPrompt = Boolean.parseBoolean(context.getAuthenticatorParams(this - .getName()).get(PROMPT_CONFIRMATION_WINDOW)); - - if (isPrompt) { - String identifierConsent = request.getParameter(IDENTIFIER_CONSENT); - if (identifierConsent != null && CONTINUE.equals(identifierConsent)) { - context.setSubject(local.getUser()); - return AuthenticatorFlowStatus.SUCCESS_COMPLETED; - } else if (identifierConsent != null && RESET.equals(identifierConsent)) { - initiateAuthenticationRequest(request, response, context); - return AuthenticatorFlowStatus.INCOMPLETE; - } else if (request.getParameter(IdentifierHandlerConstants.USER_NAME) != null) { - processAuthenticationResponse(request, response, context); - return AuthenticatorFlowStatus.SUCCESS_COMPLETED; - } else { - String identifierFirstConfirmationURL = - ConfigurationFacade.getInstance().getIdentifierFirstConfirmationURL(); - String queryParams = context.getContextIdIncludedQueryParams(); - try { - queryParams = queryParams + "&username=" + local.getUser() - .toFullQualifiedUsername(); - response.sendRedirect(identifierFirstConfirmationURL + - ("?" + queryParams)); - return AuthenticatorFlowStatus.INCOMPLETE; - } catch (IOException e) { - throw new AuthenticationFailedException(e.getMessage(), e); - } - } - } else { - context.setSubject(local.getUser()); - return AuthenticatorFlowStatus.SUCCESS_COMPLETED; - } - } - } - } - } else if (request.getParameter(IDENTIFIER_CONSENT) != null) { - //submit from the confirmation page. - initiateAuthenticationRequest(request, response, context); - return AuthenticatorFlowStatus.INCOMPLETE; - } - return super.process(request, response, context); - } - } - - @Override - protected void initiateAuthenticationRequest(HttpServletRequest request, - HttpServletResponse response, AuthenticationContext context) - throws AuthenticationFailedException { - - int throttleLimit = 3; //default allowed attempts (use this if config is not defined) - int throttleTimePeriod = 180; //default blocked time period (3 minutes) - String showAuthFailureReason = ""; - String loginPage = ConfigurationFacade.getInstance().getAuthenticationEndpointURL(); - String retryPage = ConfigurationFacade.getInstance().getAuthenticationEndpointRetryURL(); - String queryParams = context.getContextIdIncludedQueryParams(); - OBThrottleService obThrottleService = IdentityExtensionsDataHolder.getInstance().getOBThrottleService(); - - //Read authenticator configs - Map parameterMap = getAuthenticatorConfig().getParameterMap(); - if (parameterMap != null) { - throttleLimit = Integer.parseInt(parameterMap.get("throttleLimit")); - throttleTimePeriod = Integer.parseInt(parameterMap.get("throttleTimePeriod")); - showAuthFailureReason = parameterMap.get("showAuthFailureReason"); - } - - try { - String retryParam = ""; - if (context.isRetrying()) { - // Update throttling data - String userIp = IdentityUtil.getClientIpAddress(request); - obThrottleService - .updateThrottleData(OB_IDENTIFIER_AUTHENTICATOR, userIp, throttleLimit, throttleTimePeriod); - // Check if the client-ip is throttled - if (obThrottleService.isThrottled(OB_IDENTIFIER_AUTHENTICATOR, userIp)) { - retryParam = "&authFailure=true&authFailureMsg=Too.many.attempts"; - } else { - if (context.getProperty(IdentifierHandlerConstants.CONTEXT_PROP_INVALID_EMAIL_USERNAME) != null && - (Boolean) context.getProperty(IdentifierHandlerConstants. - CONTEXT_PROP_INVALID_EMAIL_USERNAME)) { - retryParam = "&authFailure=true&authFailureMsg=Login.failed"; - context.setProperty(IdentifierHandlerConstants.CONTEXT_PROP_INVALID_EMAIL_USERNAME, false); - } else { - retryParam = "&authFailure=true&authFailureMsg=Login.failed"; - } - } - } - - if (context.getProperty(USER_TENANT_DOMAIN_MISMATCH) != null && - (Boolean) context.getProperty(USER_TENANT_DOMAIN_MISMATCH)) { - retryParam = "&authFailure=true&authFailureMsg=user.tenant.domain.mismatch.message"; - context.setProperty(USER_TENANT_DOMAIN_MISMATCH, false); - } - - IdentityErrorMsgContext errorContext = IdentityUtil.getIdentityErrorMsg(); - IdentityUtil.clearIdentityErrorMsg(); - - if (errorContext != null && errorContext.getErrorCode() != null) { - log.debug("Identity error message context is not null"); - String errorCode = errorContext.getErrorCode(); - - if (errorCode.equals(IdentityCoreConstants.USER_ACCOUNT_NOT_CONFIRMED_ERROR_CODE)) { - retryParam = "&authFailure=true&authFailureMsg=account.confirmation.pending"; - String username = request.getParameter(IdentifierHandlerConstants.USER_NAME); - Object domain = IdentityUtil.threadLocalProperties.get().get(RE_CAPTCHA_USER_DOMAIN); - if (domain != null) { - username = IdentityUtil.addDomainToName(username, domain.toString()); - } - String redirectURL = loginPage + ("?" + queryParams) + IdentifierHandlerConstants.FAILED_USERNAME - + URLEncoder.encode(username, IdentifierHandlerConstants.UTF_8) + - IdentifierHandlerConstants.ERROR_CODE + errorCode + IdentifierHandlerConstants - .AUTHENTICATORS + getName() + ":" + IdentifierHandlerConstants.LOCAL + retryParam; - response.sendRedirect(redirectURL); - - } else if ("true".equals(showAuthFailureReason)) { - String reason = null; - if (errorCode.contains(":")) { - String[] errorCodeReason = errorCode.split(":"); - errorCode = errorCodeReason[0]; - if (errorCodeReason.length > 1) { - reason = errorCodeReason[1]; - } - } - int remainingAttempts = - errorContext.getMaximumLoginAttempts() - errorContext.getFailedLoginAttempts(); - - if (log.isDebugEnabled()) { - log.debug("errorCode : " + errorCode); - log.debug("username : " + request.getParameter(IdentifierHandlerConstants.USER_NAME)); - log.debug("remainingAttempts : " + remainingAttempts); - } - - if (errorCode.equals(UserCoreConstants.ErrorCode.INVALID_CREDENTIAL)) { - retryParam = retryParam + IdentifierHandlerConstants.ERROR_CODE + errorCode - + IdentifierHandlerConstants.FAILED_USERNAME + URLEncoder - .encode(request.getParameter(IdentifierHandlerConstants.USER_NAME), - IdentifierHandlerConstants.UTF_8) - + "&remainingAttempts=" + remainingAttempts; - response.sendRedirect(loginPage + ("?" + queryParams) - + IdentifierHandlerConstants.AUTHENTICATORS + getName() + ":" + - IdentifierHandlerConstants.LOCAL + retryParam); - } else if (errorCode.equals(UserCoreConstants.ErrorCode.USER_IS_LOCKED)) { - String redirectURL = retryPage; - if (remainingAttempts == 0) { - if (StringUtils.isBlank(reason)) { - redirectURL = URLEncoder.encode((redirectURL + ("?" + queryParams)), - IdentifierHandlerConstants.UTF_8) + IdentifierHandlerConstants.ERROR_CODE - + errorCode + IdentifierHandlerConstants.FAILED_USERNAME + - URLEncoder.encode(request.getParameter(IdentifierHandlerConstants.USER_NAME), - IdentifierHandlerConstants.UTF_8) + - "&remainingAttempts=0"; - } else { - redirectURL = URLEncoder.encode((redirectURL + ("?" + queryParams)), - IdentifierHandlerConstants.UTF_8) + IdentifierHandlerConstants.ERROR_CODE - + errorCode + "&lockedReason=" + reason + - IdentifierHandlerConstants.FAILED_USERNAME + - URLEncoder.encode(request.getParameter(IdentifierHandlerConstants.USER_NAME), - IdentifierHandlerConstants.UTF_8) + "&remainingAttempts=0"; - } - } else { - if (StringUtils.isBlank(reason)) { - redirectURL = URLEncoder.encode((redirectURL + ("?" + queryParams)), - IdentifierHandlerConstants.UTF_8) + IdentifierHandlerConstants.ERROR_CODE + - errorCode + IdentifierHandlerConstants.FAILED_USERNAME + - URLEncoder.encode(request.getParameter(IdentifierHandlerConstants.USER_NAME), - IdentifierHandlerConstants.UTF_8); - } else { - redirectURL = URLEncoder.encode((redirectURL + ("?" + queryParams)), - IdentifierHandlerConstants.UTF_8) + IdentifierHandlerConstants.ERROR_CODE + - errorCode + "&lockedReason=" + reason + - IdentifierHandlerConstants.FAILED_USERNAME + - URLEncoder.encode(request.getParameter(IdentifierHandlerConstants.USER_NAME), - IdentifierHandlerConstants.UTF_8); - } - } - response.sendRedirect(redirectURL); - } else { - retryParam = retryParam + IdentifierHandlerConstants.ERROR_CODE + errorCode - + IdentifierHandlerConstants.FAILED_USERNAME + URLEncoder - .encode(request.getParameter(IdentifierHandlerConstants.USER_NAME), - IdentifierHandlerConstants.UTF_8); - response.sendRedirect(loginPage + ("?" + queryParams) - + IdentifierHandlerConstants.AUTHENTICATORS + getName() + ":" - + IdentifierHandlerConstants.LOCAL + retryParam); - } - } else { - log.debug("Unknown identity error code."); - response.sendRedirect(loginPage + ("?" + queryParams) - + IdentifierHandlerConstants.AUTHENTICATORS + getName() + ":" + - IdentifierHandlerConstants.LOCAL + retryParam); - } - } else { - log.debug("Identity error message context is null"); - response.sendRedirect(loginPage + ("?" + queryParams) - + IdentifierHandlerConstants.AUTHENTICATORS + getName() + ":" + - IdentifierHandlerConstants.LOCAL + retryParam); - } - } catch (IOException e) { - throw new AuthenticationFailedException(e.getMessage(), User.getUserFromUserName(request.getParameter - (IdentifierHandlerConstants.USER_NAME)), e); - } catch (OBThrottlerException e) { - throw new AuthenticationFailedException("Error occurred while deleting throttle data.", e); - } - } - - @Override - protected void processAuthenticationResponse(HttpServletRequest request, - HttpServletResponse response, AuthenticationContext context) - throws AuthenticationFailedException { - - OBIdentifierAuthUtil.validateUsername(request.getParameter(BasicAuthenticatorConstants.USER_NAME), context); - OBThrottleService obThrottleService = IdentityExtensionsDataHolder.getInstance().getOBThrottleService(); - String username = OBIdentifierAuthUtil.preprocessUsername( - request.getParameter(IdentifierHandlerConstants.USER_NAME), context); - Map authProperties = context.getProperties(); - if (authProperties == null) { - authProperties = new HashMap<>(); - context.setProperties(authProperties); - } - - String userIp = IdentityUtil.getClientIpAddress(request); - try { - // Check if the client-ip is throttled. - if (obThrottleService.isThrottled(OB_IDENTIFIER_AUTHENTICATOR, userIp)) { - throw new AuthenticationFailedException("Too many attempts to log in.", - User.getUserFromUserName(username)); - } - } catch (OBThrottlerException e) { - throw new AuthenticationFailedException("Error occurred while deleting throttle data.", e); - } - - if (getAuthenticatorConfig().getParameterMap() != null) { - String validateUsername = getAuthenticatorConfig().getParameterMap().get("ValidateUsername"); - if (Boolean.valueOf(validateUsername)) { - boolean isUserExists; - UserStoreManager userStoreManager; - // Check if the username exists. - try { - int tenantId = IdentityTenantUtil.getTenantIdOfUser(username); - UserRealm userRealm = IdentityExtensionsDataHolder.getInstance().getRealmService() - .getTenantUserRealm(tenantId); - - if (userRealm != null) { - userStoreManager = (UserStoreManager) userRealm.getUserStoreManager(); - isUserExists = userStoreManager.isExistingUser(MultitenantUtils.getTenantAwareUsername - (username)); - } else { - throw new AuthenticationFailedException("Cannot find the user realm for the given tenant: " + - tenantId, User.getUserFromUserName(username)); - } - } catch (IdentityRuntimeException e) { - log.error("OBIdentifierAuthenticator failed while trying to get the tenant ID of " + - "the user " + username, e); - throw new AuthenticationFailedException(e.getMessage(), User.getUserFromUserName(username), e); - } catch (org.wso2.carbon.user.api.UserStoreException e) { - log.error("OBIdentifierAuthenticator failed while trying to authenticate", e); - throw new AuthenticationFailedException(e.getMessage(), User.getUserFromUserName(username), e); - } - if (!isUserExists) { - log.debug("User does not exist."); - if (IdentityUtil.threadLocalProperties.get().get(RE_CAPTCHA_USER_DOMAIN) != null) { - username = IdentityUtil.addDomainToName( - username, IdentityUtil.threadLocalProperties.get().get(RE_CAPTCHA_USER_DOMAIN) - .toString()); - } - IdentityUtil.threadLocalProperties.get().remove(RE_CAPTCHA_USER_DOMAIN); - throw new InvalidCredentialsException("User does not exist.", User.getUserFromUserName(username)); - } - String tenantDomain = MultitenantUtils.getTenantDomain(username); - authProperties.put("user-tenant-domain", tenantDomain); - } - } - - username = FrameworkUtils.prependUserStoreDomainToName(username); - authProperties.put("username", username); - Map identifierParams = new HashMap<>(); - identifierParams.put(FrameworkConstants.JSAttributes.JS_OPTIONS_USERNAME, username); - Map> contextParams = new HashMap<>(); - contextParams.put(FrameworkConstants.JSAttributes.JS_COMMON_OPTIONS, identifierParams); - //Identifier first is the first authenticator. - context.getPreviousAuthenticatedIdPs().clear(); - context.addAuthenticatorParams(contextParams); - context.setSubject(AuthenticatedUser.createLocalAuthenticatedUserFromSubjectIdentifier(username)); - if (context.getParameters().containsKey("username")) { - try { - obThrottleService.deleteRecordOnSuccessAttempt(OB_IDENTIFIER_AUTHENTICATOR, userIp); - } catch (OBThrottlerException e) { - throw new AuthenticationFailedException("Error occurred while deleting throttle data.", e); - } - } - } - - @Override - protected boolean retryAuthenticationEnabled() { - return true; - } - - @Override - public String getContextIdentifier(HttpServletRequest request) { - return request.getParameter("sessionDataKey"); - } - - @Override - public String getFriendlyName() { - return IdentifierHandlerConstants.HANDLER_FRIENDLY_NAME; - } - - @Override - public String getName() { - return IdentifierHandlerConstants.HANDLER_NAME; - } - - /** - * To get session details from SessionDataKey. - * authRequestURL need be configured in the IAM deployment.toml file. - * @param sessionDataKey session data key - * @return session data - * @throws OpenBankingException openbanking exception - */ - public String getSessionData(String sessionDataKey) throws OpenBankingException { - - BufferedReader reader = null; - String authRequestURL = null; - RealmConfiguration realmConfig = null; - - //Read authenticator configs - Map parameterMap = getAuthenticatorConfig().getParameterMap(); - if (parameterMap != null) { - authRequestURL = parameterMap.get(IdentifierHandlerConstants.AUTH_REQ_URL); - } - try { - realmConfig = IdentityExtensionsDataHolder.getInstance().getRealmService() - .getBootstrapRealm().getUserStoreManager().getRealmConfiguration(); - } catch (UserStoreException e) { - throw new OpenBankingException("Error while retrieving session data", e); - } - String adminUsername = realmConfig.getAdminUserName(); - char[] adminPassword = realmConfig.getAdminPassword().toCharArray(); - - String credentials = adminUsername + ":" + String.valueOf(adminPassword); - credentials = Base64.getEncoder().encodeToString(credentials.getBytes(StandardCharsets.UTF_8)); - - try (CloseableHttpClient client = HTTPClientUtils.getHttpsClient()) { - HttpGet dataRequest = new HttpGet(authRequestURL + sessionDataKey); - dataRequest.addHeader(IdentifierHandlerConstants.ACCEPT_HEADER, - IdentifierHandlerConstants.ACCEPT_HEADER_VALUE); - dataRequest.addHeader(IdentifierHandlerConstants.AUTH_HEADER, "Basic " + credentials); - CloseableHttpResponse dataResponse = client.execute(dataRequest); - - reader = new BufferedReader(new InputStreamReader(dataResponse.getEntity() - .getContent(), "UTF-8")); - String inputLine; - StringBuffer buffer = new StringBuffer(); - while ((inputLine = reader.readLine()) != null) { - buffer.append(inputLine); - } - - if (dataResponse.getStatusLine().getStatusCode() != HttpURLConnection.HTTP_OK) { - return null; - } else { - JSONObject sessionData = new JSONObject(buffer.toString()); - appendRedirectUri(sessionData); - return sessionData.toString(); - } - } catch (IOException e) { - throw new OpenBankingException("Error while retrieving session data", e); - } finally { - if (reader != null) { - try { - reader.close(); - } catch (IOException e) { - log.error("Error while closing buffered reader", e); - } - } - } - } - - /** - * Append redirect_uri value to session data for par requests. - * - * @param sessionData - */ - private void appendRedirectUri(JSONObject sessionData) throws OpenBankingException { - - // Handle redirect uri for par request. - // In par requests, there's no redirect_uri in request object itself, so fetch redirect uri from cached request. - if (!sessionData.has(REDIRECT_URI) && sessionData.has(REQUEST_URI)) { - - JSONObject requestObjectVal = getParRequestObject(sessionData); - if (requestObjectVal.has(REDIRECT_URI)) { - sessionData.put(REDIRECT_URI, requestObjectVal.get(REDIRECT_URI)); - } else { - log.error("redirect_uri could not be found in the par request object."); - throw new OpenBankingException("redirect_uri could not be found in the par request object."); - } - } - } - - /** - * Get redirect_uri using request_uri. - * - * @param requestUri - request_uri - * @return redirect_uri - * @throws OpenBankingException - OpenBankingException - */ - @Generated(message = "Excluding from code coverage since it requires a valid cache entry") - public String getRedirectUri(String requestUri) throws OpenBankingException { - - JSONObject requestObjectVal = getParRequestObject(requestUri); - if (requestObjectVal.has(REDIRECT_URI)) { - return requestObjectVal.get(REDIRECT_URI).toString(); - } else { - log.error("redirect_uri could not be found in the par request object."); - throw new OpenBankingException("redirect_uri could not be found in the par request object."); - } - } - - /** - * Retrieve PAR request object from session data cache. - * - * @param sessionData session data - * @return Request object json. - * @throws OpenBankingException - */ - @Generated(message = "Excluding from code coverage since it requires a valid cache entry") - private JSONObject getParRequestObject(JSONObject sessionData) throws OpenBankingException { - - //get request ref Ex -> "IVL...." from "urn::IVL..." - String requestUri = sessionData.get(REQUEST_URI).toString(); - return getParRequestObject(requestUri); - } - - /** - * Retrieve PAR request object from request_uri. - * - * @param requestUri - request_uri - * @return Request object json. - * @throws OpenBankingException - OpenBankingException - */ - @Generated(message = "Excluding from code coverage since it requires a valid cache entry") - private JSONObject getParRequestObject(String requestUri) throws OpenBankingException { - - String[] requestUriArr = requestUri.split(":"); - String requestUriRef = requestUriArr[requestUriArr.length - 1]; - return getRequestObjectUsingUriReference(requestUriRef); - } - - /** - * Retrieve PAR request object using request_uri reference. - * - * @param requestUriReference - request_uri reference (i.e:last part of request_uri split by :) - * @return Request object json. - * @throws OpenBankingException - OpenBankingException - */ - @Generated(message = "Excluding from code coverage since it requires a valid cache entry") - private JSONObject getRequestObjectUsingUriReference(String requestUriReference) throws OpenBankingException { - - SessionDataCacheKey cacheKey = new SessionDataCacheKey(requestUriReference); - SessionDataCacheEntry cacheEntry = SessionDataCache.getInstance().getValueFromCache(cacheKey); - - if (cacheEntry != null) { - String essentialClaims = cacheEntry.getoAuth2Parameters().getEssentialClaims(); - byte[] requestObject; - try { - requestObject = Base64.getDecoder().decode(essentialClaims.split("\\.")[1]); - } catch (IllegalArgumentException e) { - // Decode if the requestObject is base64-url encoded. - requestObject = Base64.getUrlDecoder().decode(essentialClaims.split("\\.")[1]); - } - return new JSONObject(new String(requestObject, StandardCharsets.UTF_8)); - } else { - log.error("Unable to fetch par request object from session data cache."); - throw new OpenBankingException("Unable to fetch par request object from session data cache."); - } - } - - /** - * Get SSA client_name using clientId. - * - * @param clientId client id. - * @param property required property. - * @return service provider value. - */ - @Generated(message = "Excluding from code coverage since it requires a service call") - public String getSPProperty (String clientId, String property) throws OpenBankingException { - - return new IdentityCommonHelper().getAppPropertyFromSPMetaData(clientId, property); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/authenticator/constants/IdentifierHandlerConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/authenticator/constants/IdentifierHandlerConstants.java deleted file mode 100644 index edc3c1dd..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/authenticator/constants/IdentifierHandlerConstants.java +++ /dev/null @@ -1,40 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.authenticator.constants; - -/** - * Constants used by the OBIdentifierAuthenticator. - */ -public class IdentifierHandlerConstants { - - public static final String CONTEXT_PROP_INVALID_EMAIL_USERNAME = "InvalidEmailUsername"; - public static final String HANDLER_NAME = "IdentifierExecutor"; - public static final String HANDLER_FRIENDLY_NAME = "ob-identifier-first"; - public static final String USER_NAME = "username"; - public static final String FAILED_USERNAME = "&failedUsername="; - public static final String ERROR_CODE = "&errorCode="; - public static final String AUTHENTICATORS = "&authenticators="; - public static final String LOCAL = "LOCAL"; - public static final String UTF_8 = "UTF-8"; - - //auth request params - public static final String AUTH_REQ_URL = "authRequestURL"; - public static final String ACCEPT_HEADER = "accept"; - public static final String ACCEPT_HEADER_VALUE = "application/json"; - public static final String AUTH_HEADER = "Authorization"; -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/authenticator/util/OBIdentifierAuthUtil.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/authenticator/util/OBIdentifierAuthUtil.java deleted file mode 100644 index 0691788d..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/authenticator/util/OBIdentifierAuthUtil.java +++ /dev/null @@ -1,70 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.authenticator.util; - -import org.apache.commons.lang.StringUtils; -import org.wso2.carbon.identity.application.authentication.framework.context.AuthenticationContext; -import org.wso2.carbon.identity.application.authentication.framework.exception.InvalidCredentialsException; -import org.wso2.carbon.identity.core.util.IdentityUtil; -import org.wso2.carbon.utils.multitenancy.MultitenantUtils; - -/** - * Util methods for OBIdentifierFirstAuthenticator. - */ -public class OBIdentifierAuthUtil { - - /** - * Add tenant domain to the username. - * - * @param username - username - * @param context - authentication context - * @return - processed username - */ - public static String preprocessUsername(String username, AuthenticationContext context) { - if (context.getSequenceConfig().getApplicationConfig().isSaaSApp()) { - return username; - } else { - if (IdentityUtil.isEmailUsernameEnabled()) { - if (StringUtils.countMatches(username, "@") == 1) { - return username + "@" + context.getTenantDomain(); - } - } else if (!username.contains("@")) { - return username + "@" + context.getTenantDomain(); - } - return username; - } - } - - /** - * Check if the username exists. - * - * @param username - username - * @param context - authentication context - * @throws InvalidCredentialsException - */ - public static void validateUsername(String username, AuthenticationContext context) - throws InvalidCredentialsException { - if (IdentityUtil.isEmailUsernameEnabled()) { - String tenantAwareUsername = MultitenantUtils.getTenantAwareUsername(username); - if (StringUtils.countMatches(tenantAwareUsername, "@") < 1) { - context.setProperty("InvalidEmailUsername", true); - throw new InvalidCredentialsException("Invalid username. Username has to be an email."); - } - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/builders/DefaultOBRequestUriRequestObjectBuilder.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/builders/DefaultOBRequestUriRequestObjectBuilder.java deleted file mode 100644 index 047a55bd..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/builders/DefaultOBRequestUriRequestObjectBuilder.java +++ /dev/null @@ -1,197 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.builders; - -import com.nimbusds.jose.JOSEException; -import com.nimbusds.jose.JOSEObject; -import com.nimbusds.jose.JWEObject; -import com.nimbusds.jose.JWSAlgorithm; -import com.nimbusds.jose.crypto.RSADecrypter; -import com.nimbusds.jwt.EncryptedJWT; -import com.nimbusds.jwt.JWTClaimsSet; -import com.nimbusds.jwt.PlainJWT; -import com.nimbusds.jwt.SignedJWT; -import net.minidev.json.JSONObject; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.base.MultitenantConstants; -import org.wso2.carbon.identity.oauth.cache.SessionDataCache; -import org.wso2.carbon.identity.oauth.cache.SessionDataCacheEntry; -import org.wso2.carbon.identity.oauth.cache.SessionDataCacheKey; -import org.wso2.carbon.identity.oauth.common.OAuth2ErrorCodes; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.RequestObjectException; -import org.wso2.carbon.identity.oauth2.model.OAuth2Parameters; -import org.wso2.carbon.identity.oauth2.util.OAuth2Util; -import org.wso2.carbon.identity.openidconnect.RequestObjectBuilder; -import org.wso2.carbon.identity.openidconnect.model.RequestObject; - -import java.security.Key; -import java.security.interfaces.RSAPrivateKey; -import java.text.ParseException; -import java.time.Instant; - -import static org.wso2.carbon.identity.openidconnect.model.Constants.JWT_PART_DELIMITER; -import static org.wso2.carbon.identity.openidconnect.model.Constants.NUMBER_OF_PARTS_IN_JWE; -import static org.wso2.carbon.identity.openidconnect.model.Constants.NUMBER_OF_PARTS_IN_JWS; - -/** - * Build Request Object from request_uri for authorize call's request object. - * Object is stored as JWT string in Session DataStore Cache. - * - * Works in-coordination with Push Authorization endpoint. - * - * To differentiate 'request' and 'request_uri' auth calls, an internal claim is added to the request object. - */ -public class DefaultOBRequestUriRequestObjectBuilder implements RequestObjectBuilder { - - private static final Log log = LogFactory.getLog(DefaultOBRequestUriRequestObjectBuilder.class); - private static final String PAR_INITIATED_REQ_OBJ = "par_initiated_request_object"; - - @Override - public RequestObject buildRequestObject(String urn, OAuth2Parameters oAuth2Parameters) - throws RequestObjectException { - - String[] sessionKey = urn.split(":"); - - SessionDataCacheKey sessionDataCacheKey = new SessionDataCacheKey(sessionKey[(sessionKey.length - 1)]); - SessionDataCacheEntry sessionDataCacheEntry = SessionDataCache.getInstance() - .getValueFromCache(sessionDataCacheKey); - RequestObject requestObject = new RequestObject(); - - if (sessionDataCacheEntry == null) { - throw new RequestObjectException(OAuth2ErrorCodes.INVALID_REQUEST, "Invalid request URI"); - } - - // Making a copy of requestObjectParam to prevent editing initial reference - String requestObjectParamValue = sessionDataCacheEntry.getoAuth2Parameters().getEssentialClaims(); - - // validate expiry - String[] jwtWithExpiry = requestObjectParamValue.split(":"); - if (Instant.now().getEpochSecond() > Long.parseLong(jwtWithExpiry[1])) { - throw new RequestObjectException(OAuth2ErrorCodes.INVALID_REQUEST, "Expired request URI"); - } - - requestObjectParamValue = jwtWithExpiry[0]; - - if (isEncrypted(requestObjectParamValue)) { - requestObjectParamValue = decrypt(requestObjectParamValue, oAuth2Parameters); - if (StringUtils.isEmpty(requestObjectParamValue)) { - return requestObject; - } - } - - setRequestObjectValues(requestObjectParamValue, requestObject); - return requestObject; - - } - - @Override - public String decrypt(String requestObject, OAuth2Parameters oAuth2Parameters) throws RequestObjectException { - EncryptedJWT encryptedJWT; - try { - encryptedJWT = EncryptedJWT.parse(requestObject); - RSAPrivateKey rsaPrivateKey = getRSAPrivateKey(oAuth2Parameters); - RSADecrypter decrypter = new RSADecrypter(rsaPrivateKey); - encryptedJWT.decrypt(decrypter); - - JWEObject jweObject = JWEObject.parse(requestObject); - jweObject.decrypt(decrypter); - - if (jweObject.getPayload() != null && jweObject.getPayload().toString() - .split(JWT_PART_DELIMITER).length == NUMBER_OF_PARTS_IN_JWS) { - return jweObject.getPayload().toString(); - } else { - return new PlainJWT(encryptedJWT.getJWTClaimsSet()).serialize(); - } - - } catch (JOSEException | IdentityOAuth2Exception | ParseException e) { - String errorMessage = "Failed to decrypt Request Object"; - log.error(errorMessage + " from " + requestObject, e); - throw new RequestObjectException(RequestObjectException.ERROR_CODE_INVALID_REQUEST, errorMessage); - } - } - - /** - * Retrieve RSA private key. - * - * @param oAuth2Parameters oAuth2Parameters - * @return RSA private key - */ - private RSAPrivateKey getRSAPrivateKey(OAuth2Parameters oAuth2Parameters) throws IdentityOAuth2Exception { - - String tenantDomain = getTenantDomainForDecryption(oAuth2Parameters); - int tenantId = OAuth2Util.getTenantId(tenantDomain); - Key key = OAuth2Util.getPrivateKey(tenantDomain, tenantId); - return (RSAPrivateKey) key; - } - - /** - * Get tenant domain from oAuth2Parameters. - * - * @param oAuth2Parameters oAuth2Parameters - * @return Tenant domain - */ - private String getTenantDomainForDecryption(OAuth2Parameters oAuth2Parameters) { - - if (StringUtils.isNotEmpty(oAuth2Parameters.getTenantDomain())) { - return oAuth2Parameters.getTenantDomain(); - } - return MultitenantConstants.SUPER_TENANT_NAME; - } - - /** - * Check whether given request object is encrypted. - * - * @param requestObject request object string - * @return true if its encrypted - */ - private boolean isEncrypted(String requestObject) { - - return requestObject.split(JWT_PART_DELIMITER).length == NUMBER_OF_PARTS_IN_JWE; - } - - /** - * Set retrieved claims to the request object instance. - * - * @param requestObjectString request object string - * @param requestObjectInstance request object instance - * @return - */ - private void setRequestObjectValues(String requestObjectString, RequestObject requestObjectInstance) throws - RequestObjectException { - - try { - JOSEObject jwt = JOSEObject.parse(requestObjectString); - if (jwt.getHeader().getAlgorithm() == null || jwt.getHeader().getAlgorithm().equals(JWSAlgorithm.NONE)) { - requestObjectInstance.setPlainJWT(PlainJWT.parse(requestObjectString)); - } else { - requestObjectInstance.setSignedJWT(SignedJWT.parse(requestObjectString)); - } - JSONObject claimSet = requestObjectInstance.getClaimsSet().toJSONObject(); - claimSet.put(PAR_INITIATED_REQ_OBJ, "true"); - requestObjectInstance.setClaimSet(JWTClaimsSet.parse(claimSet)); - } catch (ParseException e) { - String errorMessage = "No Valid JWT is found for the Request Object."; - log.error(errorMessage, e); - throw new RequestObjectException(OAuth2ErrorCodes.INVALID_REQUEST, errorMessage); - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/cache/IdentityCache.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/cache/IdentityCache.java deleted file mode 100644 index 7423ec2f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/cache/IdentityCache.java +++ /dev/null @@ -1,63 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.cache; - -import com.wso2.openbanking.accelerator.common.caching.OpenBankingBaseCache; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; - -/** - * Cache definition to store objects in open banking iam component implementations. - */ -public class IdentityCache extends OpenBankingBaseCache { - - private static final String cacheName = "OPEN_BANKING_IDENTITY_CACHE"; - - private Integer accessExpiryMinutes; - private Integer modifiedExpiryMinutes; - - /** - * Initialize with unique cache name. - */ - public IdentityCache() { - - super(cacheName); - this.accessExpiryMinutes = setAccessExpiryMinutes(); - this.modifiedExpiryMinutes = setModifiedExpiryMinutes(); - } - - @Override - public int getCacheAccessExpiryMinutes() { - return accessExpiryMinutes; - } - - @Override - public int getCacheModifiedExpiryMinutes() { - return modifiedExpiryMinutes; - } - - public int setAccessExpiryMinutes() { - - return IdentityExtensionsDataHolder.getInstance().getIdentityCacheAccessExpiry(); - } - - public int setModifiedExpiryMinutes() { - - return IdentityExtensionsDataHolder.getInstance().getIdentityCacheModifiedExpiry(); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/cache/IdentityCacheKey.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/cache/IdentityCacheKey.java deleted file mode 100644 index 090cfefb..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/cache/IdentityCacheKey.java +++ /dev/null @@ -1,62 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.cache; - -import com.wso2.openbanking.accelerator.common.caching.OpenBankingBaseCacheKey; - -import java.io.Serializable; -import java.util.Objects; - -/** - * Cache Key for Open Banking Identity cache. - */ -public class IdentityCacheKey extends OpenBankingBaseCacheKey implements Serializable { - - private static final long serialVersionUID = 143057970021542120L; - public String identityCacheKey; - - public IdentityCacheKey(String identityCacheKey) { - - this.identityCacheKey = identityCacheKey; - } - - public static IdentityCacheKey of(String identityCacheKey) { - - return new IdentityCacheKey(identityCacheKey); - } - - @Override - public boolean equals(Object o) { - - if (this == o) { - return true; - } - if (o == null || getClass() != o.getClass()) { - return false; - } - IdentityCacheKey that = (IdentityCacheKey) o; - return Objects.equals(identityCacheKey, that.identityCacheKey); - } - - @Override - public int hashCode() { - - return Objects.hash(identityCacheKey); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/claims/OBClaimProvider.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/claims/OBClaimProvider.java deleted file mode 100644 index b802f075..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/claims/OBClaimProvider.java +++ /dev/null @@ -1,63 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.claims; - -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.authz.OAuthAuthzReqMessageContext; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AccessTokenRespDTO; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AuthorizeRespDTO; -import org.wso2.carbon.identity.oauth2.token.OAuthTokenReqMessageContext; -import org.wso2.carbon.identity.openidconnect.ClaimProvider; - -import java.util.Map; - -/** - * OB specific claim provider. - */ -public class OBClaimProvider implements ClaimProvider { - - private static ClaimProvider claimProvider; - - @Override - public Map getAdditionalClaims(OAuthAuthzReqMessageContext authAuthzReqMessageContext, - OAuth2AuthorizeRespDTO authorizeRespDTO) - throws IdentityOAuth2Exception { - - return getClaimProvider().getAdditionalClaims(authAuthzReqMessageContext, authorizeRespDTO); - - } - - @Override - public Map getAdditionalClaims(OAuthTokenReqMessageContext tokenReqMessageContext, - OAuth2AccessTokenRespDTO tokenRespDTO) - throws IdentityOAuth2Exception { - - return getClaimProvider().getAdditionalClaims(tokenReqMessageContext, tokenRespDTO); - } - - public static void setClaimProvider(ClaimProvider claimProvider) { - - OBClaimProvider.claimProvider = claimProvider; - } - - public static ClaimProvider getClaimProvider() { - - return claimProvider; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/claims/OBDefaultClaimProvider.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/claims/OBDefaultClaimProvider.java deleted file mode 100644 index 187a10d9..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/claims/OBDefaultClaimProvider.java +++ /dev/null @@ -1,151 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.claims; - -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import net.minidev.json.JSONObject; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.oauth.cache.SessionDataCache; -import org.wso2.carbon.identity.oauth.cache.SessionDataCacheEntry; -import org.wso2.carbon.identity.oauth.cache.SessionDataCacheKey; -import org.wso2.carbon.identity.oauth.common.OAuthConstants; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.authz.OAuthAuthzReqMessageContext; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AccessTokenRespDTO; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AuthorizeRespDTO; -import org.wso2.carbon.identity.oauth2.token.OAuthTokenReqMessageContext; - -import java.text.ParseException; -import java.util.Arrays; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import static org.wso2.carbon.identity.openidconnect.model.Constants.JWT_PART_DELIMITER; -import static org.wso2.carbon.identity.openidconnect.model.Constants.NUMBER_OF_PARTS_IN_JWE; - -/** - * Default OB specific claim provider implementation. - */ -public class OBDefaultClaimProvider extends OBClaimProvider { - - private static final Log log = LogFactory.getLog(OBDefaultClaimProvider.class); - - @Override - public Map getAdditionalClaims(OAuthAuthzReqMessageContext authAuthzReqMessageContext, - OAuth2AuthorizeRespDTO authorizeRespDTO) - throws IdentityOAuth2Exception { - - Map claims = new HashMap<>(); - String[] cachedRequests = null; - final String sessionDataKey = authAuthzReqMessageContext.getAuthorizationReqDTO().getSessionDataKey(); - if (StringUtils.isNotBlank(sessionDataKey)) { - cachedRequests = SessionDataCache.getInstance() - .getValueFromCache(new SessionDataCacheKey(sessionDataKey)).getParamMap().get("request"); - } - if (cachedRequests != null && !(cachedRequests[0].split(JWT_PART_DELIMITER).length == NUMBER_OF_PARTS_IN_JWE)) { - JSONObject requestBody = getRequestBodyFromCache(cachedRequests); - - /* State is an optional parameter, so the authorization server must successfully authenticate and - * must NOT return state nor s_hash. (FAPI1-ADV-5.2.2.1-5) - */ - final String state = requestBody.getAsString(OAuthConstants.OAuth20Params.STATE); - if (StringUtils.isNotEmpty(state)) { - claims.put(IdentityCommonConstants.S_HASH, IdentityCommonUtil.getHashValue(state, null)); - } else { - // state is empty, removing state from cache too - removeStateFromCache(sessionDataKey); - } - } - final String responseType = authAuthzReqMessageContext.getAuthorizationReqDTO().getResponseType(); - avoidSettingATHash(responseType, authorizeRespDTO, claims); - - return claims; - - } - - @Override - public Map getAdditionalClaims(OAuthTokenReqMessageContext tokenReqMessageContext, - OAuth2AccessTokenRespDTO tokenRespDTO) - throws IdentityOAuth2Exception { - - return new HashMap<>(); - } - - /** - * If response_type value is not 'code id_token token', avoid setting at_hash claim to the authorization - * endpoint id_token as it is OPTIONAL (OIDCC-3.3.2.11). - * - * @param responseType requested auth response_type - * @param authorizeRespDTO authorizeRespDTO - * @param claims returning claims map - */ - private void avoidSettingATHash(String responseType, OAuth2AuthorizeRespDTO authorizeRespDTO, - Map claims) { - - if (StringUtils.isNotBlank(responseType)) { - List responseTypes = Arrays.asList(responseType.trim().split("\\s+")); - if (!(responseTypes.contains(IdentityCommonConstants.CODE) - && responseTypes.contains(OAuthConstants.ID_TOKEN) - && responseTypes.contains(OAuthConstants.TOKEN))) { - if (StringUtils.isNotBlank(authorizeRespDTO.getAccessToken())) { - authorizeRespDTO.setAccessToken(null); - } - claims.put(OAuthConstants.OIDCClaims.AT_HASH, null); - } - } - } - - private JSONObject getRequestBodyFromCache(String[] cachedRequests) { - - try { - if (cachedRequests.length > 0) { - return JWTUtils.decodeRequestJWT(cachedRequests[0], "body"); - } - } catch (ParseException e) { - log.error("Exception occurred when decoding request. Caused by, ", e); - } - - return new JSONObject(); - } - - /** - * If request object state value is empty, ignore the session cache state value, as FAPI-RW says only parameters - * inside the request object should be used (FAPI1-ADV-5.2.2-10). - * - * @param sessionDataKey key used to store session cache - */ - private void removeStateFromCache(String sessionDataKey) { - - final SessionDataCacheKey sessionDataCacheKey = new SessionDataCacheKey(sessionDataKey); - SessionDataCacheEntry sessionDataCacheEntry = SessionDataCache.getInstance() - .getValueFromCache(sessionDataCacheKey); - - if (sessionDataCacheEntry != null) { - sessionDataCacheEntry.getoAuth2Parameters().setState(null); - sessionDataCacheEntry.getParamMap().put(OAuthConstants.OAuth20Params.STATE, new String[]{}); - - SessionDataCache.getInstance().addToCache(sessionDataCacheKey, sessionDataCacheEntry); - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/claims/OBDefaultOIDCClaimsCallbackHandler.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/claims/OBDefaultOIDCClaimsCallbackHandler.java deleted file mode 100644 index 52c8bdbb..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/claims/OBDefaultOIDCClaimsCallbackHandler.java +++ /dev/null @@ -1,170 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.claims; - -import com.nimbusds.jose.util.Base64URL; -import com.nimbusds.jose.util.X509CertUtils; -import com.nimbusds.jwt.JWTClaimsSet; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.CertificateUtils; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.model.HttpRequestHeader; -import org.wso2.carbon.identity.oauth2.token.OAuthTokenReqMessageContext; -import org.wso2.carbon.identity.openidconnect.DefaultOIDCClaimsCallbackHandler; - -import java.security.cert.X509Certificate; -import java.util.Arrays; -import java.util.Collections; -import java.util.HashMap; -import java.util.Map; -import java.util.Optional; - -/** - * This call back handler adds ob specific additional claims to self contained JWT access token. - */ -public class OBDefaultOIDCClaimsCallbackHandler extends DefaultOIDCClaimsCallbackHandler { - - private static Log log = LogFactory.getLog(OBDefaultOIDCClaimsCallbackHandler.class); - Map identityConfigurations = IdentityExtensionsDataHolder.getInstance().getConfigurationMap(); - - - @Override - public JWTClaimsSet handleCustomClaims(JWTClaimsSet.Builder jwtClaimsSetBuilder, OAuthTokenReqMessageContext - tokenReqMessageContext) throws IdentityOAuth2Exception { - - /* accessToken property check is done to omit the following claims getting bound to id_token - The access token property is added to the ID token message context before this method is invoked. */ - try { - if (IdentityCommonUtil.getRegulatoryFromSPMetaData(tokenReqMessageContext.getOauth2AccessTokenReqDTO() - .getClientId()) && (tokenReqMessageContext.getProperty("accessToken") == null)) { - - Map userClaimsInOIDCDialect = new HashMap<>(); - JWTClaimsSet jwtClaimsSet = getJwtClaimsFromSuperClass(jwtClaimsSetBuilder, tokenReqMessageContext); - if (jwtClaimsSet != null) { - for (Map.Entry claimEntry : jwtClaimsSet.getClaims().entrySet()) { - userClaimsInOIDCDialect.put(claimEntry.getKey(), claimEntry.getValue()); - } - } - addCnfClaimToOIDCDialect(tokenReqMessageContext, userClaimsInOIDCDialect); - addConsentIDClaimToOIDCDialect(tokenReqMessageContext, userClaimsInOIDCDialect); - updateSubClaim(tokenReqMessageContext, userClaimsInOIDCDialect); - - for (Map.Entry claimEntry : userClaimsInOIDCDialect.entrySet()) { - if (IdentityCommonConstants.SCOPE.equals(claimEntry.getKey())) { - String[] nonInternalScopes = IdentityCommonUtil - .removeInternalScopes(claimEntry.getValue().toString() - .split(IdentityCommonConstants.SPACE_SEPARATOR)); - jwtClaimsSetBuilder.claim(IdentityCommonConstants.SCOPE, StringUtils.join(nonInternalScopes, - IdentityCommonConstants.SPACE_SEPARATOR)); - } else { - jwtClaimsSetBuilder.claim(claimEntry.getKey(), claimEntry.getValue()); - } - } - return jwtClaimsSetBuilder.build(); - } - } catch (OpenBankingException e) { - throw new IdentityOAuth2Exception(e.getMessage(), e); - } - return super.handleCustomClaims(jwtClaimsSetBuilder, tokenReqMessageContext); - } - - @Generated(message = "Excluding from code coverage since it makes is used to return claims from the super class") - public JWTClaimsSet getJwtClaimsFromSuperClass(JWTClaimsSet.Builder jwtClaimsSetBuilder, - OAuthTokenReqMessageContext tokenReqMessageContext) - throws IdentityOAuth2Exception { - - return super.handleCustomClaims(jwtClaimsSetBuilder, tokenReqMessageContext); - } - - private void addCnfClaimToOIDCDialect(OAuthTokenReqMessageContext tokenReqMessageContext, - Map userClaimsInOIDCDialect) { - Base64URL certThumbprint; - X509Certificate certificate; - String headerName = IdentityCommonUtil.getMTLSAuthHeader(); - - HttpRequestHeader[] requestHeaders = tokenReqMessageContext.getOauth2AccessTokenReqDTO() - .getHttpRequestHeaders(); - Optional certHeader = - Arrays.stream(requestHeaders).filter(h -> headerName.equals(h.getName())).findFirst(); - if (certHeader.isPresent()) { - try { - certificate = CertificateUtils.parseCertificate(certHeader.get().getValue()[0]); - certThumbprint = X509CertUtils.computeSHA256Thumbprint(certificate); - userClaimsInOIDCDialect.put("cnf", Collections.singletonMap("x5t#S256", certThumbprint)); - } catch (OpenBankingException e) { - log.error("Error while extracting the certificate", e); - } - } - } - - private void addConsentIDClaimToOIDCDialect(OAuthTokenReqMessageContext tokenReqMessageContext, - Map userClaimsInOIDCDialect) { - - String consentIdClaimName = - identityConfigurations.get(IdentityCommonConstants.CONSENT_ID_CLAIM_NAME).toString(); - String consentID = Arrays.stream(tokenReqMessageContext.getScope()) - .filter(scope -> scope.contains(IdentityCommonConstants.OB_PREFIX)).findFirst().orElse(null); - if (StringUtils.isEmpty(consentID)) { - consentID = Arrays.stream(tokenReqMessageContext.getScope()) - .filter(scope -> scope.contains(consentIdClaimName)) - .findFirst().orElse(StringUtils.EMPTY) - .replaceAll(consentIdClaimName, StringUtils.EMPTY); - } else { - consentID = consentID.replace(IdentityCommonConstants.OB_PREFIX, StringUtils.EMPTY); - } - - if (StringUtils.isNotEmpty(consentID)) { - userClaimsInOIDCDialect.put(consentIdClaimName, consentID); - } - } - - /** - * Update the subject claim of the JWT claims set if any of the following configurations are true - * 1. Remove tenant domain from subject (open_banking.identity.token.remove_tenant_domain_from_subject) - * 2. Remove user store domain from subject (open_banking.identity.token.remove_user_store_domain_from_subject) - * @param tokenReqMessageContext token request message context - * @param userClaimsInOIDCDialect user claims in OIDC dialect as a map - */ - private void updateSubClaim(OAuthTokenReqMessageContext tokenReqMessageContext, - Map userClaimsInOIDCDialect) { - - Object removeTenantDomainConfig = - identityConfigurations.get(IdentityCommonConstants.REMOVE_TENANT_DOMAIN_FROM_SUBJECT); - Boolean removeTenantDomain = removeTenantDomainConfig != null - && Boolean.parseBoolean(removeTenantDomainConfig.toString()); - - Object removeUserStoreDomainConfig = - identityConfigurations.get(IdentityCommonConstants.REMOVE_USER_STORE_DOMAIN_FROM_SUBJECT); - Boolean removeUserStoreDomain = removeUserStoreDomainConfig != null - && Boolean.parseBoolean(removeUserStoreDomainConfig.toString()); - - if (removeTenantDomain || removeUserStoreDomain) { - String subClaim = tokenReqMessageContext.getAuthorizedUser() - .getUsernameAsSubjectIdentifier(!removeUserStoreDomain, !removeTenantDomain); - userClaimsInOIDCDialect.put("sub", subClaim); - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/claims/RoleClaimProviderImpl.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/claims/RoleClaimProviderImpl.java deleted file mode 100644 index a0612685..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/claims/RoleClaimProviderImpl.java +++ /dev/null @@ -1,101 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.claims; - -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import org.apache.commons.lang3.ArrayUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.base.IdentityRuntimeException; -import org.wso2.carbon.identity.core.util.IdentityTenantUtil; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.authz.OAuthAuthzReqMessageContext; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AccessTokenRespDTO; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AuthorizeRespDTO; -import org.wso2.carbon.identity.oauth2.token.OAuthTokenReqMessageContext; -import org.wso2.carbon.identity.openidconnect.ClaimProvider; -import org.wso2.carbon.user.api.UserStoreException; -import org.wso2.carbon.user.api.UserStoreManager; -import org.wso2.carbon.user.core.service.RealmService; - -import java.util.Arrays; -import java.util.Collections; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -/** - * RoleClaimProviderImpl. - *

- * Adding Customer Care Officer user role to cater sso flow in consent mgt react app - */ -public class RoleClaimProviderImpl implements ClaimProvider { - private static final Log LOG = LogFactory.getLog(RoleClaimProviderImpl.class); - private static final String USER_ROLE = "user_role"; - private static final String OPENID_SCOPE = "openid"; - private static final String CUSTOMER_CARE_OFFICER = "customerCareOfficer"; - private static final String CUSTOMER_CARE_OFFICER_ROLE = "Internal/CustomerCareOfficerRole"; - private static final String CUSTOMER_CARE_OFFICER_SCOPE = "consents:read_all"; - - @Generated(message = "Do not contain logics") - @Override - public Map getAdditionalClaims(OAuthAuthzReqMessageContext oAuthAuthzReqMessageContext, - OAuth2AuthorizeRespDTO oAuth2AuthorizeRespDTO) - throws IdentityOAuth2Exception { - return Collections.emptyMap(); - } - - /** - * Method to add Role based claims for Token response to cater sso flow in consent mgt react app. - * - * @param oAuthTokenReqMessageContext token Request message context - * @param oAuth2AccessTokenRespDTO token Response DTO - * @return Map of additional claims - * @throws IdentityOAuth2Exception when failed to obtain claims - */ - @Override - public Map getAdditionalClaims(OAuthTokenReqMessageContext oAuthTokenReqMessageContext, - OAuth2AccessTokenRespDTO oAuth2AccessTokenRespDTO) - throws IdentityOAuth2Exception { - Map claims = new HashMap<>(); - - List scopes = Arrays.asList(oAuthTokenReqMessageContext.getScope()); - if (scopes.contains(CUSTOMER_CARE_OFFICER_SCOPE) && scopes.contains(OPENID_SCOPE)) { - final String userId = oAuthTokenReqMessageContext.getAuthorizedUser().getUserName(); - - try { - int tenantId = IdentityTenantUtil.getTenantIdOfUser(userId); - RealmService realmService = IdentityExtensionsDataHolder.getInstance().getRealmService(); - UserStoreManager userStoreManager = realmService.getTenantUserRealm(tenantId).getUserStoreManager(); - - String[] roles = userStoreManager.getRoleListOfUser(userId); - if (ArrayUtils.contains(roles, CUSTOMER_CARE_OFFICER_ROLE)) { - claims.put(USER_ROLE, CUSTOMER_CARE_OFFICER); - } - } catch (IdentityRuntimeException e) { - LOG.error("Error in retrieving user tenant name for user: " + userId + ". Caused by,", e); - } catch (UserStoreException e) { - LOG.error("Error in retrieving user role. Caused by,", e); - } - - } - return claims; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/clientauth/OBMutualTLSClientAuthenticator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/clientauth/OBMutualTLSClientAuthenticator.java deleted file mode 100644 index 87c6f5ed..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/clientauth/OBMutualTLSClientAuthenticator.java +++ /dev/null @@ -1,99 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.clientauth; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonHelper; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.oauth2.bean.OAuthClientAuthnContext; -import org.wso2.carbon.identity.oauth2.client.authentication.OAuthClientAuthnException; -import org.wso2.carbon.identity.oauth2.token.handler.clientauth.mutualtls.MutualTLSClientAuthenticator; -import org.wso2.carbon.identity.oauth2.token.handler.clientauth.mutualtls.utils.MutualTLSUtil; - -import java.net.MalformedURLException; -import java.net.URL; -import java.util.List; -import java.util.Map; - -import javax.servlet.http.HttpServletRequest; - -/** - * OpenBanking Mutual TLS Client Authenticator. - */ -public class OBMutualTLSClientAuthenticator extends MutualTLSClientAuthenticator { - - private static Log log = LogFactory.getLog(OBMutualTLSClientAuthenticator.class); - - @Override - public boolean canAuthenticate(HttpServletRequest request, Map bodyParams, - OAuthClientAuthnContext oAuthClientAuthnContext) { - - try { - String clientId = oAuthClientAuthnContext.getClientId(); - if (StringUtils.isEmpty(clientId)) { - clientId = (super.getClientId(request, bodyParams, oAuthClientAuthnContext) == null - && request.getParameter("client_id") != null) ? request.getParameter("client_id") : - super.getClientId(request, bodyParams, oAuthClientAuthnContext); - } - if ((IdentityCommonUtil.getRegulatoryFromSPMetaData(clientId))) { - if (new IdentityCommonHelper().isMTLSAuthentication(request)) { - log.debug("Client ID and a valid certificate was found in the request attribute hence returning " + - "true."); - return true; - } else { - log.debug("Mutual TLS authenticator cannot handle this request. Client id is not available in " + - "body params or valid certificate not found in request attributes."); - return false; - } - } else { - return super.canAuthenticate(request, bodyParams, oAuthClientAuthnContext); - } - } catch (OpenBankingException | OAuthClientAuthnException e) { - if (log.isDebugEnabled()) { - log.debug("Mutual TLS authenticator cannot handle this request. " + e.getMessage()); - } - return false; - } - } - - @Override - public URL getJWKSEndpointOfSP(ServiceProvider serviceProvider, String clientID) throws OAuthClientAuthnException { - - String jwksUri = MutualTLSUtil.getPropertyValue(serviceProvider, IdentityCommonUtil.getJWKURITransportCert()); - if (StringUtils.isEmpty(jwksUri)) { - throw new OAuthClientAuthnException("jwks endpoint not configured for the service provider for client ID: " - + clientID, "server_error"); - } else { - try { - URL url = new URL(jwksUri); - if (log.isDebugEnabled()) { - log.debug("Configured JWKS URI found: " + jwksUri); - } - - return url; - } catch (MalformedURLException var6) { - throw new OAuthClientAuthnException("URL might be malformed " + clientID, "server_error", var6); - } - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/IdentityServiceExporter.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/IdentityServiceExporter.java deleted file mode 100644 index fb03dfba..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/IdentityServiceExporter.java +++ /dev/null @@ -1,37 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.common; - -import org.wso2.carbon.identity.oauth2.client.authentication.OAuthClientAuthnService; - -/** - * Exporter service to facilitate access to identity services in data holder from other modules. - */ -public class IdentityServiceExporter { - - private static OAuthClientAuthnService oAuthClientAuthnService; - - public static OAuthClientAuthnService getOAuthClientAuthnService() { - return oAuthClientAuthnService; - } - - public static void setOAuthClientAuthnService(OAuthClientAuthnService oAuthClientAuthnService) { - IdentityServiceExporter.oAuthClientAuthnService = oAuthClientAuthnService; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/annotations/validationgroups/AttributeChecks.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/annotations/validationgroups/AttributeChecks.java deleted file mode 100644 index 4fd54abc..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/annotations/validationgroups/AttributeChecks.java +++ /dev/null @@ -1,28 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.common.annotations.validationgroups; - -/** - * Interface for grouping the validation annotations. - * Groups the validations for attributes - */ -public interface AttributeChecks { - -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/annotations/validationgroups/MandatoryChecks.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/annotations/validationgroups/MandatoryChecks.java deleted file mode 100644 index 6f2a082e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/annotations/validationgroups/MandatoryChecks.java +++ /dev/null @@ -1,28 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.common.annotations.validationgroups; - -/** - * Interface for grouping the validation annotations. - * Grouping the mandatory check constraints - */ -public interface MandatoryChecks { - -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/annotations/validationgroups/SignatureCheck.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/annotations/validationgroups/SignatureCheck.java deleted file mode 100644 index 25d302c6..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/annotations/validationgroups/SignatureCheck.java +++ /dev/null @@ -1,28 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.common.annotations.validationgroups; - -/** - * Interface for grouping the validation annotations. - * Groups the validation for signature - */ -public interface SignatureCheck { - -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/annotations/validationgroups/ValidityChecks.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/annotations/validationgroups/ValidityChecks.java deleted file mode 100644 index 477db36a..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/common/annotations/validationgroups/ValidityChecks.java +++ /dev/null @@ -1,28 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.common.annotations.validationgroups; - -/** - * Interface for grouping the validation annotations. - * Groups the validations for the validity of a JWT - */ -public interface ValidityChecks { - -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/exception/DCRValidationException.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/exception/DCRValidationException.java deleted file mode 100644 index dc8c3337..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/exception/DCRValidationException.java +++ /dev/null @@ -1,63 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.exception; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; - -/** - * DCR validation exception. - */ -public class DCRValidationException extends OpenBankingException { - - private String errorDescription; - private String errorCode; - - public String getErrorDescription() { - - return errorDescription; - } - - public void setErrorDescription(String errorDescription) { - - this.errorDescription = errorDescription; - } - - public String getErrorCode() { - - return errorCode; - } - - public void setErrorCode(String errorCode) { - - this.errorCode = errorCode; - } - - public DCRValidationException(String errorCode, String error, String errorDescription, Throwable e) { - - super(error, e); - this.errorDescription = errorDescription; - this.errorCode = errorCode; - } - - public DCRValidationException(String errorCode, String errorDescription) { - - super(errorDescription); - this.errorDescription = errorDescription; - this.errorCode = errorCode; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/model/RegistrationError.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/model/RegistrationError.java deleted file mode 100644 index 698f7239..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/model/RegistrationError.java +++ /dev/null @@ -1,47 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.model; - -/** - * Model class for dcr error response. - */ -public class RegistrationError { - - private String errorMessage; - private String errorCode; - - public String getErrorMessage() { - - return errorMessage; - } - - public void setErrorMessage(String errorMessage) { - - this.errorMessage = errorMessage; - } - - public String getErrorCode() { - - return errorCode; - } - - public void setErrorCode(String errorCode) { - - this.errorCode = errorCode; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/model/RegistrationRequest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/model/RegistrationRequest.java deleted file mode 100644 index 44663a5c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/model/RegistrationRequest.java +++ /dev/null @@ -1,366 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.model; - -import com.google.gson.annotations.SerializedName; -import com.wso2.openbanking.accelerator.identity.common.annotations.validationgroups.AttributeChecks; -import com.wso2.openbanking.accelerator.identity.common.annotations.validationgroups.MandatoryChecks; -import com.wso2.openbanking.accelerator.identity.common.annotations.validationgroups.SignatureCheck; -import com.wso2.openbanking.accelerator.identity.dcr.validation.DCRCommonConstants; -import com.wso2.openbanking.accelerator.identity.dcr.validation.annotation.ValidateAlgorithm; -import com.wso2.openbanking.accelerator.identity.dcr.validation.annotation.ValidateIssuer; -import com.wso2.openbanking.accelerator.identity.dcr.validation.annotation.ValidateRequiredParams; -import com.wso2.openbanking.accelerator.identity.dcr.validation.annotation.ValidateSignature; - -import java.util.List; -import java.util.Map; - -/** - * Model class for dcr registration request. - */ -@ValidateRequiredParams(message = "Required parameters cannot be null or empty:" + DCRCommonConstants.INVALID_META_DATA, - groups = MandatoryChecks.class) -@ValidateIssuer(issuerProperty = "issuer", ssa = "softwareStatement", - message = "Invalid issuer:" + DCRCommonConstants.INVALID_META_DATA, groups = AttributeChecks.class) -@ValidateSignature(ssaBody = "softwareStatementBody", ssa = "softwareStatement", message = "Invalid signature for SSA:" - + DCRCommonConstants.INVALID_SSA, groups = SignatureCheck.class) -@ValidateAlgorithm(idTokenAlg = "idTokenSignedResponseAlg", reqObjAlg = "requestObjectSigningAlg", - tokenAuthAlg = "tokenEndPointAuthSigningAlg", - message = "Invalid signing algorithm sent:" + DCRCommonConstants.INVALID_META_DATA, - groups = AttributeChecks.class) -public class RegistrationRequest { - - @SerializedName("aud") - private String aud; - - @SerializedName("iss") - private String issuer; - - @SerializedName("token_endpoint_auth_method") - private String tokenEndPointAuthMethod; - - @SerializedName("jwks_uri") - private String jwksURI; - - @SerializedName("grant_types") - private List grantTypes; - - @SerializedName("software_statement") - private String softwareStatement; - - @SerializedName("id_token_signed_response_alg") - private String idTokenSignedResponseAlg; - - @SerializedName("redirect_uris") - private List redirectUris; - - @SerializedName("token_endpoint_auth_signing_alg") - private String tokenEndPointAuthSigningAlg; - - @SerializedName("response_types") - private List responseTypes; - - @SerializedName("software_id") - private String softwareId; - - @SerializedName("scope") - private String scope; - - @SerializedName("application_type") - private String applicationType; - - @SerializedName("jti") - private String jti; - - @SerializedName("id_token_encrypted_response_alg") - private String idTokenEncryptionResponseAlg; - - @SerializedName("id_token_encrypted_response_enc") - private String idTokenEncryptionResponseEnc; - - @SerializedName("request_object_signing_alg") - private String requestObjectSigningAlg; - - @SerializedName("tls_client_auth_subject_dn") - private String tlsClientAuthSubjectDn; - - @SerializedName("backchannel_token_delivery_mode") - private String backchannelTokenDeliveryMode; - - @SerializedName("backchannel_authentication_request_signing_alg") - private String backchannelAuthenticationRequestSigningAlg; - - @SerializedName("backchannel_client_notification_endpoint") - private String backchannelClientNotificationEndpoint; - - @SerializedName("backchannel_user_code_parameter_supported") - private boolean backchannelUserCodeParameterSupported; - - private SoftwareStatementBody softwareStatementBody; - - private Map requestParameters; - - private Map ssaParameters; - - public Map getSsaParameters() { - - return ssaParameters; - } - - public void setSsaParameters(Map ssaParameters) { - - this.ssaParameters = ssaParameters; - } - - public Map getRequestParameters() { - - return requestParameters; - } - - public void setRequestParameters(Map requestParameters) { - - this.requestParameters = requestParameters; - } - - public SoftwareStatementBody getSoftwareStatementBody() { - - return softwareStatementBody; - } - - public void setSoftwareStatementBody(SoftwareStatementBody softwareStatementBody) { - - this.softwareStatementBody = softwareStatementBody; - } - - public boolean getBackchannelUserCodeParameterSupported() { - - return backchannelUserCodeParameterSupported; - } - - public void setBackchannelUserCodeParameterSupported(boolean backchannelUserCodeParameterSupported) { - - this.backchannelUserCodeParameterSupported = backchannelUserCodeParameterSupported; - } - - public String getBackchannelClientNotificationEndpoint() { - - return backchannelClientNotificationEndpoint; - } - - public void setBackchannelClientNotificationEndpoint(String backchannelClientNotificationEndpoint) { - - this.backchannelClientNotificationEndpoint = backchannelClientNotificationEndpoint; - } - - public String getBackchannelAuthenticationRequestSigningAlg() { - - return backchannelAuthenticationRequestSigningAlg; - } - - public void setBackchannelAuthenticationRequestSigningAlg(String backchannelAuthenticationRequestSigningAlg) { - - this.backchannelAuthenticationRequestSigningAlg = backchannelAuthenticationRequestSigningAlg; - } - - public String getBackchannelTokenDeliveryMode() { - - return backchannelTokenDeliveryMode; - } - - public void setBackchannelTokenDeliveryMode(String backchannelTokenDeliveryMode) { - - this.backchannelTokenDeliveryMode = backchannelTokenDeliveryMode; - } - - public String getTlsClientAuthSubjectDn() { - - return tlsClientAuthSubjectDn; - } - - public void setTlsClientAuthSubjectDn(String tlsClientAuthSubjectDn) { - - this.tlsClientAuthSubjectDn = tlsClientAuthSubjectDn; - } - - public String getRequestObjectSigningAlg() { - - return requestObjectSigningAlg; - } - - public void setRequestObjectSigningAlg(String requestObjectSigningAlg) { - - this.requestObjectSigningAlg = requestObjectSigningAlg; - } - - public String getIdTokenEncryptionResponseEnc() { - - return idTokenEncryptionResponseEnc; - } - - public void setIdTokenEncryptionResponseEnc(String idTokenEncryptionResponseEnc) { - - this.idTokenEncryptionResponseEnc = idTokenEncryptionResponseEnc; - } - - public String getIdTokenEncryptionResponseAlg() { - - return idTokenEncryptionResponseAlg; - } - - public void setIdTokenEncryptionResponseAlg(String idTokenEncryptionResponseAlg) { - - this.idTokenEncryptionResponseAlg = idTokenEncryptionResponseAlg; - } - - public String getApplicationType() { - - return applicationType; - } - - public void setApplicationType(String applicationType) { - - this.applicationType = applicationType; - } - - public String getScope() { - - return scope; - } - - public void setScope(String scope) { - - this.scope = scope; - } - - public String getSoftwareId() { - - return softwareId; - } - - public void setSoftwareId(String softwareId) { - - this.softwareId = softwareId; - } - - public List getResponseTypes() { - - return responseTypes; - } - - public void setResponseTypes(List responseTypes) { - - this.responseTypes = responseTypes; - } - - public String getTokenEndPointAuthSigningAlg() { - - return tokenEndPointAuthSigningAlg; - } - - public void setTokenEndPointAuthSigningAlg(String tokenEndPointAuthSigningAlg) { - - this.tokenEndPointAuthSigningAlg = tokenEndPointAuthSigningAlg; - } - - public List getCallbackUris() { - - return redirectUris; - } - - public void setCallbackUris(List redirectUris) { - - this.redirectUris = redirectUris; - } - - public String getIssuer() { - - return issuer; - } - - public void setIssuer(String issuer) { - - this.issuer = issuer; - } - - public String getTokenEndPointAuthentication() { - - return tokenEndPointAuthMethod; - } - - public void setTokenEndPointAuthentication(String tokenEndPointAuthMethod) { - - this.tokenEndPointAuthMethod = tokenEndPointAuthMethod; - } - - public List getGrantTypes() { - - return grantTypes; - } - - public void setGrantTypes(List grantTypes) { - - this.grantTypes = grantTypes; - } - - public String getSoftwareStatement() { - - return softwareStatement; - } - - public void setSoftwareStatement(String softwareStatement) { - - this.softwareStatement = softwareStatement; - } - - public String getIdTokenSignedResponseAlg() { - - return idTokenSignedResponseAlg; - } - - public void setIdTokenSignedResponseAlg(String idTokenSignedResponseAlg) { - - this.idTokenSignedResponseAlg = idTokenSignedResponseAlg; - } - - public String getAudience() { - - return aud; - } - public void setAudience(String aud) { - - this.aud = aud; - } - - public String getJti() { - - return jti; - } - - public void setJti(String jti) { - - this.jti = jti; - } - - public String getJwksURI() { - return jwksURI; - } - - public void setJwksURI(String jwksURI) { - this.jwksURI = jwksURI; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/model/RegistrationResponse.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/model/RegistrationResponse.java deleted file mode 100644 index 8078f34a..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/model/RegistrationResponse.java +++ /dev/null @@ -1,220 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.model; - -import com.google.gson.annotations.SerializedName; - -import java.util.ArrayList; -import java.util.List; - -/** - * Model class for dcr response containing common attributes. - */ -public class RegistrationResponse { - - public String getToken() { - return this.token; - } - - public void setToken(String token) { - this.token = token; - } - - @SerializedName("registration_access_token") - protected String token = null; - - @SerializedName("client_id") - protected String clientId = null; - - @SerializedName("client_id_issued_at") - protected String clientIdIssuedAt = null; - - @SerializedName("redirect_uris") - protected List redirectUris = new ArrayList<>(); - - @SerializedName("grant_types") - protected List grantTypes = new ArrayList<>(); - - @SerializedName("response_types") - protected List responseTypes = new ArrayList<>(); - - @SerializedName("application_type") - protected String applicationType = null; - - @SerializedName("id_token_signed_response_alg") - protected String idTokenSignedResponseAlg = null; - - @SerializedName("request_object_signing_alg") - protected String requestObjectSigningAlg = null; - - @SerializedName("scope") - protected String scope = null; - - @SerializedName("software_id") - protected String softwareId = null; - - @SerializedName("jwks_uri") - private String jwksURI; - - @SerializedName("token_endpoint_auth_method") - protected String tokenEndpointAuthMethod = null; - - @SerializedName("registration_client_uri") - protected String registrationClientURI = null; - - @SerializedName("software_statement") - protected String softwareStatement = null; - - public String getSoftwareStatement() { - - return softwareStatement; - } - - public void setSoftwareStatement(String softwareStatement) { - - this.softwareStatement = softwareStatement; - } - - public String getTokenEndpointAuthMethod() { - - return tokenEndpointAuthMethod; - } - - public void setTokenEndpointAuthMethod(String tokenEndpointAuthMethod) { - - this.tokenEndpointAuthMethod = tokenEndpointAuthMethod; - } - - public List getResponseTypes() { - - return responseTypes; - } - - public void setResponseTypes(List responseTypes) { - - this.responseTypes = responseTypes; - } - - - public String getClientIdIssuedAt() { - - return clientIdIssuedAt; - } - - public void setClientIdIssuedAt(String clientIdIssuedAt) { - - this.clientIdIssuedAt = clientIdIssuedAt; - } - - public String getClientId() { - - return clientId; - } - - public void setClientId(String clientId) { - - this.clientId = clientId; - } - - public List getRedirectUris() { - - return redirectUris; - } - - public void setRedirectUris(List redirectUris) { - - this.redirectUris = redirectUris; - } - - public List getGrantTypes() { - - return grantTypes; - } - - public void setGrantTypes(List grantTypes) { - - this.grantTypes = grantTypes; - } - - public String getApplicationType() { - - return applicationType; - } - - public void setApplicationType(String applicationType) { - - this.applicationType = applicationType; - } - - public String getIdTokenSignedResponseAlg() { - - return idTokenSignedResponseAlg; - } - - public void setIdTokenSignedResponseAlg(String idTokenSignedResponseAlg) { - - this.idTokenSignedResponseAlg = idTokenSignedResponseAlg; - } - - public String getRequestObjectSigningAlg() { - - return requestObjectSigningAlg; - } - - public void setRequestObjectSigningAlg(String requestObjectSigningAlg) { - - this.requestObjectSigningAlg = requestObjectSigningAlg; - } - - public String getScope() { - - return scope; - } - - public void setScope(String scope) { - - this.scope = scope; - } - - public String getSoftwareId() { - - return softwareId; - } - - public void setSoftwareId(String softwareId) { - - this.softwareId = softwareId; - } - - public String getRegistrationClientURI() { - return registrationClientURI; - } - - public void setRegistrationClientURI(String registrationClientURI) { - this.registrationClientURI = registrationClientURI; - } - - public String getJwksURI() { - return jwksURI; - } - - public void setJwksURI(String jwksURI) { - this.jwksURI = jwksURI; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/model/SoftwareStatementBody.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/model/SoftwareStatementBody.java deleted file mode 100644 index c11651c8..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/model/SoftwareStatementBody.java +++ /dev/null @@ -1,145 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.model; - -import com.google.gson.annotations.SerializedName; - -import java.util.List; - -/** - * Model class containing common attributes for software statement. - */ -public class SoftwareStatementBody { - - @SerializedName("software_environment") - private String softwareEnvironment; - - @SerializedName("software_id") - private String softwareId; - - @SerializedName("org_id") - private String orgId; - - @SerializedName("org_name") - private String orgName; - - @SerializedName("scope") - private String scopes; - - @SerializedName(value = "software_client_name" , alternate = "client_name") - private String clientName; - - @SerializedName(value = "software_redirect_uris", alternate = "redirect_uris") - private List ssaRedirectURIs; - - @SerializedName(value = "software_jwks_endpoint", alternate = "jwks_uri") - private String jwksURI; - - @SerializedName("iss") - private String ssaIssuer; - - public String getSsaIssuer() { - - return ssaIssuer; - } - - public void setSsaIssuer(String ssaIssuer) { - - this.ssaIssuer = ssaIssuer; - } - - public String getJwksURI() { - - return jwksURI; - } - - public void setJwksURI(String jwksURI) { - - this.jwksURI = jwksURI; - } - - public List getCallbackUris() { - - return ssaRedirectURIs; - } - - public void setCallbackUris(List redirectURIs) { - - this.ssaRedirectURIs = redirectURIs; - } - - public String getClientName() { - - return clientName; - } - - public void setClientName(String clientName) { - - this.clientName = clientName; - } - - public String getScopes() { - - return scopes; - } - - public void setScopes(String scopes) { - - this.scopes = scopes; - } - - public String getSoftwareEnvironment() { - - return softwareEnvironment; - } - - public void setSoftwareEnvironment(String softwareEnvironment) { - - this.softwareEnvironment = softwareEnvironment; - } - - public String getSoftwareId() { - - return softwareId; - } - - public void setSoftwareId(String softwareId) { - - this.softwareId = softwareId; - } - - public String getOrgId() { - - return orgId; - } - - public void setOrgId(String orgId) { - - this.orgId = orgId; - } - - public String getOrgName() { - - return orgName; - } - - public void setOrgName(String orgName) { - - this.orgName = orgName; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/utils/ValidatorUtils.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/utils/ValidatorUtils.java deleted file mode 100644 index ec7b9c92..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/utils/ValidatorUtils.java +++ /dev/null @@ -1,102 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.utils; - -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.common.validator.OpenBankingValidator; -import com.wso2.openbanking.accelerator.identity.dcr.exception.DCRValidationException; -import com.wso2.openbanking.accelerator.identity.dcr.model.RegistrationRequest; -import com.wso2.openbanking.accelerator.identity.dcr.validation.validationgroups.ValidationOrder; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.oauth2.OAuth2Service; -import org.wso2.carbon.identity.oauth2.bean.OAuthClientAuthnContext; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AccessTokenReqDTO; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AccessTokenRespDTO; -import org.wso2.carbon.identity.oauth2.model.HttpRequestHeader; - -/** - * Util class for validation logic implementation. - */ -public class ValidatorUtils { - - private static final Log log = LogFactory.getLog(ValidatorUtils.class); - - public static void getValidationViolations(RegistrationRequest registrationRequest) - throws DCRValidationException { - - String error = OpenBankingValidator.getInstance().getFirstViolation(registrationRequest, ValidationOrder.class); - if (error != null) { - String[] errors = error.split(":"); - throw new DCRValidationException(errors[1], errors[0]); - } - - } - - /** - * Create client credentials grant access token with PK JWT for DCR. - * @param clientId client ID - * @param tlsCert transport certificate - * @return String access token - */ - @Generated(message = "Excluding from code coverage since it requires a service call") - public static String generateAccessToken(String clientId, String tlsCert) { - OAuth2AccessTokenReqDTO tokenReqDTO = new OAuth2AccessTokenReqDTO(); - OAuthClientAuthnContext oauthClientAuthnContext = new OAuthClientAuthnContext(); - oauthClientAuthnContext.setClientId(clientId); - oauthClientAuthnContext.addAuthenticator(IdentityCommonConstants.PRIVATE_KEY); - oauthClientAuthnContext.setAuthenticated(true); - - tokenReqDTO.setoAuthClientAuthnContext(oauthClientAuthnContext); - tokenReqDTO.setGrantType(IdentityCommonConstants.CLIENT_CREDENTIALS); - tokenReqDTO.setClientId(clientId); - - String[] scopes = new String[2]; - - //add the appropriate scopes - scopes[0] = IdentityCommonUtil.getDCRScope(); - scopes[1] = IdentityCommonConstants.OPENID_SCOPE; - tokenReqDTO.setScope(scopes); - tokenReqDTO.setTenantDomain(IdentityCommonConstants.CARBON_SUPER); - - // set the tls cert as a header to bind the cnf value to the token - HttpRequestHeader[] requestHeaders = new HttpRequestHeader[1]; - requestHeaders[0] = new HttpRequestHeader(IdentityCommonUtil.getMTLSAuthHeader(), tlsCert); - tokenReqDTO.setHttpRequestHeaders(requestHeaders); - - tokenReqDTO.addAuthenticationMethodReference(IdentityCommonConstants.CLIENT_CREDENTIALS); - - OAuth2Service oAuth2Service = new OAuth2Service(); - OAuth2AccessTokenRespDTO tokenRespDTO = oAuth2Service.issueAccessToken(tokenReqDTO); - - return tokenRespDTO.getAccessToken(); - } - - /** - * Get Registration Client URI. - * @return String Registration client URI - */ - public static String getRegistrationClientURI() { - return String.valueOf(IdentityExtensionsDataHolder.getInstance() - .getConfigurationMap().getOrDefault(IdentityCommonConstants.DCR_REGISTRATION_CLIENT_URI, - IdentityCommonConstants.DEFAULT_REGISTRATION_CLIENT_URI)); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/AlgorithmValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/AlgorithmValidator.java deleted file mode 100644 index 61d9a0a3..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/AlgorithmValidator.java +++ /dev/null @@ -1,89 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation; - -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.identity.dcr.validation.annotation.ValidateAlgorithm; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import org.apache.commons.beanutils.BeanUtils; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.lang.reflect.InvocationTargetException; -import java.util.ArrayList; -import java.util.List; - -import javax.validation.ConstraintValidator; -import javax.validation.ConstraintValidatorContext; - -/** - * Validator class for validating the allowed algorithms. - */ -public class AlgorithmValidator implements ConstraintValidator { - - private String idTokenSigningAlgPath; - private String requestObjectSigningAlgPath; - private String tokenAuthSignignAlgPath; - private static Log log = LogFactory.getLog(AlgorithmValidator.class); - - @Override - public void initialize(ValidateAlgorithm validateAlgorithm) { - - this.idTokenSigningAlgPath = validateAlgorithm.idTokenAlg(); - this.requestObjectSigningAlgPath = validateAlgorithm.reqObjAlg(); - this.tokenAuthSignignAlgPath = validateAlgorithm.tokenAuthAlg(); - } - - @Override - public boolean isValid(Object object, ConstraintValidatorContext constraintValidatorContext) { - - List allowedAlgorithmsList = new ArrayList<>(); - Object allowedAlgorithms = IdentityExtensionsDataHolder.getInstance() - .getConfigurationMap().get(OpenBankingConstants.SIGNATURE_ALGORITHMS); - if (allowedAlgorithms instanceof List) { - allowedAlgorithmsList = (List) allowedAlgorithms; - } else { - allowedAlgorithmsList.add(allowedAlgorithms.toString()); - } - String requestedIdTokenSigningAlg = null; - String requestedRequestObjSignignAlg = null; - String requestedTokenAuthSigningAlg = null; - try { - requestedIdTokenSigningAlg = BeanUtils.getProperty(object, idTokenSigningAlgPath); - requestedRequestObjSignignAlg = BeanUtils.getProperty(object, requestObjectSigningAlgPath); - requestedTokenAuthSigningAlg = BeanUtils.getProperty(object, tokenAuthSignignAlgPath); - } catch (IllegalAccessException | InvocationTargetException | NoSuchMethodException e) { - log.error("Error while resolving validation fields", e); - return false; - } - if (StringUtils.isNotEmpty(requestedIdTokenSigningAlg) && - !allowedAlgorithmsList.contains(requestedIdTokenSigningAlg)) { - return false; - } - if (StringUtils.isNotEmpty(requestedRequestObjSignignAlg) && - !allowedAlgorithmsList.contains(requestedRequestObjSignignAlg)) { - return false; - } - if (StringUtils.isNotEmpty(requestedTokenAuthSigningAlg) && - !allowedAlgorithmsList.contains(requestedTokenAuthSigningAlg)) { - return false; - } - return true; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/DCRCommonConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/DCRCommonConstants.java deleted file mode 100644 index 235aac36..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/DCRCommonConstants.java +++ /dev/null @@ -1,46 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation; - -/** - * Common constants for dcr. - */ -public class DCRCommonConstants { - - public static final String SOFTWARE_ID = "software_id"; - public static final String INVALID_META_DATA = "invalid_client_metadata"; - public static final String INVALID_SSA = "invalid_software_statement"; - - public static final String DCR_VALIDATOR = "DCR.Validator"; - public static final String DCR_JWKS_ENDPOINT_SANDBOX = "DCR.JwksUrlSandbox"; - public static final String DCR_JWKS_ENDPOINT_PRODUCTION = "DCR.JwksUrlProduction"; - public static final String DCR_JWKS_CONNECTION_TIMEOUT = "DCR.JWKS-Retriever.ConnectionTimeout"; - public static final String DCR_JWKS_READ_TIMEOUT = "DCR.JWKS-Retriever.ReadTimeout"; - public static final String ENVIRONMENT_PROD = "production"; - public static final String ENVIRONMENT_SANDBOX = "sandbox"; - public static final String ARRAY_ELEMENT_SEPERATOR = "#"; - public static final String DUPLICATE_APPLICATION_NAME = "CONFLICT_EXISTING_APPLICATION"; - public static final String DCR_REGISTRATION_PARAM_SCOPE = "scope"; - public static final String DCR_REGISTRATION_PARAM_REQUIRED = "Required"; - public static final String DCR_REGISTRATION_PARAM_ALLOWED_VALUES = "AllowedValues"; - public static final String DCR_REGISTRATION_PARAM_REQUIRED_TRUE = "true"; - - public static final String POST_APPLICATION_LISTENER = "DCR.ApplicationUpdaterImpl"; - public static final String SOFTWARE_STATEMENT = "software_statement"; - public static final String REGULATORY_ISSUERS = "DCR.RegulatoryIssuers.Issuer"; -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/DefaultRegistrationValidatorImpl.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/DefaultRegistrationValidatorImpl.java deleted file mode 100644 index b7961df3..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/DefaultRegistrationValidatorImpl.java +++ /dev/null @@ -1,105 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation; - -import com.google.gson.Gson; -import com.google.gson.GsonBuilder; -import com.google.gson.JsonElement; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.identity.dcr.exception.DCRValidationException; -import com.wso2.openbanking.accelerator.identity.dcr.model.RegistrationRequest; -import com.wso2.openbanking.accelerator.identity.dcr.model.RegistrationResponse; -import com.wso2.openbanking.accelerator.identity.dcr.model.SoftwareStatementBody; -import com.wso2.openbanking.accelerator.identity.dcr.utils.ValidatorUtils; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.Map; - -/** - * Default implementation for dcr registration VALIDATOR class. - */ -public class DefaultRegistrationValidatorImpl extends RegistrationValidator { - - private static final Log log = LogFactory.getLog(DefaultRegistrationValidatorImpl.class); - - @Override - public void validatePost(RegistrationRequest registrationRequest) throws DCRValidationException { - - } - - @Override - public void validateGet(String clientId) throws DCRValidationException { - - } - - @Override - public void validateDelete(String clientId) throws DCRValidationException { - - } - - @Override - public void validateUpdate(RegistrationRequest registrationRequest) throws DCRValidationException { - - } - - /** - * method to set the software statement payload according to the specification. - * - * @param registrationRequest model containing the dcr registration details - * @param decodedSSA decoded json string of the softwarestatement payload - */ - public void setSoftwareStatementPayload(RegistrationRequest registrationRequest, String decodedSSA) { - - SoftwareStatementBody softwareStatementPayload = new GsonBuilder().create() - .fromJson(decodedSSA, SoftwareStatementBody.class); - registrationRequest.setSoftwareStatementBody(softwareStatementPayload); - - } - - @Override - @Generated(message = "Excluding from code coverage since it requires to load JWKS URI and invoke service calls") - public String getRegistrationResponse(Map spMetaData) { - - // Append registration access token and registration client URI to the DCR response if the config is enabled - if (IdentityCommonUtil.getDCRModifyResponseConfig()) { - - String tlsCert = spMetaData.get(IdentityCommonConstants.TLS_CERT).toString(); - - String clientId = spMetaData.get(IdentityCommonConstants.CLIENT_ID).toString(); - - if (!spMetaData.containsKey(IdentityCommonConstants.REGISTRATION_ACCESS_TOKEN)) { - // add the access token to the response - spMetaData.put(IdentityCommonConstants.REGISTRATION_ACCESS_TOKEN, - ValidatorUtils.generateAccessToken(clientId, tlsCert)); - } - - // add the dcr url to the response with the client id appended at the end - spMetaData.put(IdentityCommonConstants.REGISTRATION_CLIENT_URI, - ValidatorUtils.getRegistrationClientURI() + clientId); - } - - Gson gson = new Gson(); - JsonElement jsonElement = gson.toJsonTree(spMetaData); - RegistrationResponse registrationResponse = gson.fromJson(jsonElement, RegistrationResponse.class); - return gson.toJson(registrationResponse); - - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/IssuerValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/IssuerValidator.java deleted file mode 100644 index a7dfc3ef..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/IssuerValidator.java +++ /dev/null @@ -1,73 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation; - -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.identity.dcr.validation.annotation.ValidateIssuer; -import org.apache.commons.beanutils.BeanUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.lang.reflect.InvocationTargetException; -import java.text.ParseException; - -import javax.validation.ConstraintValidator; -import javax.validation.ConstraintValidatorContext; - -/** - * Validator class for validating the issuer of the registration request. - */ -public class IssuerValidator implements ConstraintValidator { - - private static final Log log = LogFactory.getLog(IssuerValidator.class); - - private String issuerPath; - private String ssaPath; - - @Override - public void initialize(ValidateIssuer validateIssuer) { - - this.issuerPath = validateIssuer.issuerProperty(); - this.ssaPath = validateIssuer.ssa(); - } - - @Override - public boolean isValid(Object registrationRequest, ConstraintValidatorContext constraintValidatorContext) { - - try { - String issuer = BeanUtils.getProperty(registrationRequest, issuerPath); - String softwareStatement = BeanUtils.getProperty(registrationRequest, ssaPath); - if (issuer != null && softwareStatement != null) { - String softwareId = JWTUtils.decodeRequestJWT(softwareStatement, "body") - .getAsString(DCRCommonConstants.SOFTWARE_ID); - if (softwareId != null && softwareId.equals(issuer)) { - return true; - } - } else { - return true; - } - - } catch (IllegalAccessException | InvocationTargetException | NoSuchMethodException e) { - log.error("Error while resolving validation fields", e); - } catch (ParseException e) { - log.error("Error while parsing the softwareStatement", e); - } - - return false; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/RegistrationValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/RegistrationValidator.java deleted file mode 100644 index 5390f667..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/RegistrationValidator.java +++ /dev/null @@ -1,91 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation; - -import com.wso2.openbanking.accelerator.identity.dcr.exception.DCRValidationException; -import com.wso2.openbanking.accelerator.identity.dcr.model.RegistrationRequest; - -import java.util.Map; - -/** - * Abstract class to perform spec specific validation for each crud operation. - * Implementation class for this should be configured in open-banking.xml - */ -public abstract class RegistrationValidator { - - private static RegistrationValidator registrationValidator; - - public static RegistrationValidator getRegistrationValidator() { - - return registrationValidator; - } - - public static void setRegistrationValidator(RegistrationValidator registrationValidator) { - - RegistrationValidator.registrationValidator = registrationValidator; - } - - /** - * method to set the software statement payload according to the specification. - * - * @param registrationRequest model containing the dcr registration details - * @param decodedSSA decoded json string of the softwarestatement payload - */ - public abstract void setSoftwareStatementPayload(RegistrationRequest registrationRequest, String decodedSSA); - - /** - * validate the request parameters when creating a registration. - * - * @param registrationRequest request - * @throws DCRValidationException if any validation failure occurs - */ - public abstract void validatePost(RegistrationRequest registrationRequest) throws DCRValidationException; - - /** - * do any validations before retrieving created registration details. - * - * @param clientId client ID of the registered application - * @throws DCRValidationException if any validation failure occurs - */ - public abstract void validateGet(String clientId) throws DCRValidationException; - - /** - * do any validations before deleting a created application. - * - * @param clientId client ID of the registered application - * @throws DCRValidationException if any validation failure occurs - */ - public abstract void validateDelete(String clientId) throws DCRValidationException; - - /** - * validate the request parameters when creating a registration. - * - * @param registrationRequest request - * @throws DCRValidationException if any validation failure occurs - */ - public abstract void validateUpdate(RegistrationRequest registrationRequest) throws DCRValidationException; - - /** - * method to return the response according to the implemented specification when retrieving registered data. - * - * @param clientMetaData object map containing the registered client meta data - * @return JSON string containing attributes of client that should be returned - */ - public abstract String getRegistrationResponse(Map clientMetaData); - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/RequiredParamsValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/RequiredParamsValidator.java deleted file mode 100644 index 2b37bc65..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/RequiredParamsValidator.java +++ /dev/null @@ -1,157 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation; - -import com.fasterxml.jackson.databind.ObjectMapper; -import com.wso2.openbanking.accelerator.identity.dcr.model.RegistrationRequest; -import com.wso2.openbanking.accelerator.identity.dcr.validation.annotation.ValidateRequiredParams; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.List; -import java.util.Locale; -import java.util.Map; - -import javax.validation.ConstraintValidator; -import javax.validation.ConstraintValidatorContext; - - -/** - * Validator class for validating the required parameters. - */ -public class RequiredParamsValidator implements ConstraintValidator { - - private static final Log log = LogFactory.getLog(RequiredParamsValidator.class); - - private static ObjectMapper objMapper = new ObjectMapper(); - - @Override - public boolean isValid(Object registrationRequestObject, ConstraintValidatorContext constraintValidatorContext) { - - RegistrationRequest registrationRequest = (RegistrationRequest) registrationRequestObject; - - Map requestParameterMap = objMapper.convertValue(registrationRequest, Map.class); - - Map> dcrConfigs = IdentityExtensionsDataHolder.getInstance() - .getDcrRegistrationConfigMap(); - - for (Map.Entry> paramConfig : dcrConfigs.entrySet()) { - //convert first letter to lowercase in DCR registration config parameters - String camelCaseConfigParam = convertFirstLetterToLowerCase(paramConfig.getKey()); - //check whether required parameters are available in the request as expected - if (DCRCommonConstants.DCR_REGISTRATION_PARAM_REQUIRED_TRUE - .equalsIgnoreCase((String) paramConfig.getValue() - .get(DCRCommonConstants.DCR_REGISTRATION_PARAM_REQUIRED))) { - if (requestParameterMap.get(camelCaseConfigParam) == null) { - constraintValidatorContext.disableDefaultConstraintViolation(); - constraintValidatorContext - .buildConstraintViolationWithTemplate("Required parameter " + camelCaseConfigParam + - " cannot be null:" + DCRCommonConstants.INVALID_META_DATA) - .addConstraintViolation(); - return false; - } - //validate list type required parameters - if (requestParameterMap.get(camelCaseConfigParam) instanceof List) { - List param = (List) requestParameterMap.get(camelCaseConfigParam); - if (param.isEmpty()) { - constraintValidatorContext.disableDefaultConstraintViolation(); - constraintValidatorContext - .buildConstraintViolationWithTemplate("Required parameter " + camelCaseConfigParam + - " cannot be empty:" + DCRCommonConstants.INVALID_META_DATA) - .addConstraintViolation(); - return false; - } - } - //validate string type required parameters - if (requestParameterMap.get(camelCaseConfigParam) instanceof String) { - String param = (String) requestParameterMap.get(camelCaseConfigParam); - if (StringUtils.isBlank(param)) { - constraintValidatorContext.disableDefaultConstraintViolation(); - constraintValidatorContext - .buildConstraintViolationWithTemplate("Required parameter " + camelCaseConfigParam + - " cannot be empty:" + DCRCommonConstants.INVALID_META_DATA) - .addConstraintViolation(); - return false; - } - } - } - //checks whether tag is set in config and is not empty. - if (paramConfig.getValue().get(DCRCommonConstants.DCR_REGISTRATION_PARAM_ALLOWED_VALUES) != null && - requestParameterMap.get(camelCaseConfigParam) != null) { - //checks whether allowed values configurations contain any empty values - if (!((List) paramConfig.getValue().get(DCRCommonConstants.DCR_REGISTRATION_PARAM_ALLOWED_VALUES)) - .contains("")) { - //validate against allowed values provided in config - List allowedList = (List) paramConfig.getValue() - .get(DCRCommonConstants.DCR_REGISTRATION_PARAM_ALLOWED_VALUES); - //validate array type parameters - if (requestParameterMap.get(camelCaseConfigParam) instanceof List) { - List params = (ArrayList) requestParameterMap.get(camelCaseConfigParam); - for (Object paramObject : params) { - if (paramObject instanceof String) { - String param = (String) paramObject; - if (!allowedList.contains(param)) { - constraintValidatorContext.disableDefaultConstraintViolation(); - constraintValidatorContext - .buildConstraintViolationWithTemplate("Invalid " + - camelCaseConfigParam + " provided:" + - DCRCommonConstants.INVALID_META_DATA).addConstraintViolation(); - return false; - } - } - - } - } - //validate string type parameters - if (requestParameterMap.get(camelCaseConfigParam) instanceof String) { - String param = (String) requestParameterMap.get(camelCaseConfigParam); - //check scope validation since request is sending a space separated scopes list - if (camelCaseConfigParam.equalsIgnoreCase(DCRCommonConstants.DCR_REGISTRATION_PARAM_SCOPE)) { - List scopeList = Arrays.asList(param.split(" ")); - for (String scope : scopeList) { - if (!allowedList.contains(scope)) { - constraintValidatorContext - .buildConstraintViolationWithTemplate("Invalid " + - camelCaseConfigParam + " provided:" + - DCRCommonConstants.INVALID_META_DATA).addConstraintViolation(); - return false; - } - } - } else if (!allowedList.contains(param)) { - constraintValidatorContext.disableDefaultConstraintViolation(); - constraintValidatorContext - .buildConstraintViolationWithTemplate("Invalid " + - camelCaseConfigParam + " provided:" + - DCRCommonConstants.INVALID_META_DATA).addConstraintViolation(); - return false; - } - } - } - } - } - return true; - } - - private String convertFirstLetterToLowerCase(String configParameterValue) { - return configParameterValue.substring(0, 1).toLowerCase(Locale.ENGLISH) + configParameterValue.substring(1); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/SignatureValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/SignatureValidator.java deleted file mode 100644 index 0c5dd3f0..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/SignatureValidator.java +++ /dev/null @@ -1,110 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation; - -import com.nimbusds.jose.JOSEException; -import com.nimbusds.jose.proc.BadJOSEException; -import com.nimbusds.jwt.SignedJWT; -import com.wso2.openbanking.accelerator.common.identity.IdentityConstants; -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.common.util.OpenBankingUtils; -import com.wso2.openbanking.accelerator.identity.dcr.validation.annotation.ValidateSignature; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import org.apache.commons.beanutils.BeanUtils; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.lang.reflect.InvocationTargetException; -import java.net.MalformedURLException; -import java.text.ParseException; - -import javax.validation.ConstraintValidator; -import javax.validation.ConstraintValidatorContext; - -/** - * Validator class for signature validation of SSA. - */ -public class SignatureValidator implements ConstraintValidator { - - private static final Log log = LogFactory.getLog(SignatureValidator.class); - - private String softwareStatementPath; - private String ssaBodyPath; - - @Override - public void initialize(ValidateSignature validateSignature) { - - this.softwareStatementPath = validateSignature.ssa(); - this.ssaBodyPath = validateSignature.ssaBody(); - } - - @Override - public boolean isValid(Object registrationRequest, - ConstraintValidatorContext constraintValidatorContext) { - - try { - String softwareStatement = BeanUtils.getProperty(registrationRequest, softwareStatementPath); - if (StringUtils.isEmpty(softwareStatement)) { - return true; - } - SignedJWT signedJWT = SignedJWT.parse(softwareStatement); - String jwtString = signedJWT.getParsedString(); - String alg = signedJWT.getHeader().getAlgorithm().getName(); - String softwareEnvironmentFromSSA = OpenBankingUtils.getSoftwareEnvironmentFromSSA(jwtString); - String jwksURL; - - if (IdentityConstants.PRODUCTION.equals(softwareEnvironmentFromSSA)) { - // validate the signature against production jwks - jwksURL = IdentityExtensionsDataHolder.getInstance().getConfigurationMap() - .get(DCRCommonConstants.DCR_JWKS_ENDPOINT_PRODUCTION).toString(); - if (log.isDebugEnabled()) { - log.debug(String.format("Validating the signature from Production JwksUrl %s", - jwksURL.replaceAll("[\r\n]", ""))); - } - } else { - // else validate the signature against sandbox jwks - jwksURL = IdentityExtensionsDataHolder.getInstance().getConfigurationMap() - .get(DCRCommonConstants.DCR_JWKS_ENDPOINT_SANDBOX).toString(); - if (log.isDebugEnabled()) { - log.debug(String.format("Validating the signature from Sandbox JwksUrl %s", - jwksURL.replaceAll("[\r\n]", ""))); - } - } - return isValidateJWTSignature(jwksURL, jwtString, alg); - } catch (IllegalAccessException | InvocationTargetException | NoSuchMethodException e) { - log.error("Error while resolving validation fields", e); - } catch (ParseException e) { - log.error("Error while parsing the JWT string", e); - } - return false; - } - - private boolean isValidateJWTSignature(String jwksURL, String jwtString, String alg) { - - try { - return JWTUtils.validateJWTSignature(jwtString, jwksURL, alg); - } catch (ParseException e) { - log.error("Error while parsing the JWT string", e); - } catch (JOSEException | BadJOSEException | MalformedURLException e) { - log.error("Error occurred while validating the signature", e); - } - return false; - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/annotation/ValidateAlgorithm.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/annotation/ValidateAlgorithm.java deleted file mode 100644 index 57b56d2d..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/annotation/ValidateAlgorithm.java +++ /dev/null @@ -1,53 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation.annotation; - -import com.wso2.openbanking.accelerator.identity.dcr.validation.AlgorithmValidator; - -import java.lang.annotation.Documented; -import java.lang.annotation.Retention; -import java.lang.annotation.Target; - -import javax.validation.Constraint; -import javax.validation.Payload; - -import static java.lang.annotation.ElementType.TYPE; -import static java.lang.annotation.RetentionPolicy.RUNTIME; - -/** - * Annotation class for validating algorithm. - */ -@Target(TYPE) -@Retention(RUNTIME) -@Documented -@Constraint(validatedBy = {AlgorithmValidator.class}) -public @interface ValidateAlgorithm { - - String message() default "Invalid algorithm provided"; - - Class[] groups() default {}; - - Class[] payload() default {}; - - String idTokenAlg() default "idTokenAlg"; - - String reqObjAlg() default "reqObjAlg"; - - String tokenAuthAlg() default "tokenAuthAlg"; - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/annotation/ValidateIssuer.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/annotation/ValidateIssuer.java deleted file mode 100644 index c3d0651e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/annotation/ValidateIssuer.java +++ /dev/null @@ -1,52 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation.annotation; - -import com.wso2.openbanking.accelerator.identity.dcr.validation.IssuerValidator; - -import java.lang.annotation.Documented; -import java.lang.annotation.Retention; -import java.lang.annotation.Target; - -import javax.validation.Constraint; -import javax.validation.Payload; - -import static java.lang.annotation.ElementType.TYPE; -import static java.lang.annotation.RetentionPolicy.RUNTIME; - -/** - * Annotation class for issuer validation. - */ -@Target(TYPE) -@Retention(RUNTIME) -@Documented -@Constraint(validatedBy = {IssuerValidator.class}) -public @interface ValidateIssuer { - - String message() default "Invalid issuer"; - - Class[] groups() default {}; - - Class[] payload() default {}; - - String issuerProperty() default "issuerProperty"; - - String ssa() default "ssa"; -} - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/annotation/ValidateRequiredParams.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/annotation/ValidateRequiredParams.java deleted file mode 100644 index d8f0faf2..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/annotation/ValidateRequiredParams.java +++ /dev/null @@ -1,46 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation.annotation; - -import com.wso2.openbanking.accelerator.identity.dcr.validation.RequiredParamsValidator; - -import java.lang.annotation.Documented; -import java.lang.annotation.Retention; -import java.lang.annotation.Target; - -import javax.validation.Constraint; -import javax.validation.Payload; - -import static java.lang.annotation.ElementType.TYPE; -import static java.lang.annotation.RetentionPolicy.RUNTIME; - -/** - * Annotation class for required parameters validation. - */ -@Target(TYPE) -@Retention(RUNTIME) -@Documented -@Constraint(validatedBy = {RequiredParamsValidator.class}) -public @interface ValidateRequiredParams { - - String message() default "Missing required parameters"; - - Class[] groups() default {}; - - Class[] payload() default {}; -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/annotation/ValidateSignature.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/annotation/ValidateSignature.java deleted file mode 100644 index d7696e97..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/annotation/ValidateSignature.java +++ /dev/null @@ -1,50 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation.annotation; - -import com.wso2.openbanking.accelerator.identity.dcr.validation.SignatureValidator; - -import java.lang.annotation.Documented; -import java.lang.annotation.ElementType; -import java.lang.annotation.Retention; -import java.lang.annotation.Target; - -import javax.validation.Constraint; -import javax.validation.Payload; - -import static java.lang.annotation.RetentionPolicy.RUNTIME; - -/** - * Annotation class for signature validation. - */ -@Target(ElementType.TYPE) -@Retention(RUNTIME) -@Documented -@Constraint(validatedBy = {SignatureValidator.class}) -public @interface ValidateSignature { - - String message() default "Invalid signature for the provided SSA"; - - Class[] groups() default {}; - - Class[] payload() default {}; - - String ssaBody() default "ssaBody"; - - String ssa() default "ssa"; -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/AttributeChecks.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/AttributeChecks.java deleted file mode 100644 index f65793bf..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/AttributeChecks.java +++ /dev/null @@ -1,29 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation.validationgroups; - -/** - * Interface for grouping the validation annotations. - * Groups the validations for attributes - */ -@Deprecated -public interface AttributeChecks { - -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/MandatoryChecks.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/MandatoryChecks.java deleted file mode 100644 index cb938dbb..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/MandatoryChecks.java +++ /dev/null @@ -1,29 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation.validationgroups; - -/** - * Interface for grouping the validation annotations. - * Grouping the mandatory check constraints - */ -@Deprecated -public interface MandatoryChecks { - -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/SignatureCheck.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/SignatureCheck.java deleted file mode 100644 index 4ce4bacb..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/SignatureCheck.java +++ /dev/null @@ -1,29 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation.validationgroups; - -/** - * Interface for grouping the validation annotations. - * Groups the validation for signature - */ -@Deprecated -public interface SignatureCheck { - -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/ValidationOrder.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/ValidationOrder.java deleted file mode 100644 index d42af484..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/ValidationOrder.java +++ /dev/null @@ -1,29 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation.validationgroups; - -import javax.validation.GroupSequence; - -/** - * Class to define the order of execution for the hibernate validation groups. - */ -@GroupSequence({MandatoryChecks.class, AttributeChecks.class, SignatureCheck.class}) -@Deprecated -public interface ValidationOrder { - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/ValidityChecks.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/ValidityChecks.java deleted file mode 100644 index 4a011cca..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationgroups/ValidityChecks.java +++ /dev/null @@ -1,29 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation.validationgroups; - -/** - * Interface for grouping the validation annotations. - * Groups the validations for the validity of a JWT - */ -@Deprecated -public interface ValidityChecks { - -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationorder/ValidationOrder.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationorder/ValidationOrder.java deleted file mode 100644 index 9a2355a4..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/validation/validationorder/ValidationOrder.java +++ /dev/null @@ -1,32 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation.validationorder; - -import com.wso2.openbanking.accelerator.identity.common.annotations.validationgroups.AttributeChecks; -import com.wso2.openbanking.accelerator.identity.common.annotations.validationgroups.MandatoryChecks; -import com.wso2.openbanking.accelerator.identity.common.annotations.validationgroups.SignatureCheck; - -import javax.validation.GroupSequence; - -/** - * Class to define the order of execution for the hibernate validation groups. - */ -@GroupSequence({MandatoryChecks.class, AttributeChecks.class, SignatureCheck.class}) -public interface ValidationOrder { - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dispute/resolution/DisputeResolutionFilter.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dispute/resolution/DisputeResolutionFilter.java deleted file mode 100644 index d8bbcda7..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/dispute/resolution/DisputeResolutionFilter.java +++ /dev/null @@ -1,148 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dispute.resolution; - -import com.fasterxml.jackson.databind.ObjectMapper; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.data.publisher.common.util.OBDataPublisherUtil; -import com.wso2.openbanking.accelerator.identity.token.wrapper.RequestWrapper; -import com.wso2.openbanking.accelerator.identity.token.wrapper.ResponseWrapper; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.io.IOException; -import java.time.Instant; -import java.util.Enumeration; -import java.util.HashMap; -import java.util.Map; -import java.util.StringJoiner; - -import javax.servlet.Filter; -import javax.servlet.FilterChain; -import javax.servlet.ServletException; -import javax.servlet.ServletRequest; -import javax.servlet.ServletResponse; -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; - -import static com.wso2.openbanking.accelerator.common.util.OpenBankingUtils.isPublishableDisputeData; -import static com.wso2.openbanking.accelerator.common.util.OpenBankingUtils.reduceStringLength; - -/** - * Dispute Resolution Filter. - */ -public class DisputeResolutionFilter implements Filter { - - private static final Log log = LogFactory.getLog(DisputeResolutionFilter.class); - - @Override - public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) - throws IOException, ServletException { - - //Checking Dispute Resolution Feature is Enabled - if (!OpenBankingConfigParser.getInstance().isDisputeResolutionEnabled()) { - chain.doFilter(request, response); - } else { - HttpServletRequest httpRequest = (HttpServletRequest) request; - HttpServletResponse httpResponse = (HttpServletResponse) response; - - // Create a custom response wrapper to capture the response output - ResponseWrapper responseWrapper = new ResponseWrapper(httpResponse); - - // Create a custom request wrapper to capture the request - RequestWrapper requestWrapper = new RequestWrapper(httpRequest); - - // Retrieve the captured request output - byte[] requestContent = requestWrapper.getCapturedRequest(); - - // Convert the request content to JSON - String jsonRequest = new String(requestContent, httpRequest.getCharacterEncoding()); - - //get headers from requestWrapper - Enumeration headersRequestWrapper = requestWrapper.getHeaderNames(); - - // Capture error request information - String httpMethod = httpRequest.getMethod(); - String resourceURL = httpRequest.getRequestURL().toString(); - Map requestParams = httpRequest.getParameterMap(); - - // Convert requestParams to JSON string representation - ObjectMapper objectMapper = new ObjectMapper(); - String requestParamsJson = objectMapper.writeValueAsString(requestParams); - - String requestBody = StringUtils.defaultIfEmpty(jsonRequest, requestParamsJson); - - Enumeration headersMap = headersRequestWrapper; - - //Convert the headerMap to a string - StringJoiner joiner = new StringJoiner(", "); - while (headersMap.hasMoreElements()) { - String element = headersMap.nextElement(); - joiner.add(element); - } - String headers = joiner.toString(); - - chain.doFilter(requestWrapper, responseWrapper); - - // Retrieve the captured response output - byte[] responseContent = responseWrapper.getData(); - - // Convert the response content to JSON - String jsonResponse = new String(responseContent, httpResponse.getCharacterEncoding()); - - Map disputeResolutionData = new HashMap<>(); - - // Capture the response information - int statusCode = httpResponse.getStatus(); - String responseBody = jsonResponse; - - long unixTimestamp = Instant.now().getEpochSecond(); - - //reduced Headers, Request and Response Body Lengths - requestBody = reduceStringLength(requestBody, - OpenBankingConfigParser.getInstance().getMaxRequestBodyLength()); - responseBody = reduceStringLength(responseBody, - OpenBankingConfigParser.getInstance().getMaxResponseBodyLength()); - headers = reduceStringLength(headers, - OpenBankingConfigParser.getInstance().getMaxHeaderLength()); - - // Add the captured data put into the disputeResolutionData Map - disputeResolutionData.put(OpenBankingConstants.REQUEST_BODY, requestBody); - disputeResolutionData.put(OpenBankingConstants.RESPONSE_BODY, responseBody); - disputeResolutionData.put(OpenBankingConstants.STATUS_CODE, statusCode); - disputeResolutionData.put(OpenBankingConstants.HTTP_METHOD, httpMethod); - disputeResolutionData.put(OpenBankingConstants.ELECTED_RESOURCE, resourceURL); - disputeResolutionData.put(OpenBankingConstants.TIMESTAMP, unixTimestamp); - disputeResolutionData.put(OpenBankingConstants.HEADERS, headers); - - //Checking configurations to publish Dispute Data - if (isPublishableDisputeData(statusCode)) { - OBDataPublisherUtil.publishData(OpenBankingConstants.DISPUTE_RESOLUTION_STREAM_NAME, - OpenBankingConstants.DISPUTE_RESOLUTION_STREAM_VERSION, disputeResolutionData); - } - - response.getOutputStream().write(responseWrapper.getData()); - } - - } - -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/grant/type/handlers/OBAuthorizationCodeGrantHandler.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/grant/type/handlers/OBAuthorizationCodeGrantHandler.java deleted file mode 100644 index 40a19ba5..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/grant/type/handlers/OBAuthorizationCodeGrantHandler.java +++ /dev/null @@ -1,83 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.grant.type.handlers; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AccessTokenRespDTO; -import org.wso2.carbon.identity.oauth2.token.OAuthTokenReqMessageContext; -import org.wso2.carbon.identity.oauth2.token.handlers.grant.AuthorizationCodeGrantHandler; - -/** - * OB specific authorization code grant handler. - */ -public class OBAuthorizationCodeGrantHandler extends AuthorizationCodeGrantHandler { - - @Override - public OAuth2AccessTokenRespDTO issue(OAuthTokenReqMessageContext tokReqMsgCtx) throws IdentityOAuth2Exception { - - try { - if (IdentityCommonUtil.getRegulatoryFromSPMetaData(tokReqMsgCtx.getOauth2AccessTokenReqDTO() - .getClientId())) { - OAuth2AccessTokenRespDTO oAuth2AccessTokenRespDTO = super.issue(tokReqMsgCtx); - executeInitialStep(oAuth2AccessTokenRespDTO, tokReqMsgCtx); - tokReqMsgCtx.setScope(IdentityCommonUtil.removeInternalScopes(tokReqMsgCtx.getScope())); - publishUserAccessTokenData(oAuth2AccessTokenRespDTO); - return oAuth2AccessTokenRespDTO; - } - } catch (OpenBankingException e) { - throw new IdentityOAuth2Exception(e.getMessage()); - } - return super.issue(tokReqMsgCtx); - } - - /** - * Extend this method to publish access token related data. - * - * @param oAuth2AccessTokenRespDTO - */ - - public void publishUserAccessTokenData(OAuth2AccessTokenRespDTO oAuth2AccessTokenRespDTO) - throws IdentityOAuth2Exception { - - } - - /** - * Extend this method to perform any actions which requires internal scopes. - * - * @param oAuth2AccessTokenRespDTO - * @param tokReqMsgCtx - */ - public void executeInitialStep(OAuth2AccessTokenRespDTO oAuth2AccessTokenRespDTO, - OAuthTokenReqMessageContext tokReqMsgCtx) throws IdentityOAuth2Exception { - - } - - /** - * Extend this method to perform any actions related when issuing refresh token. - * - * @return - */ - @Override - public boolean issueRefreshToken() throws IdentityOAuth2Exception { - - return super.issueRefreshToken(); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/grant/type/handlers/OBClientCredentialsGrantHandler.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/grant/type/handlers/OBClientCredentialsGrantHandler.java deleted file mode 100644 index ba66983c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/grant/type/handlers/OBClientCredentialsGrantHandler.java +++ /dev/null @@ -1,85 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.grant.type.handlers; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.wso2.carbon.identity.oauth.common.exception.InvalidOAuthClientException; -import org.wso2.carbon.identity.oauth.dao.OAuthAppDO; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AccessTokenRespDTO; -import org.wso2.carbon.identity.oauth2.token.OAuthTokenReqMessageContext; -import org.wso2.carbon.identity.oauth2.token.handlers.grant.ClientCredentialsGrantHandler; -import org.wso2.carbon.identity.oauth2.util.OAuth2Util; - -import java.util.Arrays; - -/** - * OB specific client credentials code grant handler. - */ -public class OBClientCredentialsGrantHandler extends ClientCredentialsGrantHandler { - - @Override - public OAuth2AccessTokenRespDTO issue(OAuthTokenReqMessageContext tokReqMsgCtx) throws IdentityOAuth2Exception { - - try { - if (IdentityCommonUtil.getRegulatoryFromSPMetaData(tokReqMsgCtx.getOauth2AccessTokenReqDTO() - .getClientId())) { - if (IdentityCommonUtil.getDCRModifyResponseConfig() && tokReqMsgCtx.getScope().length > 0 && - Arrays.asList(tokReqMsgCtx.getScope()).contains(IdentityCommonUtil.getDCRScope())) { - long validityPeriod = 999999999; - OAuthAppDO oAuthAppDO = OAuth2Util - .getAppInformationByClientId(tokReqMsgCtx.getOauth2AccessTokenReqDTO().getClientId()); - oAuthAppDO.setApplicationAccessTokenExpiryTime(validityPeriod); - tokReqMsgCtx.setValidityPeriod(validityPeriod); - } - OAuth2AccessTokenRespDTO oAuth2AccessTokenRespDTO = super.issue(tokReqMsgCtx); - executeInitialStep(oAuth2AccessTokenRespDTO, tokReqMsgCtx); - tokReqMsgCtx.setScope(IdentityCommonUtil.removeInternalScopes(tokReqMsgCtx.getScope())); - publishUserAccessTokenData(oAuth2AccessTokenRespDTO); - return oAuth2AccessTokenRespDTO; - } - } catch (OpenBankingException | InvalidOAuthClientException e) { - throw new IdentityOAuth2Exception(e.getMessage()); - } - return super.issue(tokReqMsgCtx); - } - - /** - * Extend this method to publish access token related data. - * - * @param oAuth2AccessTokenRespDTO - */ - - public void publishUserAccessTokenData(OAuth2AccessTokenRespDTO oAuth2AccessTokenRespDTO) - throws IdentityOAuth2Exception { - - } - - /** - * Extend this method to perform any actions which requires internal scopes. - * - * @param oAuth2AccessTokenRespDTO - * @param tokReqMsgCtx - */ - public void executeInitialStep(OAuth2AccessTokenRespDTO oAuth2AccessTokenRespDTO, - OAuthTokenReqMessageContext tokReqMsgCtx) throws IdentityOAuth2Exception { - - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/grant/type/handlers/OBPasswordGrantHandler.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/grant/type/handlers/OBPasswordGrantHandler.java deleted file mode 100644 index cbb978ea..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/grant/type/handlers/OBPasswordGrantHandler.java +++ /dev/null @@ -1,72 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.grant.type.handlers; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AccessTokenRespDTO; -import org.wso2.carbon.identity.oauth2.token.OAuthTokenReqMessageContext; -import org.wso2.carbon.identity.oauth2.token.handlers.grant.PasswordGrantHandler; - -/** - * OB specific password grant handler. - */ -public class OBPasswordGrantHandler extends PasswordGrantHandler { - - @Override - public OAuth2AccessTokenRespDTO issue(OAuthTokenReqMessageContext tokReqMsgCtx) throws IdentityOAuth2Exception { - - try { - if (IdentityCommonUtil.getRegulatoryFromSPMetaData(tokReqMsgCtx.getOauth2AccessTokenReqDTO() - .getClientId())) { - OAuth2AccessTokenRespDTO oAuth2AccessTokenRespDTO = super.issue(tokReqMsgCtx); - executeInitialStep(oAuth2AccessTokenRespDTO, tokReqMsgCtx); - tokReqMsgCtx.setScope(IdentityCommonUtil.removeInternalScopes(tokReqMsgCtx.getScope())); - publishUserAccessTokenData(oAuth2AccessTokenRespDTO); - return oAuth2AccessTokenRespDTO; - } - } catch (OpenBankingException e) { - throw new IdentityOAuth2Exception(e.getMessage()); - } - return super.issue(tokReqMsgCtx); - } - - /** - * Extend this method to publish access token related data. - * - * @param oAuth2AccessTokenRespDTO - */ - - public void publishUserAccessTokenData(OAuth2AccessTokenRespDTO oAuth2AccessTokenRespDTO) - throws IdentityOAuth2Exception { - - } - - /** - * Extend this method to perform any actions which requires internal scopes. - * - * @param oAuth2AccessTokenRespDTO - * @param tokReqMsgCtx - */ - public void executeInitialStep(OAuth2AccessTokenRespDTO oAuth2AccessTokenRespDTO, - OAuthTokenReqMessageContext tokReqMsgCtx) throws IdentityOAuth2Exception { - - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/grant/type/handlers/OBRefreshGrantHandler.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/grant/type/handlers/OBRefreshGrantHandler.java deleted file mode 100644 index 8f9c458e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/grant/type/handlers/OBRefreshGrantHandler.java +++ /dev/null @@ -1,120 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.grant.type.handlers; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.lang3.ArrayUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AccessTokenRespDTO; -import org.wso2.carbon.identity.oauth2.token.OAuthTokenReqMessageContext; -import org.wso2.carbon.identity.oauth2.token.handlers.grant.RefreshGrantHandler; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashSet; - -/** - * OB specific refresh grant handler. - */ -public class OBRefreshGrantHandler extends RefreshGrantHandler { - - private static final Log log = LogFactory.getLog(OBRefreshGrantHandler.class); - - @Override - public OAuth2AccessTokenRespDTO issue(OAuthTokenReqMessageContext tokReqMsgCtx) throws IdentityOAuth2Exception { - - try { - if (IdentityCommonUtil.getRegulatoryFromSPMetaData(tokReqMsgCtx.getOauth2AccessTokenReqDTO() - .getClientId())) { - OAuth2AccessTokenRespDTO oAuth2AccessTokenRespDTO = super.issue(tokReqMsgCtx); - executeInitialStep(oAuth2AccessTokenRespDTO, tokReqMsgCtx); - tokReqMsgCtx.setScope(IdentityCommonUtil.removeInternalScopes(tokReqMsgCtx.getScope())); - publishUserAccessTokenData(oAuth2AccessTokenRespDTO); - if (tokReqMsgCtx.getScope().length == 0) { - oAuth2AccessTokenRespDTO.setAuthorizedScopes(""); - } - return oAuth2AccessTokenRespDTO; - } - } catch (OpenBankingException e) { - throw new IdentityOAuth2Exception(e.getMessage()); - } - return super.issue(tokReqMsgCtx); - } - - /** - * Extend this method to publish access token related data. - * - * @param oAuth2AccessTokenRespDTO - */ - - public void publishUserAccessTokenData(OAuth2AccessTokenRespDTO oAuth2AccessTokenRespDTO) - throws IdentityOAuth2Exception { - - } - - /** - * Extend this method to perform any actions which requires internal scopes. - * - * @param oAuth2AccessTokenRespDTO - * @param tokReqMsgCtx - */ - public void executeInitialStep(OAuth2AccessTokenRespDTO oAuth2AccessTokenRespDTO, - OAuthTokenReqMessageContext tokReqMsgCtx) throws IdentityOAuth2Exception { - - } - - @Override - public boolean validateScope(OAuthTokenReqMessageContext tokReqMsgCtx) throws IdentityOAuth2Exception { - - String[] grantedScopes = tokReqMsgCtx.getScope(); - if (!super.validateScope(tokReqMsgCtx)) { - return false; - } - - String[] requestedScopes = tokReqMsgCtx.getOauth2AccessTokenReqDTO().getScope(); - if (ArrayUtils.isNotEmpty(requestedScopes)) { - //Adding internal scopes. - ArrayList requestedScopeList = new ArrayList<>(Arrays.asList(requestedScopes)); - String consentIdClaim = IdentityExtensionsDataHolder.getInstance().getConfigurationMap() - .get(IdentityCommonConstants.CONSENT_ID_CLAIM_NAME).toString(); - for (String scope : grantedScopes) { - if (scope.startsWith(consentIdClaim)) { - if (log.isDebugEnabled()) { - log.debug(String.format("Adding custom scope %s to the requested scopes", scope)); - } - requestedScopeList.add(scope); - } - } - - // remove duplicates in requestedScopeList - requestedScopeList = new ArrayList<>(new HashSet<>(requestedScopeList)); - - String[] modifiedScopes = requestedScopeList.toArray(new String[0]); - if (modifiedScopes.length != 0) { - tokReqMsgCtx.setScope(modifiedScopes); - } - } - return true; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/idtoken/OBIDTokenBuilder.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/idtoken/OBIDTokenBuilder.java deleted file mode 100644 index 11a97f30..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/idtoken/OBIDTokenBuilder.java +++ /dev/null @@ -1,293 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.idtoken; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonHelper; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedUser; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.authz.OAuthAuthzReqMessageContext; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AccessTokenRespDTO; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AuthorizeRespDTO; -import org.wso2.carbon.identity.oauth2.token.OAuthTokenReqMessageContext; -import org.wso2.carbon.identity.openidconnect.DefaultIDTokenBuilder; -import org.wso2.carbon.utils.multitenancy.MultitenantUtils; - -import java.net.URI; -import java.net.URISyntaxException; -import java.nio.charset.StandardCharsets; -import java.util.Arrays; -import java.util.Collections; -import java.util.List; -import java.util.Map; -import java.util.UUID; -import java.util.regex.Matcher; -import java.util.regex.Pattern; -import java.util.stream.Collectors; - -/** - * OB specific IDToken builder. - */ -public class OBIDTokenBuilder extends DefaultIDTokenBuilder { - - private static final Log log = LogFactory.getLog(OBIDTokenBuilder.class); - - public OBIDTokenBuilder() throws IdentityOAuth2Exception { - - } - - Map identityConfigurations = IdentityExtensionsDataHolder.getInstance().getConfigurationMap(); - Object ppidProperty = identityConfigurations.get(IdentityCommonConstants.ENABLE_SUBJECT_AS_PPID); - Object removeTenantDomainConfig = identityConfigurations. - get(IdentityCommonConstants.REMOVE_TENANT_DOMAIN_FROM_SUBJECT); - Boolean removeTenantDomain = removeTenantDomainConfig != null - && Boolean.parseBoolean(removeTenantDomainConfig.toString()); - Object removeUserStoreDomainConfig = identityConfigurations. - get(IdentityCommonConstants.REMOVE_USER_STORE_DOMAIN_FROM_SUBJECT); - Boolean removeUserStoreDomain = removeUserStoreDomainConfig != null - && Boolean.parseBoolean(removeUserStoreDomainConfig.toString()); - - // method to set the subject claim in id token returned in authorization as a pairwise pseudonymous ID - @Override - protected String getSubjectClaim(OAuthAuthzReqMessageContext authzReqMessageContext, - OAuth2AuthorizeRespDTO authorizeRespDTO, - String clientId, - String spTenantDomain, - AuthenticatedUser authorizedUser) throws IdentityOAuth2Exception { - - String callBackUri = authzReqMessageContext.getAuthorizationReqDTO().getCallbackUrl(); - String userId = StringUtils.EMPTY; - String subject = StringUtils.EMPTY; - String sectorIdentifierUri = null; - boolean setSubjectAsPPID = false; - if (ppidProperty != null) { - setSubjectAsPPID = Boolean.parseBoolean(ppidProperty.toString()); - } - try { - // for non regulatory scenarios, need to return the user id as the subject - if (!IdentityCommonUtil.getRegulatoryFromSPMetaData(clientId)) { - return super.getSubjectClaim(authzReqMessageContext, authorizeRespDTO, clientId, spTenantDomain, - authorizedUser); - } - sectorIdentifierUri = getSectorIdentifierUri(clientId); - } catch (OpenBankingException e) { - log.error("Error occurred while retrieving service provider data", e); - throw new IdentityOAuth2Exception("Error occurred while retrieving service provider data"); - } - if (setSubjectAsPPID) { - if (authzReqMessageContext.getAuthorizationReqDTO().getUser() != null) { - userId = authzReqMessageContext.getAuthorizationReqDTO().getUser() - .getUsernameAsSubjectIdentifier(false, false); - } - subject = getSubjectClaimValue(sectorIdentifierUri, userId, callBackUri); - if (StringUtils.isNotBlank(subject)) { - return subject; - } else { - log.error("Subject claim cannot be empty"); - throw new IdentityOAuth2Exception("Subject claim cannot be empty"); - } - } else if (removeTenantDomain || removeUserStoreDomain) { - /* Update the subject claim of the JWT claims set if any of the following configurations are true - and if PPID is as the subject claim is not enabled. - 1. open_banking.identity.token.remove_user_store_domain_from_subject - 2. open_banking.identity.token.remove_tenant_domain_from_subject */ - return authorizedUser.getUsernameAsSubjectIdentifier(!removeUserStoreDomain, !removeTenantDomain); - } else { - return MultitenantUtils.getTenantAwareUsername(super.getSubjectClaim(authzReqMessageContext, - authorizeRespDTO, clientId, spTenantDomain, authorizedUser)); - } - } - - // method to set the subject claim in Id token returned in token response - @Override - protected String getSubjectClaim(OAuthTokenReqMessageContext tokenReqMessageContext, - OAuth2AccessTokenRespDTO tokenRespDTO, - String clientId, - String spTenantDomain, - AuthenticatedUser authorizedUser) throws IdentityOAuth2Exception { - - String callBackUri = tokenRespDTO.getCallbackURI(); - String userId = StringUtils.EMPTY; - String subject = StringUtils.EMPTY; - String sectorIdentifierUri = null; - boolean setSubjectAsPPID = false; - if (ppidProperty != null) { - setSubjectAsPPID = Boolean.parseBoolean(ppidProperty.toString()); - } - try { - // for non regulatory scenarios, need to return the user id as the subject - if (!IdentityCommonUtil.getRegulatoryFromSPMetaData(clientId)) { - return super.getSubjectClaim(tokenReqMessageContext, tokenRespDTO, clientId, spTenantDomain, - authorizedUser); - } - sectorIdentifierUri = getSectorIdentifierUri(clientId); - } catch (OpenBankingException e) { - log.error("Error occurred while retrieving service provider data", e); - throw new IdentityOAuth2Exception("Error occurred while retrieving service provider data"); - } - if (setSubjectAsPPID) { - if (tokenReqMessageContext.getAuthorizedUser() != null) { - userId = tokenReqMessageContext.getAuthorizedUser() - .getUsernameAsSubjectIdentifier(false, false); - } - subject = getSubjectClaimValue(sectorIdentifierUri, userId, callBackUri); - if (StringUtils.isNotBlank(subject)) { - return subject; - } else { - log.error("Subject claim cannot be empty"); - throw new IdentityOAuth2Exception("Subject claim cannot be empty"); - } - } else if (removeTenantDomain || removeUserStoreDomain) { - /* Update the subject claim of the JWT claims set if any of the following configurations are true - and if PPID is as the subject claim is not enabled. - 1. open_banking.identity.token.remove_user_store_domain_from_subject - 2. open_banking.identity.token.remove_tenant_domain_from_subject */ - return authorizedUser.getUsernameAsSubjectIdentifier(!removeUserStoreDomain, !removeTenantDomain); - } else { - return MultitenantUtils.getTenantAwareUsername(super.getSubjectClaim(tokenReqMessageContext, - tokenRespDTO, clientId, spTenantDomain, authorizedUser)); - } - } - - /** - * Get the subject claim as a UUID with userId and call back uri host name as seed. - * - * @param callBackUri redirect uri of the data recipient - * @param userID user identification of the consumer - * @return - */ - private String getSubjectFromCallBackUris(String callBackUri, String userID) { - - List uris = unwrapURIString(callBackUri); - - if (!uris.isEmpty()) { - URI uri; - try { - // assuming all URIs have the same hostname, we just take the first URI - uri = new URI(uris.get(0)); - } catch (URISyntaxException e) { - log.error("Error while retrieving the host name of the redirect url ", e); - return StringUtils.EMPTY; - } - String hostname = uri.getHost(); - String seed = hostname.concat(userID); - return UUID.nameUUIDFromBytes(seed.getBytes(StandardCharsets.UTF_8)).toString(); - } - log.error("Redirect URIs cannot be empty"); - return StringUtils.EMPTY; - } - - /** - * Get the subject claim as a UUID with userId and sector identifier uri host name as seed. - * - * @param sectorIdentifierUri sector identifier uri of the data recipient - * @param userID user identification of the consumer - * @return - */ - private String getSubjectFromSectorIdentifierUri(String sectorIdentifierUri, String userID) { - - URI uri; - try { - // assuming all URIs have the same hostname, we just take the first URI - uri = new URI(sectorIdentifierUri); - } catch (URISyntaxException e) { - log.error("Error while retrieving the host name of the redirect url ", e); - return StringUtils.EMPTY; - } - String hostname = uri.getHost(); - String seed = hostname.concat(userID); - return UUID.nameUUIDFromBytes(seed.getBytes(StandardCharsets.UTF_8)).toString(); - } - - /** - * Given a string in regex format, unwrap and create list of seperate URIs. - * - * @param uriString The joined URIs in string format. - * @return List of URIs. - */ - private List unwrapURIString(String uriString) { - - Pattern outerPattern = Pattern.compile("regexp=\\((.*?)\\)"); - Pattern innerPattern = Pattern.compile("\\^(.*?)\\$"); - - Matcher matcher = outerPattern.matcher(uriString); - - String delimitedUri; - - if (matcher.find()) { - // remove regex= part - delimitedUri = matcher.group(1); - } else { - // URI is not having regex format - return Collections.singletonList(uriString); - } - - String[] uris = delimitedUri.split("\\|"); - - // remove ^..$ part - return Arrays.stream(uris) - .map(uri -> { - Matcher m = innerPattern.matcher(uri); - if (m.find()) { - return m.group(1); - } else { - return uri; - } - }).collect(Collectors.toList()); - } - - /** - * Get the sector identifier uri from sp metadata. - * - * @param clientId consumer id - * @return - */ - @Generated(message = "Excluding from code coverage since it requires a service call") - protected String getSectorIdentifierUri(String clientId) throws OpenBankingException { - - return (new IdentityCommonHelper()).getAppPropertyFromSPMetaData(clientId, "sector_identifier_uri"); - } - - /** - * Validate and get subject value. - * - * @param sectorIdentifierUri sector identifier uri - * @param userId user id - * @param callBackUri call back uris - * @return subject value if validated or return empty string - */ - private String getSubjectClaimValue(String sectorIdentifierUri, String userId, String callBackUri) { - - if (StringUtils.isNotBlank(sectorIdentifierUri) && StringUtils.isNotBlank(userId)) { - log.debug("Calculating subject claim using sector identifier uri "); - return getSubjectFromSectorIdentifierUri(sectorIdentifierUri, userId); - } else if (StringUtils.isNotBlank(callBackUri) && StringUtils.isNotBlank(userId)) { - log.debug("Calculating subject claim using redirect uris "); - return getSubjectFromCallBackUris(callBackUri, userId); - } - return StringUtils.EMPTY; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/interceptor/OBDefaultIntrospectionDataProvider.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/interceptor/OBDefaultIntrospectionDataProvider.java deleted file mode 100644 index b936ff6c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/interceptor/OBDefaultIntrospectionDataProvider.java +++ /dev/null @@ -1,45 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.interceptor; - -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.dto.OAuth2IntrospectionResponseDTO; -import org.wso2.carbon.identity.oauth2.dto.OAuth2TokenValidationRequestDTO; - -import java.util.HashMap; -import java.util.Map; - -/** - * OB specific default introspection data provider implementation. - */ -public class OBDefaultIntrospectionDataProvider extends OBIntrospectionDataProvider { - - - @Override - public Map getIntrospectionData(OAuth2TokenValidationRequestDTO oAuth2TokenValidationRequestDTO, - OAuth2IntrospectionResponseDTO oAuth2IntrospectionResponseDTO) - throws IdentityOAuth2Exception { - - if (oAuth2IntrospectionResponseDTO.isActive()) { - return oAuth2IntrospectionResponseDTO.getProperties(); - } else { - return new HashMap<>(); - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/interceptor/OBIntrospectionDataProvider.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/interceptor/OBIntrospectionDataProvider.java deleted file mode 100644 index 5429a6ec..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/interceptor/OBIntrospectionDataProvider.java +++ /dev/null @@ -1,65 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.interceptor; - -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.lang.StringUtils; -import org.wso2.carbon.identity.core.handler.AbstractIdentityHandler; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.IntrospectionDataProvider; -import org.wso2.carbon.identity.oauth2.dto.OAuth2IntrospectionResponseDTO; -import org.wso2.carbon.identity.oauth2.dto.OAuth2TokenValidationRequestDTO; - -import java.util.Map; - -/** - * OB specific introspection data provider. - */ -public class OBIntrospectionDataProvider extends AbstractIdentityHandler implements IntrospectionDataProvider { - - private static IntrospectionDataProvider introspectionDataProvider; - - @Override - public Map getIntrospectionData(OAuth2TokenValidationRequestDTO oAuth2TokenValidationRequestDTO, - OAuth2IntrospectionResponseDTO oAuth2IntrospectionResponseDTO) - throws IdentityOAuth2Exception { - - Map additionalDataMap = getIntrospectionDataProvider() - .getIntrospectionData(oAuth2TokenValidationRequestDTO, oAuth2IntrospectionResponseDTO); - String[] nonInternalScopes = IdentityCommonUtil.removeInternalScopes(oAuth2IntrospectionResponseDTO.getScope() - .split(IdentityCommonConstants.SPACE_SEPARATOR)); - oAuth2IntrospectionResponseDTO.setScope(StringUtils.join(nonInternalScopes, - IdentityCommonConstants.SPACE_SEPARATOR)); - additionalDataMap.put(IdentityCommonConstants.SCOPE, StringUtils.join(nonInternalScopes, - IdentityCommonConstants.SPACE_SEPARATOR)); - oAuth2IntrospectionResponseDTO.setProperties(additionalDataMap); - return additionalDataMap; - } - - public static IntrospectionDataProvider getIntrospectionDataProvider() { - - return introspectionDataProvider; - } - - public static void setIntrospectionDataProvider(IntrospectionDataProvider introspectionDataProvider) { - - OBIntrospectionDataProvider.introspectionDataProvider = introspectionDataProvider; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/internal/IdentityExtensionsDataHolder.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/internal/IdentityExtensionsDataHolder.java deleted file mode 100644 index c7a14c02..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/internal/IdentityExtensionsDataHolder.java +++ /dev/null @@ -1,459 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.internal; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.common.util.OpenBankingUtils; -import com.wso2.openbanking.accelerator.consent.mgt.service.ConsentCoreService; -import com.wso2.openbanking.accelerator.identity.auth.extensions.adaptive.function.OpenBankingAuthenticationWorker; -import com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator.OBRequestObjectValidator; -import com.wso2.openbanking.accelerator.identity.auth.extensions.response.handler.OBResponseTypeHandler; -import com.wso2.openbanking.accelerator.identity.claims.OBClaimProvider; -import com.wso2.openbanking.accelerator.identity.common.IdentityServiceExporter; -import com.wso2.openbanking.accelerator.identity.dcr.validation.DCRCommonConstants; -import com.wso2.openbanking.accelerator.identity.dcr.validation.RegistrationValidator; -import com.wso2.openbanking.accelerator.identity.interceptor.OBIntrospectionDataProvider; -import com.wso2.openbanking.accelerator.identity.listener.application.AbstractApplicationUpdater; -import com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.PushAuthRequestValidator; -import com.wso2.openbanking.accelerator.identity.token.DefaultTokenFilter; -import com.wso2.openbanking.accelerator.identity.token.TokenFilter; -import com.wso2.openbanking.accelerator.identity.token.validators.OBIdentityFilterValidator; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.throttler.service.OBThrottleService; -import org.wso2.carbon.identity.application.authentication.framework.JsFunctionRegistry; -import org.wso2.carbon.identity.application.mgt.ApplicationManagementService; -import org.wso2.carbon.identity.oauth.OAuthAdminServiceImpl; -import org.wso2.carbon.identity.oauth2.IntrospectionDataProvider; -import org.wso2.carbon.identity.oauth2.OAuth2Service; -import org.wso2.carbon.identity.oauth2.client.authentication.OAuthClientAuthnService; -import org.wso2.carbon.identity.openidconnect.ClaimProvider; -import org.wso2.carbon.identity.openidconnect.RequestObjectService; -import org.wso2.carbon.user.core.service.RealmService; - -import java.security.KeyStore; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import static com.wso2.openbanking.accelerator.common.util.OpenBankingUtils.getClassInstanceFromFQN; -import static com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants.PUSH_AUTH_REQUEST_VALIDATOR; -import static com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants.REQUEST_VALIDATOR; -import static com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants.RESPONSE_HANDLER; - -/** - * Data Holder for Open Banking Common. - */ -public class IdentityExtensionsDataHolder { - - private static volatile IdentityExtensionsDataHolder instance = new IdentityExtensionsDataHolder(); - private ApplicationManagementService applicationManagementService; - private RequestObjectService requestObjectService; - private OAuthAdminServiceImpl oAuthAdminService; - private OpenBankingConfigurationService openBankingConfigurationService; - private Map configurationMap; - private Map> dcrRegistrationConfigMap; - private List tokenValidators = new ArrayList<>(); - private DefaultTokenFilter defaultTokenFilter; - private RegistrationValidator registrationValidator; - private ClaimProvider claimProvider; - private IntrospectionDataProvider introspectionDataProvider; - private OBRequestObjectValidator obRequestObjectValidator; - private PushAuthRequestValidator pushAuthRequestValidator; - private KeyStore trustStore = null; - private OBResponseTypeHandler obResponseTypeHandler; - private AbstractApplicationUpdater abstractApplicationUpdater; - private int identityCacheAccessExpiry; - private int identityCacheModifiedExpiry; - private RealmService realmService; - private OBThrottleService obThrottleService; - private ConsentCoreService consentCoreService; - private OAuthClientAuthnService oAuthClientAuthnService; - private OAuth2Service oAuth2Service; - private JsFunctionRegistry jsFunctionRegistry; - - private Map workers = new HashMap<>(); - - private IdentityExtensionsDataHolder() { - - } - - public static IdentityExtensionsDataHolder getInstance() { - - if (instance == null) { - synchronized (IdentityExtensionsDataHolder.class) { - if (instance == null) { - instance = new IdentityExtensionsDataHolder(); - } - } - } - return instance; - } - - /** - * To get the the instance of {@link ApplicationManagementService}. - * - * @return applicationManagementService - */ - public ApplicationManagementService getApplicationManagementService() { - - return applicationManagementService; - } - - /** - * To set the RequestObjectService. - * - * @param requestObjectService instance of {@link RequestObjectService} - */ - public void setRequestObjectService(RequestObjectService requestObjectService) { - - this.requestObjectService = requestObjectService; - } - - public RequestObjectService getRequestObjectService() { - - return requestObjectService; - } - - - /** - * To set the ApplicationManagementService. - * - * @param applicationManagementService instance of {@link ApplicationManagementService} - */ - public void setApplicationManagementService(ApplicationManagementService applicationManagementService) { - - this.applicationManagementService = applicationManagementService; - } - - /** - * To get the the instance of {@link OAuthAdminServiceImpl}. - * - * @return oauthAdminService - */ - public OAuthAdminServiceImpl getOauthAdminService() { - - return oAuthAdminService; - } - - /** - * To set the OauthAdminService. - * - * @param oauthAdminService instance of {@link OAuthAdminServiceImpl} - */ - public void setOauthAdminService(OAuthAdminServiceImpl oauthAdminService) { - - this.oAuthAdminService = oauthAdminService; - } - - public OpenBankingConfigurationService getOpenBankingConfigurationService() { - - return openBankingConfigurationService; - } - - public void setConfigurationMap(Map confMap) { - - configurationMap = confMap; - } - - public Map getConfigurationMap() { - - return configurationMap; - } - - public void setOpenBankingConfigurationService( - OpenBankingConfigurationService openBankingConfigurationService) { - - this.openBankingConfigurationService = openBankingConfigurationService; - this.configurationMap = openBankingConfigurationService.getConfigurations(); - this.dcrRegistrationConfigMap = openBankingConfigurationService.getDCRRegistrationConfigurations(); - this.setTokenFilterValidators(); - TokenFilter.setValidators(getTokenFilterValidators()); - this.setDefaultTokenFilterImpl(); - TokenFilter.setDefaultTokenFilter(getDefaultTokenFilterImpl()); - RegistrationValidator dcrValidator = - (RegistrationValidator) OpenBankingUtils.getClassInstanceFromFQN(openBankingConfigurationService - .getConfigurations().get(DCRCommonConstants.DCR_VALIDATOR).toString()); - this.setRegistrationValidator(dcrValidator); - RegistrationValidator.setRegistrationValidator(dcrValidator); - obRequestObjectValidator = (OBRequestObjectValidator) - getClassInstanceFromFQN(IdentityExtensionsDataHolder.getInstance() - .getConfigurationMap().get(REQUEST_VALIDATOR).toString()); - PushAuthRequestValidator pushAuthRequestValidatorImpl = (PushAuthRequestValidator) - getClassInstanceFromFQN(IdentityExtensionsDataHolder.getInstance() - .getConfigurationMap().get(PUSH_AUTH_REQUEST_VALIDATOR).toString()); - this.setPushAuthRequestValidator(pushAuthRequestValidatorImpl); - PushAuthRequestValidator.setRegistrationValidator(pushAuthRequestValidatorImpl); - obResponseTypeHandler = (OBResponseTypeHandler) getClassInstanceFromFQN(openBankingConfigurationService - .getConfigurations().get(RESPONSE_HANDLER).toString()); - abstractApplicationUpdater = (AbstractApplicationUpdater) OpenBankingUtils.getClassInstanceFromFQN - (openBankingConfigurationService.getConfigurations().get(DCRCommonConstants.POST_APPLICATION_LISTENER) - .toString()); - this.setClaimProvider((ClaimProvider) OpenBankingUtils.getClassInstanceFromFQN(openBankingConfigurationService - .getConfigurations().get(IdentityCommonConstants.CLAIM_PROVIDER).toString())); - OBClaimProvider.setClaimProvider(getClaimProvider()); - this.setIntrospectionDataProvider((IntrospectionDataProvider) OpenBankingUtils - .getClassInstanceFromFQN(openBankingConfigurationService.getConfigurations() - .get(IdentityCommonConstants.INTROSPECTION_DATA_PROVIDER).toString())); - OBIntrospectionDataProvider.setIntrospectionDataProvider(getIntrospectionDataProvider()); - setIdentityCacheAccessExpiry((String) openBankingConfigurationService - .getConfigurations().get("IdentityCache.CacheAccessExpiry")); - setIdentityCacheModifiedExpiry((String) openBankingConfigurationService - .getConfigurations().get("IdentityCache.CacheModifiedExpiry")); - - Map authenticationWorkers = openBankingConfigurationService.getAuthenticationWorkers(); - authenticationWorkers.forEach((key, value) -> - addWorker((OpenBankingAuthenticationWorker) OpenBankingUtils.getClassInstanceFromFQN(value), key)); - } - - public List getTokenFilterValidators() { - - return tokenValidators; - - } - - public void setTokenFilterValidators() { - - for (Object element : extractTokenFilterValidators()) { - tokenValidators - .add((OBIdentityFilterValidator) OpenBankingUtils.getClassInstanceFromFQN(element.toString())); - } - } - - private List extractTokenFilterValidators() { - - Object validators = configurationMap.get(IdentityCommonConstants.TOKEN_VALIDATORS); - - if (validators != null) { - if (validators instanceof List) { - return (List) configurationMap.get(IdentityCommonConstants.TOKEN_VALIDATORS); - } else { - return Arrays.asList(validators); - } - } else { - return new ArrayList<>(); - } - } - - public DefaultTokenFilter getDefaultTokenFilterImpl() { - - return defaultTokenFilter; - - } - - public void setDefaultTokenFilterImpl() { - - defaultTokenFilter = - (DefaultTokenFilter) OpenBankingUtils.getClassInstanceFromFQN(configurationMap - .get(IdentityCommonConstants.TOKEN_FILTER).toString()); - } - - public OBResponseTypeHandler getObResponseTypeHandler() { - return obResponseTypeHandler; - } - - public RegistrationValidator getRegistrationValidator() { - - return registrationValidator; - } - - public void setRegistrationValidator(RegistrationValidator registrationValidator) { - - this.registrationValidator = registrationValidator; - } - - public ClaimProvider getClaimProvider() { - - return claimProvider; - } - - public void setClaimProvider(ClaimProvider claimProvider) { - - this.claimProvider = claimProvider; - } - - public IntrospectionDataProvider getIntrospectionDataProvider() { - - return introspectionDataProvider; - } - - public void setIntrospectionDataProvider(IntrospectionDataProvider introspectionDataProvider) { - - this.introspectionDataProvider = introspectionDataProvider; - } - - public OBRequestObjectValidator getObRequestObjectValidator() { - return obRequestObjectValidator; - } - - public PushAuthRequestValidator getPushAuthRequestValidator() { - - return pushAuthRequestValidator; - } - - public void setPushAuthRequestValidator(PushAuthRequestValidator pushAuthRequestValidator) { - - this.pushAuthRequestValidator = pushAuthRequestValidator; - } - - public void setDcrRegistrationConfigMap(Map> dcrRegConfigMap) { - - dcrRegistrationConfigMap = dcrRegConfigMap; - } - - public Map> getDcrRegistrationConfigMap() { - - return dcrRegistrationConfigMap; - } - - public AbstractApplicationUpdater getAbstractApplicationUpdater() { - - return abstractApplicationUpdater; - } - - public void setAbstractApplicationUpdater(AbstractApplicationUpdater abstractApplicationUpdater) { - - this.abstractApplicationUpdater = abstractApplicationUpdater; - } - - - public int getIdentityCacheAccessExpiry() { - - return identityCacheAccessExpiry; - } - - public void setIdentityCacheAccessExpiry(String identityCacheAccessExpiry) { - - this.identityCacheAccessExpiry = identityCacheAccessExpiry == null ? 60 : - Integer.parseInt(identityCacheAccessExpiry); - } - - public int getIdentityCacheModifiedExpiry() { - - return identityCacheModifiedExpiry; - } - - public void setIdentityCacheModifiedExpiry(String identityCacheModifiedExpiry) { - - this.identityCacheModifiedExpiry = identityCacheModifiedExpiry == null ? 60 : - Integer.parseInt(identityCacheModifiedExpiry); - } - - public KeyStore getTrustStore() { - return trustStore; - } - - public void setTrustStore(KeyStore trustStore) { - this.trustStore = trustStore; - } - - public RealmService getRealmService() { - - if (realmService == null) { - throw new RuntimeException("Realm Service is not available. Component did not start correctly."); - } - return realmService; - } - - void setRealmService(RealmService realmService) { - - this.realmService = realmService; - } - - /** - * Return OBThrottleService. - * - * @return OBThrottleService - */ - public OBThrottleService getOBThrottleService() { - return obThrottleService; - } - - /** - * Set OBThrottleService. - */ - public void setOBThrottleService(OBThrottleService obThrottleService) { - this.obThrottleService = obThrottleService; - } - public ConsentCoreService getConsentCoreService() { - - return consentCoreService; - } - - public void setConsentCoreService(ConsentCoreService consentCoreService) { - - this.consentCoreService = consentCoreService; - } - - /** - * Return OAuthClientAuthnService. - * - * @return OAuthClientAuthnService - */ - public OAuthClientAuthnService getOAuthClientAuthnService() { - return oAuthClientAuthnService; - } - - /** - * Set OAuthClientAuthnService. - */ - public void setOAuthClientAuthnService(OAuthClientAuthnService oAuthClientAuthnService) { - this.oAuthClientAuthnService = oAuthClientAuthnService; - IdentityServiceExporter.setOAuthClientAuthnService(oAuthClientAuthnService); - } - - /** - * To get the instance of {@link OAuth2Service}. - * - * @return OAuth2Service - */ - public OAuth2Service getOAuth2Service() { - - return oAuth2Service; - } - - /** - * To set the OAuth2Service. - * - * @param oAuth2Service instance of {@link OAuth2Service} - */ - public void setOAuth2Service(OAuth2Service oAuth2Service) { - - this.oAuth2Service = oAuth2Service; - } - - public void setJsFunctionRegistry(JsFunctionRegistry jsFunctionRegistry) { - - this.jsFunctionRegistry = jsFunctionRegistry; - } - - public JsFunctionRegistry getJsFunctionRegistry() { - - return jsFunctionRegistry; - } - - public Map getWorkers() { - return workers; - } - - public void addWorker(OpenBankingAuthenticationWorker worker, String workerName) { - this.workers.put(workerName, worker); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/internal/IdentityExtensionsServiceComponent.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/internal/IdentityExtensionsServiceComponent.java deleted file mode 100644 index a1f76804..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/internal/IdentityExtensionsServiceComponent.java +++ /dev/null @@ -1,279 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.internal; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.consent.mgt.service.ConsentCoreService; -import com.wso2.openbanking.accelerator.identity.app2app.App2AppAuthenticator; -import com.wso2.openbanking.accelerator.identity.auth.extensions.adaptive.function.OpenBankingAuthenticationWorkerFunction; -import com.wso2.openbanking.accelerator.identity.auth.extensions.adaptive.function.OpenBankingAuthenticationWorkerFunctionImpl; -import com.wso2.openbanking.accelerator.identity.authenticator.OBIdentifierAuthenticator; -import com.wso2.openbanking.accelerator.identity.claims.OBClaimProvider; -import com.wso2.openbanking.accelerator.identity.claims.RoleClaimProviderImpl; -import com.wso2.openbanking.accelerator.identity.clientauth.OBMutualTLSClientAuthenticator; -import com.wso2.openbanking.accelerator.identity.interceptor.OBIntrospectionDataProvider; -import com.wso2.openbanking.accelerator.identity.keyidprovider.OBKeyIDProvider; -import com.wso2.openbanking.accelerator.identity.listener.TokenRevocationListener; -import com.wso2.openbanking.accelerator.identity.listener.application.OBApplicationManagementListener; -import com.wso2.openbanking.accelerator.throttler.service.OBThrottleService; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.osgi.framework.BundleContext; -import org.osgi.service.component.ComponentContext; -import org.osgi.service.component.annotations.Activate; -import org.osgi.service.component.annotations.Component; -import org.osgi.service.component.annotations.Deactivate; -import org.osgi.service.component.annotations.Reference; -import org.osgi.service.component.annotations.ReferenceCardinality; -import org.osgi.service.component.annotations.ReferencePolicy; -import org.wso2.carbon.identity.application.authentication.framework.ApplicationAuthenticator; -import org.wso2.carbon.identity.application.authentication.framework.JsFunctionRegistry; -import org.wso2.carbon.identity.application.mgt.ApplicationManagementService; -import org.wso2.carbon.identity.application.mgt.listener.ApplicationMgtListener; -import org.wso2.carbon.identity.oauth.OAuthAdminServiceImpl; -import org.wso2.carbon.identity.oauth.event.OAuthEventInterceptor; -import org.wso2.carbon.identity.oauth2.IntrospectionDataProvider; -import org.wso2.carbon.identity.oauth2.OAuth2Service; -import org.wso2.carbon.identity.oauth2.client.authentication.OAuthClientAuthenticator; -import org.wso2.carbon.identity.oauth2.client.authentication.OAuthClientAuthnService; -import org.wso2.carbon.identity.oauth2.keyidprovider.KeyIDProvider; -import org.wso2.carbon.identity.openidconnect.ClaimProvider; -import org.wso2.carbon.identity.openidconnect.RequestObjectService; -import org.wso2.carbon.user.core.service.RealmService; - -/** - * Identity open banking common data holder. - */ -@Component( - name = "com.wso2.openbanking.accelerator.identity.IdentityExtensionsServiceComponent", - immediate = true -) -public class IdentityExtensionsServiceComponent { - - private static Log log = LogFactory.getLog(IdentityExtensionsServiceComponent.class); - - @Activate - protected void activate(ComponentContext context) { - - BundleContext bundleContext = context.getBundleContext(); - log.debug("Registering OB related Identity services."); - bundleContext.registerService(ApplicationMgtListener.class, new OBApplicationManagementListener(), null); - bundleContext.registerService(OAuthClientAuthenticator.class.getName(), - new OBMutualTLSClientAuthenticator(), null); - bundleContext.registerService(ApplicationManagementService.class, ApplicationManagementService.getInstance(), - null); - bundleContext.registerService(ClaimProvider.class.getName(), new OBClaimProvider(), null); - bundleContext.registerService(IntrospectionDataProvider.class.getName(), new OBIntrospectionDataProvider(), - null); - bundleContext.registerService(KeyIDProvider.class.getName(), new OBKeyIDProvider(), null); - bundleContext.registerService(ApplicationAuthenticator.class.getName(), - new OBIdentifierAuthenticator(), null); - bundleContext.registerService(ClaimProvider.class.getName(), new RoleClaimProviderImpl(), null); - bundleContext.registerService(OAuthEventInterceptor.class, new TokenRevocationListener(), null); - App2AppAuthenticator app2AppAuthenticator = new App2AppAuthenticator(); - bundleContext.registerService(ApplicationAuthenticator.class.getName(), - app2AppAuthenticator, null); - - if (IdentityExtensionsDataHolder.getInstance().getJsFunctionRegistry() != null) { - JsFunctionRegistry jsFunctionRegistry = IdentityExtensionsDataHolder.getInstance().getJsFunctionRegistry(); - OpenBankingAuthenticationWorkerFunction worker = new OpenBankingAuthenticationWorkerFunctionImpl(); - jsFunctionRegistry.register(JsFunctionRegistry.Subsystem.SEQUENCE_HANDLER, "OBAuthenticationWorker", - worker); - } - - } - - @Reference( - name = "ApplicationManagementService", - service = ApplicationManagementService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetApplicationManagementService" - ) - protected void setApplicationManagementService(ApplicationManagementService mgtService) { - - IdentityExtensionsDataHolder.getInstance().setApplicationManagementService(mgtService); - } - - protected void unsetApplicationManagementService(ApplicationManagementService mgtService) { - - IdentityExtensionsDataHolder.getInstance().setApplicationManagementService(null); - } - - @Reference( - name = "RequestObjectService", - service = RequestObjectService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetRequestObjectService" - ) - protected void setRequestObjectService(RequestObjectService requestObjectService) { - - IdentityExtensionsDataHolder.getInstance().setRequestObjectService(requestObjectService); - } - - protected void unsetRequestObjectService(RequestObjectService requestObjectService) { - - IdentityExtensionsDataHolder.getInstance().setRequestObjectService(null); - } - - @Reference( - service = OAuthAdminServiceImpl.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetOauthAdminService" - ) - protected void setOauthAdminService(OAuthAdminServiceImpl oauthAdminService) { - - IdentityExtensionsDataHolder.getInstance().setOauthAdminService(oauthAdminService); - } - - protected void unsetOauthAdminService(OAuthAdminServiceImpl oAuthAdminService) { - - IdentityExtensionsDataHolder.getInstance().setOauthAdminService(null); - } - - @Reference( - service = OpenBankingConfigurationService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetConfigService" - ) - public void setConfigService(OpenBankingConfigurationService openBankingConfigurationService) { - - IdentityExtensionsDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - } - - public void unsetConfigService(OpenBankingConfigurationService openBankingConfigurationService) { - - IdentityExtensionsDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - } - - @Reference( - name = "realm.service", - service = RealmService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetRealmService" - ) - protected void setRealmService(RealmService realmService) { - - log.debug("Setting the Realm Service"); - IdentityExtensionsDataHolder.getInstance().setRealmService(realmService); - } - - protected void unsetRealmService(RealmService realmService) { - - log.debug("UnSetting the Realm Service"); - IdentityExtensionsDataHolder.getInstance().setRealmService(null); - } - - @Reference(name = "open.banking.throttle.service", - service = OBThrottleService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetOBThrottleService" - ) - protected void setOBThrottleService(OBThrottleService throttleService) { - - log.debug("OBThrottleService bound to the ob-identifier-authenticator"); - IdentityExtensionsDataHolder.getInstance().setOBThrottleService(throttleService); - } - - protected void unsetOBThrottleService(OBThrottleService throttleService) { - - log.debug("OBThrottleService unbound from the ob-identifier-authenticator"); - IdentityExtensionsDataHolder.getInstance().setOBThrottleService(null); - } - - @Reference( - service = com.wso2.openbanking.accelerator.consent.mgt.service.ConsentCoreService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetConsentCoreService" - ) - public void setConsentCoreService(ConsentCoreService consentCoreService) { - - log.debug("Setting the Consent Core Service"); - IdentityExtensionsDataHolder.getInstance().setConsentCoreService(consentCoreService); - } - - public void unsetConsentCoreService(ConsentCoreService consentCoreService) { - - log.debug("UnSetting the Consent Core Service"); - IdentityExtensionsDataHolder.getInstance().setConsentCoreService(null); - } - - @Reference(name = "oauth.client.authn.service", - service = OAuthClientAuthnService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetOAuthClientAuthnService" - ) - protected void setOAuthClientAuthnService(OAuthClientAuthnService oAuthClientAuthnService) { - IdentityExtensionsDataHolder.getInstance().setOAuthClientAuthnService(oAuthClientAuthnService); - } - - protected void unsetOAuthClientAuthnService(OAuthClientAuthnService oAuthClientAuthnService) { - IdentityExtensionsDataHolder.getInstance().setOAuthClientAuthnService(null); - } - - @Reference( - service = OAuth2Service.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetOAuth2Service" - ) - public void setOAuth2Service(OAuth2Service oAuth2Service) { - log.debug("Setting the OAuth2 Service"); - IdentityExtensionsDataHolder.getInstance().setOAuth2Service(oAuth2Service); - } - - public void unsetOAuth2Service(OAuth2Service oAuth2Service) { - log.debug("UnSetting the OAuth2 Service"); - IdentityExtensionsDataHolder.getInstance().setOAuth2Service(null); - } - - @Deactivate - protected void deactivate(ComponentContext ctxt) { - - if (IdentityExtensionsDataHolder.getInstance().getJsFunctionRegistry() != null) { - JsFunctionRegistry jsFunctionRegistry = IdentityExtensionsDataHolder.getInstance().getJsFunctionRegistry(); - jsFunctionRegistry.register(JsFunctionRegistry.Subsystem.SEQUENCE_HANDLER, "OBAuthenticationWorker", - null); - } - log.debug("Open banking Key Manager Extensions component is deactivated"); - } - - @Reference( - service = JsFunctionRegistry.class, - cardinality = ReferenceCardinality.OPTIONAL, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetJsFunctionRegistry" - ) - public void setJsFunctionRegistry(JsFunctionRegistry jsFunctionRegistry) { - - IdentityExtensionsDataHolder.getInstance().setJsFunctionRegistry(jsFunctionRegistry); - } - - public void unsetJsFunctionRegistry(JsFunctionRegistry jsFunctionRegistry) { - - IdentityExtensionsDataHolder.getInstance().setJsFunctionRegistry(null); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/keyidprovider/OBKeyIDProvider.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/keyidprovider/OBKeyIDProvider.java deleted file mode 100644 index 1d362066..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/keyidprovider/OBKeyIDProvider.java +++ /dev/null @@ -1,70 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.keyidprovider; - -import com.nimbusds.jose.JWSAlgorithm; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.keyidprovider.DefaultKeyIDProviderImpl; - -import java.security.cert.Certificate; -import java.util.Optional; - -/** - * OB specific Key ID provider implementation. - */ -public class OBKeyIDProvider extends DefaultKeyIDProviderImpl { - - private static final Log log = LogFactory.getLog(OBKeyIDProvider.class); - - /** - * Method containing the KeyID calculation logic for OB. - * - * @param certificate Signing Certificate. - * @param signatureAlgorithm Signature Algorithm configured. - * @param tenantDomain tenant domain of the user. - * @return Key ID as a String. - * @throws IdentityOAuth2Exception When fail to generate the Key ID. - */ - @Override - public String getKeyId(Certificate certificate, JWSAlgorithm signatureAlgorithm, String tenantDomain) - throws IdentityOAuth2Exception { - - String kid; - Optional primaryCertKid = Optional.ofNullable(IdentityExtensionsDataHolder.getInstance() - .getConfigurationMap().get(IdentityCommonConstants.SIGNING_CERT_KID)); - if (primaryCertKid.isPresent()) { - kid = primaryCertKid.get().toString(); - if (log.isDebugEnabled()) { - log.debug("KID value is configured in the open-banking.xml. Therefore returning configured value :" - + kid + " as the KID"); - } - return kid; - } else { - if (log.isDebugEnabled()) { - log.debug("KID value is not configured in the open-banking.xml Therefore calling the default Key ID " + - "provider implementation"); - } - return super.getKeyId(certificate, signatureAlgorithm, tenantDomain); - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/listener/TokenRevocationListener.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/listener/TokenRevocationListener.java deleted file mode 100644 index 2953ce34..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/listener/TokenRevocationListener.java +++ /dev/null @@ -1,108 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.listener; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.oauth.event.AbstractOAuthEventInterceptor; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.dto.OAuthRevocationRequestDTO; -import org.wso2.carbon.identity.oauth2.dto.OAuthRevocationResponseDTO; -import org.wso2.carbon.identity.oauth2.model.AccessTokenDO; -import org.wso2.carbon.identity.oauth2.model.RefreshTokenValidationDataDO; - -import java.util.Map; - -/** - * Event listener to revoke consents when access token is revoked. - */ -public class TokenRevocationListener extends AbstractOAuthEventInterceptor { - - private static final Log log = LogFactory.getLog(TokenRevocationListener.class); - private static final ConsentCoreServiceImpl consentCoreService = new ConsentCoreServiceImpl(); - - /** - * Revoke the consent bound to the access token after revoking the access token. - * - * @param revokeRequestDTO - * @param revokeResponseDTO - * @param accessTokenDO - * @param refreshTokenDO - * @param params - * @throws IdentityOAuth2Exception - */ - @Override - @Generated(message = "Excluding from code coverage since it requires a service call") - public void onPostTokenRevocationByClient(OAuthRevocationRequestDTO revokeRequestDTO, - OAuthRevocationResponseDTO revokeResponseDTO, - AccessTokenDO accessTokenDO, - RefreshTokenValidationDataDO refreshTokenDO, - Map params) throws IdentityOAuth2Exception { - - if (!revokeRequestDTO.getoAuthClientAuthnContext().isAuthenticated()) { - return; - } - String consentId = ""; - if (accessTokenDO != null) { - consentId = getConsentIdFromScopes(accessTokenDO.getScope()); - } else if (refreshTokenDO != null) { - consentId = getConsentIdFromScopes(refreshTokenDO.getScope()); - } - if (StringUtils.isNotEmpty(consentId)) { - try { - // Skip consent revocation if the request is from consent revocation flow. - boolean isConsentRevocationFlow = revokeRequestDTO.getoAuthClientAuthnContext().getParameters(). - containsKey(OpenBankingConstants.IS_CONSENT_REVOCATION_FLOW) && (boolean) revokeRequestDTO. - getoAuthClientAuthnContext().getParameter(OpenBankingConstants.IS_CONSENT_REVOCATION_FLOW); - if (!isConsentRevocationFlow) { - consentCoreService.revokeConsentWithReason(consentId, OpenBankingConstants. - DEFAULT_STATUS_FOR_REVOKED_CONSENTS, null, false, "Revoked by token revocation"); - } - } catch (ConsentManagementException e) { - log.error(String.format("Error occurred while revoking consent on token revocation. %s", - e.getMessage().replaceAll("[\r\n]", ""))); - } - } - } - - /** - * Return consent-id when a string array of scopes is given. - * - * @param scopes - * @return - */ - public String getConsentIdFromScopes(String[] scopes) { - - String consentIdClaim = OpenBankingConfigParser.getInstance().getConfiguration() - .get(OpenBankingConstants.CONSENT_ID_CLAIM_NAME).toString(); - if (scopes != null) { - for (String scope : scopes) { - if (scope.contains(consentIdClaim)) { - return scope.split(consentIdClaim)[1]; - } - } - } - return null; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/listener/application/AbstractApplicationUpdater.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/listener/application/AbstractApplicationUpdater.java deleted file mode 100644 index 6317af07..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/listener/application/AbstractApplicationUpdater.java +++ /dev/null @@ -1,72 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.listener.application; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import org.wso2.carbon.identity.application.common.model.LocalAndOutboundAuthenticationConfig; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.application.common.model.ServiceProviderProperty; -import org.wso2.carbon.identity.oauth.dto.OAuthConsumerAppDTO; - -import java.util.Map; - -/** - * Abstract class for extending methods to be invoked by the application listener. - */ -public abstract class AbstractApplicationUpdater { - - public abstract void setOauthAppProperties(boolean isRegulatoryApp, OAuthConsumerAppDTO oauthApplication, - Map spMetaData) throws OpenBankingException; - - public abstract void setServiceProviderProperties(boolean isRegulatoryApp, ServiceProvider serviceProvider, - ServiceProviderProperty[] serviceProvideProperties) - throws OpenBankingException; - - public abstract void setAuthenticators(boolean isRegulatoryApp, String tenantDomain, - ServiceProvider serviceProvider, - LocalAndOutboundAuthenticationConfig localAndOutboundAuthenticationConfig) - throws OpenBankingException; - - public abstract void setConditionalAuthScript (boolean isRegulatoryApp, ServiceProvider serviceProvider, - LocalAndOutboundAuthenticationConfig localAndOutboundAuthenticationConfig) - throws OpenBankingException; - - public abstract void publishData(Map spMetaData, OAuthConsumerAppDTO oAuthConsumerAppDTO) - throws OpenBankingException; - - public abstract void doPreCreateApplication(boolean isRegulatoryApp, ServiceProvider serviceProvider, - LocalAndOutboundAuthenticationConfig - localAndOutboundAuthenticationConfig, - String tenantDomain, String userName) throws OpenBankingException; - - public abstract void doPostGetApplication(ServiceProvider serviceProvider, String applicationName, - String tenantDomain) throws OpenBankingException; - - public abstract void doPreUpdateApplication(boolean isRegulatoryApp, OAuthConsumerAppDTO oauthApplication, - ServiceProvider serviceProvider, LocalAndOutboundAuthenticationConfig - localAndOutboundAuthenticationConfig, String tenantDomain, - String userName) - throws OpenBankingException; - - public abstract void doPreDeleteApplication(String applicationName, String tenantDomain, String userName) - throws OpenBankingException; - - public abstract void doPostDeleteApplication(ServiceProvider serviceProvider, String tenantDomain, String userName) - throws OpenBankingException; - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/listener/application/ApplicationUpdaterImpl.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/listener/application/ApplicationUpdaterImpl.java deleted file mode 100644 index 01055137..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/listener/application/ApplicationUpdaterImpl.java +++ /dev/null @@ -1,278 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.listener.application; - -import com.google.gson.Gson; -import com.wso2.openbanking.accelerator.common.config.TextFileReader; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.context.CarbonContext; -import org.wso2.carbon.identity.application.common.IdentityApplicationManagementException; -import org.wso2.carbon.identity.application.common.model.AuthenticationStep; -import org.wso2.carbon.identity.application.common.model.IdentityProvider; -import org.wso2.carbon.identity.application.common.model.LocalAndOutboundAuthenticationConfig; -import org.wso2.carbon.identity.application.common.model.LocalAuthenticatorConfig; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.application.common.model.ServiceProviderProperty; -import org.wso2.carbon.identity.application.common.model.script.AuthenticationScriptConfig; -import org.wso2.carbon.identity.application.mgt.ApplicationManagementService; -import org.wso2.carbon.identity.oauth.IdentityOAuthAdminException; -import org.wso2.carbon.identity.oauth.dto.OAuthConsumerAppDTO; - -import java.io.IOException; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.List; -import java.util.Map; - -/** - * Default implementation class for AbstractApplicationUpdater which should be extended for spec specific. - * tasks - */ -public class ApplicationUpdaterImpl extends AbstractApplicationUpdater { - - private static final Log logger = LogFactory.getLog(ApplicationUpdaterImpl.class); - - public void setOauthAppProperties(boolean isRegulatoryApp, OAuthConsumerAppDTO oauthApplication, - Map spMetaData) throws OpenBankingException { - - } - - public void setServiceProviderProperties(boolean isRegulatoryApp, ServiceProvider serviceProvider, - ServiceProviderProperty[] serviceProvideProperties) - throws OpenBankingException { - - } - - public void setAuthenticators (boolean isRegulatoryApp, String tenantDomain, ServiceProvider serviceProvider, - LocalAndOutboundAuthenticationConfig localAndOutboundAuthenticationConfig) - throws OpenBankingException { - - IdentityExtensionsDataHolder identityExtensionsDataHolder = IdentityExtensionsDataHolder.getInstance(); - if (isRegulatoryApp) { - - Map configMap = identityExtensionsDataHolder.getConfigurationMap(); - List authSteps = new ArrayList<>(); - - //Read the identity provider from open-banking.xml - String idpName = configMap.get(IdentityCommonConstants.IDENTITY_PROVIDER_NAME) == null - ? null : configMap.get(IdentityCommonConstants.IDENTITY_PROVIDER_NAME).toString(); - String idpStep = configMap.get(IdentityCommonConstants.IDENTITY_PROVIDER_STEP) == null - ? null : configMap.get(IdentityCommonConstants.IDENTITY_PROVIDER_STEP).toString(); - - IdentityProvider configuredIdentityProvider = null; - - if (StringUtils.isNotEmpty(idpName)) { - try { - IdentityProvider[] federatedIdPs = identityExtensionsDataHolder - .getApplicationManagementService().getAllIdentityProviders(tenantDomain); - if (federatedIdPs != null && federatedIdPs.length > 0) { - for (IdentityProvider identityProvider : federatedIdPs) { - if (idpName.equals(identityProvider.getIdentityProviderName())) { - configuredIdentityProvider = identityProvider; - break; - } - } - } - } catch (IdentityApplicationManagementException e) { - throw new OpenBankingException("Error while reading configured Identity providers", e); - } - - } - - if (StringUtils.isNotEmpty(idpStep) && idpStep.equals("1")) { - //Step 1 - Federated Authentication - if (configuredIdentityProvider != null) { - IdentityProvider[] identityProviders = new IdentityProvider[1]; - identityProviders[0] = configuredIdentityProvider; - - AuthenticationStep federatedAuthStep = new AuthenticationStep(); - federatedAuthStep.setStepOrder(1); - federatedAuthStep.setFederatedIdentityProviders(identityProviders); - //set step 1 - authSteps.add(federatedAuthStep); - if (logger.isDebugEnabled()) { - logger.debug("Authentication step 1 added: " + idpName); - } - localAndOutboundAuthenticationConfig.setAuthenticationSteps( - authSteps.toArray(new AuthenticationStep[0])); - } else { - throw new OpenBankingException("Error! An Identity Provider has not been configured."); - } - } else { - //Step 1 - Default basic authentication - LocalAuthenticatorConfig localAuthenticatorConfig = new LocalAuthenticatorConfig(); - LocalAuthenticatorConfig[] localAuthenticatorConfigs = new LocalAuthenticatorConfig[1]; - AuthenticationStep basicAuthenticationStep = new AuthenticationStep(); - - String authenticatorDisplayName = configMap. - get(IdentityCommonConstants.PRIMARY_AUTHENTICATOR_DISPLAYNAME).toString(); - String authenticatorName = configMap.get(IdentityCommonConstants.PRIMARY_AUTHENTICATOR_NAME).toString(); - - localAuthenticatorConfig.setDisplayName(authenticatorDisplayName); - localAuthenticatorConfig.setEnabled(true); - localAuthenticatorConfig.setName(authenticatorName); - localAuthenticatorConfigs[0] = localAuthenticatorConfig; - - basicAuthenticationStep.setStepOrder(1); - basicAuthenticationStep.setLocalAuthenticatorConfigs(localAuthenticatorConfigs); - basicAuthenticationStep.setAttributeStep(true); - basicAuthenticationStep.setSubjectStep(true); - //set step 1 - authSteps.add(basicAuthenticationStep); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("Authentication step 1 added: %s", authenticatorName)); - } - - //Step 2 - federated authentication - if (configuredIdentityProvider != null) { - IdentityProvider[] identityProviders = new IdentityProvider[1]; - identityProviders[0] = configuredIdentityProvider; - - AuthenticationStep federatedAuthStep = new AuthenticationStep(); - federatedAuthStep.setStepOrder(2); - federatedAuthStep.setFederatedIdentityProviders(identityProviders); - //set step 2 - authSteps.add(federatedAuthStep); - if (logger.isDebugEnabled()) { - logger.debug("Authentication step 2 added: " + idpName); - } - } - localAndOutboundAuthenticationConfig.setAuthenticationSteps(authSteps.toArray( - new AuthenticationStep[0])); - } - } - } - - public void setConditionalAuthScript(boolean isRegulatoryApp, ServiceProvider serviceProvider, - LocalAndOutboundAuthenticationConfig localAndOutboundAuthenticationConfig) - throws OpenBankingException { - - if (isRegulatoryApp) { - if (localAndOutboundAuthenticationConfig.getAuthenticationScriptConfig() == null) { - TextFileReader textFileReader = TextFileReader.getInstance(); - try { - String authScript = textFileReader.readFile - (IdentityCommonConstants.CONDITIONAL_COMMON_AUTH_SCRIPT_FILE_NAME); - if (StringUtils.isNotEmpty(authScript)) { - AuthenticationScriptConfig scriptConfig = new AuthenticationScriptConfig(); - scriptConfig.setContent(authScript); - scriptConfig.setEnabled(true); - localAndOutboundAuthenticationConfig.setAuthenticationScriptConfig(scriptConfig); - } - } catch (IOException e) { - throw new OpenBankingException("Error occurred while reading file", e); - } - - } - } - } - - public void publishData(Map spMetaData, OAuthConsumerAppDTO oAuthConsumerAppDTO) - throws OpenBankingException { - - } - - public void doPreCreateApplication(boolean isRegulatoryApp, ServiceProvider serviceProvider, - LocalAndOutboundAuthenticationConfig localAndOutboundAuthenticationConfig, - String tenantDomain, String userName) throws OpenBankingException { - - } - - public void doPostGetApplication(ServiceProvider serviceProvider, String applicationName, - String tenantDomain) throws OpenBankingException { - - } - - public void doPreUpdateApplication(boolean isRegulatoryApp, OAuthConsumerAppDTO oauthApplication, - ServiceProvider serviceProvider, LocalAndOutboundAuthenticationConfig - localAndOutboundAuthenticationConfig, String tenantDomain, - String userName) throws OpenBankingException { - - try { - boolean updateAuthenticator = false; - List spProperties = new ArrayList<>(Arrays.asList - (serviceProvider.getSpProperties())); - ServiceProviderProperty appCreateRequest = spProperties.stream() - .filter(serviceProviderProperty -> serviceProviderProperty.getName() - .equalsIgnoreCase("AppCreateRequest")).findAny().orElse(null); - // Authenticators are updated only when creating the app or when an authenticator change - // is made from the IS carbon console - if (appCreateRequest != null) { - updateAuthenticator = Boolean.parseBoolean(appCreateRequest.getValue()); - // Removing the added additional SP property to identify create and update requests - spProperties.remove(appCreateRequest); - serviceProvider.setSpProperties(spProperties.toArray(new ServiceProviderProperty[0])); - } else { - ApplicationManagementService applicationManagementService = IdentityExtensionsDataHolder.getInstance() - .getApplicationManagementService(); - ServiceProvider existingSP = applicationManagementService - .getServiceProvider(serviceProvider.getApplicationID()); - // Checking whether any change have been made in the Local & Outbound Configs of the SP - if (!new Gson().toJson(localAndOutboundAuthenticationConfig).equals(new Gson().toJson(existingSP - .getLocalAndOutBoundAuthenticationConfig()))) { - updateAuthenticator = true; - } - } - - if (localAndOutboundAuthenticationConfig == null) { - localAndOutboundAuthenticationConfig = new LocalAndOutboundAuthenticationConfig(); - } - localAndOutboundAuthenticationConfig.setUseTenantDomainInLocalSubjectIdentifier(true); - localAndOutboundAuthenticationConfig.setUseUserstoreDomainInLocalSubjectIdentifier(true); - - if (updateAuthenticator) { - localAndOutboundAuthenticationConfig.setAuthenticationType("flow"); - setAuthenticators(isRegulatoryApp, tenantDomain, serviceProvider, localAndOutboundAuthenticationConfig); - setConditionalAuthScript(isRegulatoryApp, serviceProvider, localAndOutboundAuthenticationConfig); - } - //update service provider Properties - setServiceProviderProperties(isRegulatoryApp, serviceProvider, serviceProvider.getSpProperties()); - serviceProvider.setLocalAndOutBoundAuthenticationConfig(localAndOutboundAuthenticationConfig); - IdentityExtensionsDataHolder identityExtensionsDataHolder = IdentityExtensionsDataHolder.getInstance(); - Map spMetaData = IdentityCommonUtil.getSpMetaData(serviceProvider); - //update oauth application - setOauthAppProperties(isRegulatoryApp, oauthApplication, spMetaData); - if (StringUtils.isNotBlank(CarbonContext.getThreadLocalCarbonContext().getUsername())) { - identityExtensionsDataHolder.getOauthAdminService().updateConsumerApplication(oauthApplication); - } - publishData(spMetaData, oauthApplication); - } catch (IdentityOAuthAdminException e) { - throw new OpenBankingException("Error occurred while updating application ", e); - } catch (IdentityApplicationManagementException e) { - throw new OpenBankingException("Error occurred while retrieving service provider ", e); - } - } - - public void doPreDeleteApplication(String applicationName, String tenantDomain, String userName) - throws OpenBankingException { - - } - - public void doPostDeleteApplication(ServiceProvider serviceProvider, String tenantDomain, String userName) - throws OpenBankingException { - - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/listener/application/OBApplicationManagementListener.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/listener/application/OBApplicationManagementListener.java deleted file mode 100644 index 8b27eea6..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/listener/application/OBApplicationManagementListener.java +++ /dev/null @@ -1,179 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.listener.application; - -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.dcr.validation.DCRCommonConstants; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.application.common.IdentityApplicationManagementException; -import org.wso2.carbon.identity.application.common.model.LocalAndOutboundAuthenticationConfig; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.application.common.model.ServiceProviderProperty; -import org.wso2.carbon.identity.application.mgt.listener.AbstractApplicationMgtListener; -import org.wso2.carbon.identity.oauth.IdentityOAuthAdminException; -import org.wso2.carbon.identity.oauth.OAuthAdminServiceImpl; -import org.wso2.carbon.identity.oauth.dto.OAuthConsumerAppDTO; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.List; - -/** - * Application listener. - */ -public class OBApplicationManagementListener extends AbstractApplicationMgtListener { - - private static final Log log = LogFactory.getLog(OBApplicationManagementListener.class); - private IdentityExtensionsDataHolder identityExtensionsDataHolder = IdentityExtensionsDataHolder.getInstance(); - - @Override - public int getDefaultOrderId() { - - return 1000; - } - - @Override - public boolean doPreUpdateApplication(ServiceProvider serviceProvider, String tenantDomain, String userName) - throws IdentityApplicationManagementException { - - try { - boolean isRegulatory = false; - List regulatoryIssuerList = new ArrayList<>(); - Object regulatoryIssuers = identityExtensionsDataHolder.getConfigurationMap() - .get(DCRCommonConstants.REGULATORY_ISSUERS); - if (regulatoryIssuers != null) { - if (regulatoryIssuers instanceof List) { - regulatoryIssuerList = (List) regulatoryIssuers; - } else { - regulatoryIssuerList.add(regulatoryIssuers.toString()); - } - } - - List spProperties = new ArrayList<>(Arrays.asList - (serviceProvider.getSpProperties())); - - ServiceProviderProperty ssaIssuerProperty = spProperties.stream() - .filter(serviceProviderProperty -> serviceProviderProperty.getName() - .equalsIgnoreCase("ssaIssuer")).findAny().orElse(null); - - ServiceProviderProperty regulatoryProperty = spProperties.stream() - .filter(serviceProviderProperty -> serviceProviderProperty.getName() - .equalsIgnoreCase(OpenBankingConstants.REGULATORY)).findAny().orElse(null); - - if (ssaIssuerProperty != null) { - String ssaIssuer = ssaIssuerProperty.getValue(); - isRegulatory = regulatoryIssuerList.stream().anyMatch(issuer -> issuer.equals(ssaIssuer)); - } else if (regulatoryProperty != null) { - isRegulatory = Boolean.parseBoolean(regulatoryProperty.getValue()); - } - - //check whether regulatory property is already stored - if (regulatoryProperty == null && isRegulatory) { - - spProperties.add(IdentityCommonUtil.getServiceProviderProperty(OpenBankingConstants.REGULATORY, - "true")); - - } else if (regulatoryProperty == null && !isRegulatory) { - - spProperties.add(IdentityCommonUtil.getServiceProviderProperty(OpenBankingConstants.REGULATORY, - "false")); - - } else if (regulatoryProperty != null && isRegulatory) { - spProperties.remove(regulatoryProperty); - spProperties.add(IdentityCommonUtil.getServiceProviderProperty(OpenBankingConstants.REGULATORY, - "true")); - } - serviceProvider.setSpProperties(spProperties.toArray(new ServiceProviderProperty[0])); - OAuthConsumerAppDTO oAuthConsumerAppDTO; - OAuthAdminServiceImpl oAuthAdminService = identityExtensionsDataHolder.getOauthAdminService(); - - oAuthConsumerAppDTO = oAuthAdminService - .getOAuthApplicationDataByAppName(serviceProvider.getApplicationName()); - oAuthConsumerAppDTO.setTokenType("JWT"); - LocalAndOutboundAuthenticationConfig localAndOutboundAuthenticationConfig = serviceProvider - .getLocalAndOutBoundAuthenticationConfig(); - - identityExtensionsDataHolder.getAbstractApplicationUpdater() - .doPreUpdateApplication(isRegulatory, oAuthConsumerAppDTO, serviceProvider, - localAndOutboundAuthenticationConfig, tenantDomain, userName); - - } catch (OpenBankingException e) { - log.error("Error occurred while updating application.", e); - return false; - } catch (IdentityOAuthAdminException e) { - //returning true here because this error code is returned when there is no oauth app created - //when running integration tests of IS, test cases fail since apps are update before key generation in tests - if ("OAUTH-60002".equals(e.getErrorCode())) { - return true; - } - log.error("Error while retrieving oauth application", e); - return false; - } - return true; - } - - @Override - public boolean doPostGetServiceProvider(ServiceProvider serviceProvider, String applicationName, - String tenantDomain) throws IdentityApplicationManagementException { - - try { - identityExtensionsDataHolder.getAbstractApplicationUpdater() - .doPostGetApplication(serviceProvider, applicationName, tenantDomain); - } catch (OpenBankingException e) { - log.error("Error occurred while updating application.", e); - return false; - } - return true; - - } - - @Override - public boolean doPreDeleteApplication(String applicationName, String tenantDomain, String userName) - throws IdentityApplicationManagementException { - - try { - identityExtensionsDataHolder.getAbstractApplicationUpdater() - .doPreDeleteApplication(applicationName, tenantDomain, userName); - } catch (OpenBankingException e) { - log.error("Error occurred while updating application.", e); - return false; - } - return true; - - } - - @Override - public boolean doPostDeleteApplication(ServiceProvider serviceProvider, String tenantDomain, String userName) - throws IdentityApplicationManagementException { - - try { - identityExtensionsDataHolder.getAbstractApplicationUpdater() - .doPostDeleteApplication(serviceProvider, tenantDomain, userName); - } catch (OpenBankingException e) { - log.error("Error occurred while updating application.", e); - return false; - } - return true; - - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/PushAuthRequestValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/PushAuthRequestValidator.java deleted file mode 100644 index bbbbeeb5..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/PushAuthRequestValidator.java +++ /dev/null @@ -1,304 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator; - -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.constants.PushAuthRequestConstants; -import com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.exception.PushAuthRequestValidatorException; -import com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.model.PushAuthErrorResponse; -import com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.util.PushAuthRequestValidatorUtils; -import net.minidev.json.JSONObject; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.HttpStatus; -import org.wso2.carbon.identity.oauth2.OAuth2Service; -import org.wso2.carbon.identity.oauth2.dto.OAuth2ClientValidationResponseDTO; - -import java.text.ParseException; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import javax.servlet.http.HttpServletRequest; - -import static org.wso2.carbon.identity.openidconnect.model.Constants.JWT_PART_DELIMITER; -import static org.wso2.carbon.identity.openidconnect.model.Constants.NUMBER_OF_PARTS_IN_JWE; - -/** - * The extension class for enforcing OB Push Auth Request Validations. - */ -public class PushAuthRequestValidator { - - private static final Log log = LogFactory.getLog(PushAuthRequestValidator.class); - private static PushAuthRequestValidator pushAuthRequestValidator; - private static final String ERROR_DESCRIPTION = "error_description"; - private static final String ERROR = "error"; - - public static PushAuthRequestValidator getPushAuthRequestValidator() { - - return pushAuthRequestValidator; - } - - public static void setRegistrationValidator(PushAuthRequestValidator pushAuthRequestValidator) { - - PushAuthRequestValidator.pushAuthRequestValidator = pushAuthRequestValidator; - } - - public final Map validateParams(HttpServletRequest request, - Map> parameterMap) - throws PushAuthRequestValidatorException { - - Map parameters = new HashMap<>(); - - for (Map.Entry> paramEntry : parameterMap.entrySet()) { - if (paramEntry.getValue().size() > 1) { - if (log.isDebugEnabled()) { - log.debug("Repeated param found:" + paramEntry.getKey()); - } - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST, "Repeated parameter found in the request"); - } - parameters.put(paramEntry.getKey(), paramEntry.getValue().get(0)); - } - // push auth request cannot contain "request_uri" parameter - if (parameters.containsKey(PushAuthRequestConstants.REQUEST_URI)) { - log.error("Request does not allow request_uri parameter"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST, "Request does not allow request_uri parameter"); - } - JSONObject requestBodyJson; - JSONObject requestHeaderJson; - String requestObjectString; - - // if "request" parameter is available, decode it and put it into parameter map - if (parameters.containsKey(PushAuthRequestConstants.REQUEST)) { - // validate form body when "request" parameter is present - PushAuthRequestValidatorUtils.validateRequestFormBody(parameters); - - try { - String requestParam = parameters.get(PushAuthRequestConstants.REQUEST).toString(); - // check whether request is of type JWE - if (requestParam.split(JWT_PART_DELIMITER).length == NUMBER_OF_PARTS_IN_JWE) { - // decrypt JWE - requestObjectString = PushAuthRequestValidatorUtils.decrypt(requestParam, - parameters.get(PushAuthRequestConstants.CLIENT_ID) != null ? - parameters.get(PushAuthRequestConstants.CLIENT_ID).toString() : null); - } else { - requestObjectString = requestParam; - } - // decode jwt assuming it is a JWS otherwise, it will throw parse exception - requestBodyJson = JWTUtils.decodeRequestJWT(requestObjectString, PushAuthRequestConstants.BODY); - requestHeaderJson = JWTUtils.decodeRequestJWT(requestObjectString, PushAuthRequestConstants.HEADER); - // add to parameters map - parameters.put(PushAuthRequestConstants.DECODED_JWT_BODY, requestBodyJson); - parameters.put(PushAuthRequestConstants.DECODED_JWT_HEADER, requestHeaderJson); - - } catch (ParseException e) { - log.error("Exception while decoding JWT. Returning error.", e); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST_OBJECT, - "Unable to decode JWT.", e); - } - if (requestBodyJson != null && requestHeaderJson != null) { - - validateRedirectUri(requestBodyJson); - - // validate client id and redirect uri - OAuth2ClientValidationResponseDTO oAuth2ClientValidationResponseDTO = - getClientValidationInfo(requestBodyJson); - - if (!oAuth2ClientValidationResponseDTO.isValidClient()) { - log.error(oAuth2ClientValidationResponseDTO.getErrorMsg()); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST, - oAuth2ClientValidationResponseDTO.getErrorMsg()); - } - validateSignatureAlgorithm(requestHeaderJson - .get(PushAuthRequestConstants.ALG_HEADER)); - validateSignature(requestObjectString, requestBodyJson); - validateResponseType(requestBodyJson); - validateNonceParameter(requestBodyJson); - validateScope(requestBodyJson); - validateAudience(requestBodyJson); - validateIssuer(requestBodyJson); - validateExpirationTime(requestBodyJson); - validateNotBeforeClaim(requestBodyJson); - validatePKCEParameters(requestBodyJson); - - if (StringUtils.isNotBlank(requestBodyJson.getAsString(PushAuthRequestConstants.REQUEST)) - || StringUtils.isNotBlank(requestBodyJson - .getAsString(PushAuthRequestConstants.REQUEST_URI))) { - log.error("Both request and request_uri parameters are not allowed in the request object"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST_OBJECT, - "Both request and request_uri parameters are not allowed in the request object"); - } - } else { - log.error("Invalid JWT as request"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST, "Invalid JWT as request"); - } - } - - // call additional validations from toolkit extensions - validateAdditionalParams(parameters); - return parameters; - } - - /** - * Extend this method to perform additional validations on toolkits. - * - * @param parameters parameter map - */ - public void validateAdditionalParams(Map parameters) throws PushAuthRequestValidatorException { - - } - - /** - * Extend this method to validate the redirect URI of the request object. - * @param requestBodyJson request object - * @throws PushAuthRequestValidatorException if validation fails - */ - public void validateRedirectUri(JSONObject requestBodyJson) - throws PushAuthRequestValidatorException { - - PushAuthRequestValidatorUtils.validateRedirectUri(requestBodyJson); - } - - /** - * Extend this method to validate scopes of the request object. - * @param requestBodyJson request object - * @throws PushAuthRequestValidatorException if validation fails - */ - public void validateScope(JSONObject requestBodyJson) - throws PushAuthRequestValidatorException { - - PushAuthRequestValidatorUtils.validateScope(requestBodyJson); - } - - /** - * Extend this method to validate signature algorithm of the request object. - * @param algorithm signature algorithm - * @throws PushAuthRequestValidatorException if validation fails - */ - public void validateSignatureAlgorithm(Object algorithm) - throws PushAuthRequestValidatorException { - - PushAuthRequestValidatorUtils.validateSignatureAlgorithm(algorithm); - } - - /** - * Extend this method to validate nonce parameter of the request object. - * @param requestBodyJson request object - * @throws PushAuthRequestValidatorException if validation fails - */ - public void validateNonceParameter(JSONObject requestBodyJson) - throws PushAuthRequestValidatorException { - - PushAuthRequestValidatorUtils.validateNonceParameter(requestBodyJson); - } - - /** - * Extend this method to validate issuer of the request object. - * @param requestBodyJson request object - * @throws PushAuthRequestValidatorException if validation fails - */ - public void validateIssuer(JSONObject requestBodyJson) - throws PushAuthRequestValidatorException { - - PushAuthRequestValidatorUtils.validateIssuer(requestBodyJson); - } - - /** - * Extend this method to validate expiration time of the request object. - * @param requestBodyJson request object - * @throws PushAuthRequestValidatorException if validation fails - */ - public void validateExpirationTime(JSONObject requestBodyJson) - throws PushAuthRequestValidatorException { - - PushAuthRequestValidatorUtils.validateExpirationTime(requestBodyJson); - } - - /** - * Extend this method to validate nbf claim of the request object. - * @param requestBodyJson request object - * @throws PushAuthRequestValidatorException if validation fails - */ - public void validateNotBeforeClaim(JSONObject requestBodyJson) - throws PushAuthRequestValidatorException { - - PushAuthRequestValidatorUtils.validateNotBeforeClaim(requestBodyJson); - } - - /** - * Extend this method to create error response on toolkits. Set necessary status codes and error payloads to - * PushAuthErrorResponse. - * - * @param httpStatusCode Http status code - * @param errorCode Error code - * @param errorDescription Error description - */ - public PushAuthErrorResponse createErrorResponse(int httpStatusCode, String errorCode, String errorDescription) { - - PushAuthErrorResponse pushAuthErrorResponse = new PushAuthErrorResponse(); - JSONObject errorResponse = new JSONObject(); - errorResponse.put(ERROR_DESCRIPTION, errorDescription); - errorResponse.put(ERROR, errorCode); - pushAuthErrorResponse.setPayload(errorResponse); - pushAuthErrorResponse.setHttpStatusCode(httpStatusCode); - - return pushAuthErrorResponse; - } - - @Generated(message = "Excluding from code coverage since it requires a service call") - protected OAuth2ClientValidationResponseDTO getClientValidationInfo(JSONObject requestBodyJson) { - - return new OAuth2Service().validateClientInfo(requestBodyJson.getAsString(PushAuthRequestConstants.CLIENT_ID), - requestBodyJson.getAsString(PushAuthRequestConstants.REDIRECT_URI)); - } - - @Generated(message = "Excluding from code coverage since it requires a service call") - protected void validateSignature(String requestObjectString, JSONObject requestBodyJson) - throws PushAuthRequestValidatorException { - - PushAuthRequestValidatorUtils.validateSignature(requestObjectString, requestBodyJson); - } - - @Generated(message = "Excluding from code coverage since it requires a service call") - protected void validateAudience(JSONObject requestBodyJson) - throws PushAuthRequestValidatorException { - - PushAuthRequestValidatorUtils.validateAudience(requestBodyJson); - } - - protected void validatePKCEParameters(JSONObject requestBodyJson) - throws PushAuthRequestValidatorException { - - PushAuthRequestValidatorUtils.validatePKCEParameters(requestBodyJson); - } - - protected void validateResponseType(JSONObject requestBodyJson) - throws PushAuthRequestValidatorException { - - PushAuthRequestValidatorUtils.validateResponseType(requestBodyJson); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/constants/PushAuthRequestConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/constants/PushAuthRequestConstants.java deleted file mode 100644 index 55fd9e4c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/constants/PushAuthRequestConstants.java +++ /dev/null @@ -1,58 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.constants; - -/** - * Constant class for push auth request. - */ -public class PushAuthRequestConstants { - - // PAR request parameters - public static final String REQUEST = "request"; - public static final String REQUEST_URI = "request_uri"; - public static final String RESPONSE_TYPE = "response_type"; - public static final String NONCE = "nonce"; - public static final String SCOPE = "scope"; - public static final String CODE_CHALLENGE = "code_challenge"; - public static final String CODE_CHALLENGE_METHOD = "code_challenge_method"; - public static final String CLIENT_ID = "client_id"; - public static final String REDIRECT_URI = "redirect_uri"; - - // error constants - public static final String SERVER_ERROR = "server_error"; - public static final String INVALID_REQUEST = "invalid_request"; - public static final String INVALID_REQUEST_OBJECT = "invalid_request_object"; - - // custom jwt body constants - public static final String DECODED_JWT_BODY = "decodedJWTBody"; - public static final String DECODED_JWT_HEADER = "decodedJWTHeader"; - - // jwt constants - public static final String ALG_HEADER = "alg"; - public static final String AUDIENCE = "aud"; - public static final String ISSUER = "iss"; - public static final String EXPIRY = "exp"; - public static final long ONE_HOUR_IN_MILLIS = 3600000; - public static final String NOT_BEFORE = "nbf"; - public static final String ALG_HEADER_NONE = "none"; - - // jwt parts - public static final String BODY = "body"; - public static final String HEADER = "head"; -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/exception/PushAuthRequestValidatorException.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/exception/PushAuthRequestValidatorException.java deleted file mode 100644 index bcc08f33..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/exception/PushAuthRequestValidatorException.java +++ /dev/null @@ -1,79 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.exception; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; - -/** - * PAR validation exception. - */ -public class PushAuthRequestValidatorException extends OpenBankingException { - - private String errorDescription; - private String errorCode; - private int httpStatusCode; - - public int getHttpStatusCode() { - - return httpStatusCode; - } - - public void setHttpStatusCode(int httpStatusCode) { - - this.httpStatusCode = httpStatusCode; - } - - public String getErrorDescription() { - - return errorDescription; - } - - public void setErrorDescription(String errorDescription) { - - this.errorDescription = errorDescription; - } - - public String getErrorCode() { - - return errorCode; - } - - public void setErrorCode(String errorCode) { - - this.errorCode = errorCode; - } - - public PushAuthRequestValidatorException(int httpStatusCode, String errorCode, String errorDescription, - Throwable e) { - - super(errorDescription, e); - this.errorDescription = errorDescription; - this.errorCode = errorCode; - this.httpStatusCode = httpStatusCode; - - } - - public PushAuthRequestValidatorException(int httpStatusCode, String errorCode, String errorDescription) { - - super(errorDescription); - this.errorDescription = errorDescription; - this.errorCode = errorCode; - this.httpStatusCode = httpStatusCode; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/model/PushAuthErrorResponse.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/model/PushAuthErrorResponse.java deleted file mode 100644 index 37a207e3..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/model/PushAuthErrorResponse.java +++ /dev/null @@ -1,48 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.model; - -import net.minidev.json.JSONObject; - -/** - * Model class for push authorisation error. - */ -public class PushAuthErrorResponse { - - private int httpStatusCode = 0; - private JSONObject payload = null; - - public int getHttpStatusCode() { - - return httpStatusCode; - } - public void setHttpStatusCode(int httpStatusCode) { - - this.httpStatusCode = httpStatusCode; - } - - public JSONObject getPayload() { - - return payload; - } - public void setPayload(JSONObject payload) { - - this.payload = payload; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/util/PushAuthRequestValidatorUtils.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/util/PushAuthRequestValidatorUtils.java deleted file mode 100644 index 0513bf17..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/util/PushAuthRequestValidatorUtils.java +++ /dev/null @@ -1,648 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.util; - -import com.nimbusds.jose.JOSEException; -import com.nimbusds.jose.JWEObject; -import com.nimbusds.jose.JWSHeader; -import com.nimbusds.jose.crypto.RSADecrypter; -import com.nimbusds.jose.crypto.RSASSAVerifier; -import com.nimbusds.jwt.EncryptedJWT; -import com.nimbusds.jwt.PlainJWT; -import com.nimbusds.jwt.SignedJWT; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.constants.PushAuthRequestConstants; -import com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.exception.PushAuthRequestValidatorException; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonHelper; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.HttpStatus; -import org.wso2.carbon.identity.application.common.model.FederatedAuthenticatorConfig; -import org.wso2.carbon.identity.application.common.model.IdentityProvider; -import org.wso2.carbon.identity.application.common.model.Property; -import org.wso2.carbon.identity.application.common.model.ServiceProviderProperty; -import org.wso2.carbon.identity.application.common.util.IdentityApplicationConstants; -import org.wso2.carbon.identity.application.common.util.IdentityApplicationManagementUtil; -import org.wso2.carbon.identity.core.util.IdentityUtil; -import org.wso2.carbon.identity.oauth.common.OAuth2ErrorCodes; -import org.wso2.carbon.identity.oauth.common.OAuthConstants; -import org.wso2.carbon.identity.oauth.common.exception.InvalidOAuthClientException; -import org.wso2.carbon.identity.oauth.config.OAuthServerConfiguration; -import org.wso2.carbon.identity.oauth.dao.OAuthAppDO; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.util.OAuth2Util; -import org.wso2.carbon.identity.oauth2.validators.jwt.JWKSBasedJWTValidator; -import org.wso2.carbon.idp.mgt.IdentityProviderManagementException; -import org.wso2.carbon.idp.mgt.IdentityProviderManager; -import org.wso2.carbon.utils.multitenancy.MultitenantConstants; - -import java.net.MalformedURLException; -import java.net.URL; -import java.security.Key; -import java.security.PublicKey; -import java.security.cert.Certificate; -import java.security.interfaces.RSAPrivateKey; -import java.security.interfaces.RSAPublicKey; -import java.text.ParseException; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Date; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import static org.wso2.carbon.identity.oauth2.util.OAuth2Util.getX509CertOfOAuthApp; -import static org.wso2.carbon.identity.openidconnect.model.Constants.JWKS_URI; -import static org.wso2.carbon.identity.openidconnect.model.Constants.JWT_PART_DELIMITER; -import static org.wso2.carbon.identity.openidconnect.model.Constants.NUMBER_OF_PARTS_IN_JWS; - -/** - * The utility functions required for the push authorization module. - */ -public class PushAuthRequestValidatorUtils { - - private static Log log = LogFactory.getLog(PushAuthRequestValidatorUtils.class); - private static final String OIDC_IDP_ENTITY_ID = "IdPEntityId"; - private static final String OAUTH2_TOKEN_EP_URL = "OAuth2TokenEPUrl"; - private static final String OIDC_ID_TOKEN_ISSUER_ID = "OAuth.OpenIDConnect.IDTokenIssuerID"; - private static final ArrayList ALLOWED_FORM_BODY_PARAMS = new ArrayList() { - { - add("client_id"); - add("client_assertion"); - add("client_assertion_type"); - } - }; - - /** - * Check whether push auth request only contains client_id, client_assertion and client_assertion_type when request - * parameter is present. - */ - public static void validateRequestFormBody(Map parameters) - throws PushAuthRequestValidatorException { - - for (Map.Entry parameter: parameters.entrySet()) { - if (!PushAuthRequestConstants.REQUEST.equalsIgnoreCase(parameter.getKey()) && - !ALLOWED_FORM_BODY_PARAMS.contains(parameter.getKey())) { - log.error("Invalid parameters found in the request"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST, "Invalid parameters found in the request"); - } - } - } - - /** - * Check whether the algorithm used to sign the request object is valid. - */ - public static void validateSignatureAlgorithm(Object algorithm) throws PushAuthRequestValidatorException { - - boolean isValid = false; - if (algorithm != null && StringUtils.isNotBlank((String) algorithm)) { - List allowedAlgorithmsList = new ArrayList<>(); - Object allowedAlgorithms = IdentityExtensionsDataHolder.getInstance() - .getConfigurationMap().get(OpenBankingConstants.SIGNATURE_ALGORITHMS); - if (allowedAlgorithms instanceof List) { - allowedAlgorithmsList = (List) allowedAlgorithms; - } else { - allowedAlgorithmsList.add(allowedAlgorithms.toString()); - } - isValid = allowedAlgorithmsList.isEmpty() || allowedAlgorithmsList.contains(algorithm); - } - if (!isValid) { - log.error("Invalid request object signing algorithm"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST_OBJECT, - "Invalid request object signing algorithm"); - } - - } - - /** - * Checks whether the given authentication flow requires nonce as a mandatory parameter. - */ - public static boolean isNonceMandatory(String responseType) { - - // nonce parameter is required for the OIDC hybrid flow and implicit flow grant types requesting ID_TOKEN. - return Arrays.stream(responseType.split("\\s+")).anyMatch(OAuthConstants.ID_TOKEN::equals); - } - - /** - * Validates the nonce parameter as mandatory. - */ - public static void validateNonceParameter(JSONObject requestBodyJson) throws PushAuthRequestValidatorException { - - if (StringUtils.isNotBlank(requestBodyJson.getAsString(PushAuthRequestConstants.RESPONSE_TYPE))) { - if (isNonceMandatory(requestBodyJson.getAsString(PushAuthRequestConstants.RESPONSE_TYPE)) && - requestBodyJson.getAsString(PushAuthRequestConstants.NONCE) == null) { - log.error("Invalid Nonce parameter in the request"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST, "Invalid Nonce parameter in the request"); - } - } - } - - /** - * Validates the mandatory PKCE parameters. - */ - public static void validatePKCEParameters(JSONObject requestBodyJson) throws PushAuthRequestValidatorException { - - if (StringUtils.isEmpty(requestBodyJson.getAsString(PushAuthRequestConstants.CODE_CHALLENGE))) { - log.error("Mandatory parameter code_challenge, not found in the request"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST_OBJECT, - "Mandatory parameter code_challenge, not found in the request"); - } - - if (StringUtils.isEmpty(requestBodyJson.getAsString(PushAuthRequestConstants.CODE_CHALLENGE_METHOD))) { - log.error("Mandatory parameter code_challenge_method, not found in the request"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST_OBJECT, - "Mandatory parameter code_challenge_method, not found in the request"); - } - } - - /** - * Validates the mandatory response_type parameter. - */ - public static void validateResponseType(JSONObject requestBodyJson) throws PushAuthRequestValidatorException { - - if (StringUtils.isEmpty(requestBodyJson.getAsString(PushAuthRequestConstants.RESPONSE_TYPE))) { - log.error("Mandatory parameter response_type, not found in the request"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST_OBJECT, - "Mandatory parameter response_type, not found in the request"); - } - } - - /** - * Validates the redirect URI parameter to verify if the parameter is available and not null. - */ - public static void validateRedirectUri(JSONObject requestBodyJson) throws PushAuthRequestValidatorException { - - if (StringUtils.isBlank(requestBodyJson.getAsString(PushAuthRequestConstants.REDIRECT_URI))) { - log.error("Mandatory parameter redirect_uri, not found in the request"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST, - "Mandatory parameter redirect_uri, not found in the request"); - } - } - - /** - * Validates the scope parameter to verify only allowed scopes for the application are passed in the request. - */ - public static void validateScope(JSONObject requestBodyJson) throws PushAuthRequestValidatorException { - - if (StringUtils.isNotBlank(requestBodyJson.getAsString(PushAuthRequestConstants.SCOPE))) { - - List requestedScopes = Arrays.asList(requestBodyJson.getAsString(PushAuthRequestConstants.SCOPE) - .split("\\s+")); - - List allowedScopes; - try { - allowedScopes = Arrays.asList(new IdentityCommonHelper() - .getAppPropertyFromSPMetaData(requestBodyJson.getAsString(PushAuthRequestConstants.CLIENT_ID), - IdentityCommonConstants.SCOPE).split("\\s+")); - } catch (OpenBankingException e) { - log.error("Error while retrieving sp meta data", e); - throw new PushAuthRequestValidatorException(HttpStatus.SC_INTERNAL_SERVER_ERROR, - PushAuthRequestConstants.SERVER_ERROR, "Error while retrieving sp meta data", e); - } - - for (String scope : requestedScopes) { - if (!allowedScopes.contains(scope)) { - log.error("Invalid scopes in the request"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST, "Invalid scopes in the request"); - } - } - } else { - log.error("Mandatory parameter scope, not found in the request"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST, - "Mandatory parameter scope, not found in the request"); - } - } - - /** - * Validates the audience parameter to check whether it matches any supported audience value. - */ - @Generated(message = "Excluding from code coverage since it requires a service call") - public static void validateAudience(JSONObject requestBodyJson) throws PushAuthRequestValidatorException { - - String clientId = requestBodyJson.getAsString(PushAuthRequestConstants.CLIENT_ID); - Object audValue = requestBodyJson.get(PushAuthRequestConstants.AUDIENCE); - boolean isValid = false; - - if (audValue == null) { - log.error("aud parameter is missing in the request object"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST_OBJECT, "aud parameter is missing in the request object"); - } - - List validAudUrls = getAllowedPARAudienceValues(getSPTenantDomainFromClientId(clientId)); - - if (audValue instanceof String) { - // If the aud value is a string, check whether it is equal to the allowed audience value. - isValid = validAudUrls.contains(audValue); - } else if (audValue instanceof JSONArray) { - // If the aud value is an array, check whether it is equal to one of the allowed audience values. - JSONArray audArray = (JSONArray) audValue; - for (Object aud : audArray) { - if (validAudUrls.contains(aud)) { - isValid = true; - break; - } - } - } - - if (!isValid) { - log.error("Invalid audience value in the request"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST_OBJECT, "Invalid audience value in the request"); - } - } - - /** - * Validates the issuer parameter to check whether its similar to the client id. - */ - public static void validateIssuer(JSONObject requestBodyJson) throws PushAuthRequestValidatorException { - - String issuer = requestBodyJson.getAsString(PushAuthRequestConstants.ISSUER); - String clientId = requestBodyJson.getAsString(PushAuthRequestConstants.CLIENT_ID); - boolean isValid = false; - - if (StringUtils.isNotBlank(issuer) && StringUtils.isNotBlank(clientId)) { - isValid = issuer.equals(clientId); - } - if (!isValid) { - log.error("Invalid issuer in the request"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST_OBJECT, "Invalid issuer in the request"); - } - } - - - /** - * Validates the expiration status of the request object. - */ - public static void validateExpirationTime(JSONObject requestBodyJson) throws PushAuthRequestValidatorException { - - String exp = requestBodyJson.getAsString(PushAuthRequestConstants.EXPIRY); - - if (StringUtils.isNotBlank(exp)) { - Date expirationTime = new Date(Integer.parseInt(exp) * 1000L); - long timeStampSkewMillis = OAuthServerConfiguration.getInstance().getTimeStampSkewInSeconds() * 1000; - long expirationTimeInMillis = expirationTime.getTime(); - long currentTimeInMillis = System.currentTimeMillis(); - // exp parameter should not be over 1 hour in the future. - if ((expirationTimeInMillis - (currentTimeInMillis + timeStampSkewMillis)) > - PushAuthRequestConstants.ONE_HOUR_IN_MILLIS) { - log.error("exp parameter in the request object is over 1 hour in the future"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST_OBJECT, "exp parameter in the request object " + - "is over 1 hour in the future"); - } - // exp parameter should not be in the past. - if ((currentTimeInMillis + timeStampSkewMillis) > expirationTimeInMillis) { - log.error("Request object expired"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST, "Request object expired"); - } - } else { - log.error("exp parameter is missing in the request object"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST_OBJECT, "exp parameter is missing in the request object"); - } - } - - /** - * Validates the nbf claim in the request object. - */ - public static void validateNotBeforeClaim(JSONObject requestBodyJson) throws PushAuthRequestValidatorException { - - String nbf = requestBodyJson.getAsString(PushAuthRequestConstants.NOT_BEFORE); - - if (StringUtils.isNotBlank(nbf)) { - Date notBeforeTime = new Date(Integer.parseInt(nbf) * 1000L); - long timeStampSkewMillis = OAuthServerConfiguration.getInstance().getTimeStampSkewInSeconds() * 1000; - long notBeforeTimeInMillis = notBeforeTime.getTime(); - long currentTimeInMillis = System.currentTimeMillis(); - //request object should be used on or after nbf value. - if ((currentTimeInMillis + timeStampSkewMillis) < notBeforeTimeInMillis) { - log.error("Request object is not valid yet"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST_OBJECT, "Request object is not valid yet"); - } - // nbf parameter should not be over 1 hour in the past. - if (((currentTimeInMillis + timeStampSkewMillis) - notBeforeTimeInMillis) > - PushAuthRequestConstants.ONE_HOUR_IN_MILLIS) { - log.error("nbf parameter in the request object is over 1 hour in the past"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST_OBJECT, "nbf parameter in the request object " + - "is over 1 hour in the past"); - } - } else { - log.error("nbf parameter is missing in the request object"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST_OBJECT, - "nbf parameter is missing in the request object"); - } - } - - /** - * Validates signature of the request object. - */ - @Generated(message = "Excluding from code coverage since it requires several service calls") - public static void validateSignature(String requestObject, JSONObject requestBodyJson) - throws PushAuthRequestValidatorException { - - String jwksUri = null; - ServiceProviderProperty[] spProperties = null; - - // Get Service provider properties - try { - spProperties = OAuth2Util.getServiceProvider(requestBodyJson - .getAsString(PushAuthRequestConstants.CLIENT_ID)).getSpProperties(); - } catch (IdentityOAuth2Exception exception) { - log.error("Unable to extract Service Provider Properties", exception); - throw new PushAuthRequestValidatorException(HttpStatus.SC_INTERNAL_SERVER_ERROR, - PushAuthRequestConstants.SERVER_ERROR, exception.getMessage(), exception); - } - - // Extract JWKS Uri from properties - if (spProperties != null) { - for (ServiceProviderProperty spProperty : spProperties) { - if (JWKS_URI.equals(spProperty.getName())) { - jwksUri = spProperty.getValue(); - } - } - } - if (log.isDebugEnabled()) { - log.debug("Retrieved JWKS URI: " + jwksUri); - } - - SignedJWT jwt; - - try { - jwt = SignedJWT.parse(requestObject); - } catch (ParseException exception) { - log.error("Unable to parse JWT object", exception); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST, exception.getMessage(), exception); - } - - boolean isVerified = false; - - if (StringUtils.isBlank(jwksUri)) { - log.debug("Validating from certificate"); - String tenantDomain = getSPTenantDomainFromClientId(requestBodyJson - .getAsString(PushAuthRequestConstants.CLIENT_ID)); - - // Validate from Certificate Content - Certificate certificate; - try { - certificate = getX509CertOfOAuthApp(requestBodyJson - .getAsString(PushAuthRequestConstants.CLIENT_ID), tenantDomain); - } catch (IdentityOAuth2Exception exception) { - log.error("Unable to get certificate from app", exception); - throw new PushAuthRequestValidatorException(HttpStatus.SC_INTERNAL_SERVER_ERROR, - PushAuthRequestConstants.SERVER_ERROR, exception.getMessage(), exception); - } - - isVerified = isSignatureVerified(jwt, certificate); - } else { - log.debug("Validating from JWKS URI"); - - // Validate from JWKS Uri - String alg = jwt.getHeader().getAlgorithm().getName(); - Map options = new HashMap<>(); - try { - isVerified = new JWKSBasedJWTValidator().validateSignature(jwt.getParsedString(), jwksUri, - alg, options); - } catch (IdentityOAuth2Exception exception) { - log.error("Unable to validate JWT using JWKS URL", exception); - String errorMessage = getCustomSignatureValidationErrorMessage(exception); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST_OBJECT, errorMessage, exception); - } - } - if (!isVerified) { - log.error("Request object signature validation failed"); - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, - PushAuthRequestConstants.INVALID_REQUEST, - "Request object signature validation failed"); - } - } - - /** - * Validate signature of a sign JWT against a given certificate. - */ - @Generated(message = "Excluding from code coverage since it requires several service calls") - private static boolean isSignatureVerified(SignedJWT signedJWT, Certificate x509Certificate) { - JWSHeader header = signedJWT.getHeader(); - if (x509Certificate == null) { - if (log.isDebugEnabled()) { - log.debug("Unable to locate certificate for JWT " + header.toString()); - } - return false; - } else { - String alg = signedJWT.getHeader().getAlgorithm().getName(); - if (log.isDebugEnabled()) { - log.debug("Signature Algorithm found in the JWT Header: " + alg); - } - - // allowed RS and PS for the moment - if (alg.indexOf("RS") != 0 && alg.indexOf("PS") != 0) { - if (log.isDebugEnabled()) { - log.debug("Signature Algorithm not supported yet : " + alg); - } - return false; - } else { - PublicKey publicKey = x509Certificate.getPublicKey(); - if (publicKey instanceof RSAPublicKey) { - RSASSAVerifier verifier = new RSASSAVerifier((RSAPublicKey) publicKey); - - try { - return signedJWT.verify(verifier); - } catch (JOSEException e) { - if (log.isDebugEnabled()) { - log.debug("Unable to verify the signature of the request object: " + signedJWT.serialize()); - } - return false; - } - } else { - log.debug("Public key is not an RSA public key."); - return false; - } - } - } - } - - /** - * Return the alias of the resident IDP (issuer of Authorization Server), PAR Endpoint and Token Endpoint - * to validate the audience value of the PAR Request Object. - */ - @Generated(message = "Excluding from code coverage since it requires several service calls") - private static List getAllowedPARAudienceValues(String tenantDomain) - throws PushAuthRequestValidatorException { - - List validAudUrls = new ArrayList<>(); - String residentIdpAlias = StringUtils.EMPTY; - IdentityProvider residentIdP; - try { - residentIdP = IdentityProviderManager.getInstance().getResidentIdP(tenantDomain); - FederatedAuthenticatorConfig oidcFedAuthn = IdentityApplicationManagementUtil - .getFederatedAuthenticator(residentIdP.getFederatedAuthenticatorConfigs(), - IdentityApplicationConstants.Authenticator.OIDC.NAME); - - Property idPEntityIdProperty = - IdentityApplicationManagementUtil.getProperty(oidcFedAuthn.getProperties(), OIDC_IDP_ENTITY_ID); - if (idPEntityIdProperty != null) { - residentIdpAlias = idPEntityIdProperty.getValue(); - if (log.isDebugEnabled()) { - log.debug("Found IdPEntityID: " + residentIdpAlias + " for tenantDomain: " + tenantDomain); - } - } - - Property oAuth2TokenEPUrlProperty = - IdentityApplicationManagementUtil.getProperty(oidcFedAuthn.getProperties(), OAUTH2_TOKEN_EP_URL); - if (oAuth2TokenEPUrlProperty != null) { - // add Token EP Url as a valid "aud" value - validAudUrls.add(oAuth2TokenEPUrlProperty.getValue()); - if (log.isDebugEnabled()) { - log.debug("Found OAuth2TokenEPUrl: " + oAuth2TokenEPUrlProperty.getValue() + - " for tenantDomain: " + tenantDomain); - } - } - } catch (IdentityProviderManagementException e) { - log.error("Error while loading OAuth2TokenEPUrl of the resident IDP of tenant:" + tenantDomain, e); - throw new PushAuthRequestValidatorException(HttpStatus.SC_INTERNAL_SERVER_ERROR, - OAuth2ErrorCodes.SERVER_ERROR, "Server Error while validating audience " + - "of Request Object.", e); - } - - if (StringUtils.isEmpty(residentIdpAlias)) { - residentIdpAlias = IdentityUtil.getProperty(OIDC_ID_TOKEN_ISSUER_ID); - if (StringUtils.isNotEmpty(residentIdpAlias)) { - if (log.isDebugEnabled()) { - log.debug("'IdPEntityID' property was empty for tenantDomain: " + tenantDomain + ". Using " + - "OIDC IDToken Issuer value: " + residentIdpAlias + " as alias to identify Resident IDP."); - } - } - } - - // add IdPEntityID or the "issuer" as a valid "aud" value - validAudUrls.add(residentIdpAlias); - - try { - URL residentIdPUrl = new URL(residentIdpAlias); - // derive PAR EP URL from the residentIdP base URL - URL parEpUrl = new URL(residentIdPUrl, IdentityCommonConstants.PAR_ENDPOINT); - // add PAR EP URL as a valid "aud" value - validAudUrls.add(parEpUrl.toString()); - } catch (MalformedURLException exception) { - log.error("Error occurred while deriving PAR endpoint URL.", exception); - throw new PushAuthRequestValidatorException(HttpStatus.SC_INTERNAL_SERVER_ERROR, - OAuth2ErrorCodes.SERVER_ERROR, "Server Error while deriving PAR endpoint URL.", exception); - } - - return validAudUrls; - } - - /** - * Return the tenant domain for a given client. - */ - public static String getSPTenantDomainFromClientId(String clientId) { - - try { - OAuthAppDO oAuthAppDO = OAuth2Util.getAppInformationByClientId(clientId); - return OAuth2Util.getTenantDomainOfOauthApp(oAuthAppDO); - } catch (IdentityOAuth2Exception | InvalidOAuthClientException e) { - return MultitenantConstants.SUPER_TENANT_DOMAIN_NAME; - } - } - - /** - * Decrypt an encrypted request object. - */ - public static String decrypt(String requestObject, String clientId) throws PushAuthRequestValidatorException { - EncryptedJWT encryptedJWT; - try { - encryptedJWT = EncryptedJWT.parse(requestObject); - RSAPrivateKey rsaPrivateKey = getRSAPrivateKey(clientId); - RSADecrypter decrypter = new RSADecrypter(rsaPrivateKey); - encryptedJWT.decrypt(decrypter); - - JWEObject jweObject = JWEObject.parse(requestObject); - jweObject.decrypt(decrypter); - - if (jweObject.getPayload() != null && jweObject.getPayload().toString() - .split(JWT_PART_DELIMITER).length == NUMBER_OF_PARTS_IN_JWS) { - return jweObject.getPayload().toString(); - } else { - return new PlainJWT(encryptedJWT.getJWTClaimsSet()).serialize(); - } - - } catch (JOSEException | IdentityOAuth2Exception | ParseException e) { - String errorMessage = "Failed to decrypt Request Object"; - if (log.isDebugEnabled()) { - log.debug(errorMessage + " from " + requestObject, e); - } - throw new PushAuthRequestValidatorException(HttpStatus.SC_BAD_REQUEST, "invalid_request", errorMessage, e); - } - } - - /** - * Get RSA private key from tenant domain for registered client. - */ - private static RSAPrivateKey getRSAPrivateKey(String clientId) throws IdentityOAuth2Exception { - - String tenantDomain = PushAuthRequestValidatorUtils.getSPTenantDomainFromClientId(clientId); - int tenantId = OAuth2Util.getTenantId(tenantDomain); - Key key = OAuth2Util.getPrivateKey(tenantDomain, tenantId); - return (RSAPrivateKey) key; - } - - /** - * Get custom error message for signature validation errors. - */ - private static String getCustomSignatureValidationErrorMessage(IdentityOAuth2Exception exception) { - - String errorMessage = exception.getCause().getMessage(); - if (StringUtils.isEmpty(errorMessage)) { - return exception.getMessage(); - } - - if (errorMessage.equalsIgnoreCase("JWT before use time")) { - return "Invalid not before time. 'nbf' must be a past value."; - } - - if (errorMessage.equalsIgnoreCase("Expired JWT")) { - return "Invalid expiry time. 'exp' claim must be a future value."; - } - - return errorMessage; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/sp/metadata/extension/SPMetadataFilter.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/sp/metadata/extension/SPMetadataFilter.java deleted file mode 100644 index a3b26111..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/sp/metadata/extension/SPMetadataFilter.java +++ /dev/null @@ -1,35 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.sp.metadata.extension; - -import java.util.Map; - -/** - * The interface to define how the service provider metadata filter extension should be implemented. - */ -public interface SPMetadataFilter { - - /** - * Filter logic for metadata map. - * @param metadata - * @return - */ - Map filter(Map metadata); - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/sp/metadata/extension/impl/DefaultSPMetadataFilter.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/sp/metadata/extension/impl/DefaultSPMetadataFilter.java deleted file mode 100644 index 6d539fbc..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/sp/metadata/extension/impl/DefaultSPMetadataFilter.java +++ /dev/null @@ -1,51 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.sp.metadata.extension.impl; - -import com.google.common.collect.ImmutableMap; -import com.wso2.openbanking.accelerator.identity.sp.metadata.extension.SPMetadataFilter; - -import java.util.Map; -import java.util.stream.Collectors; - -/** - * Impl of the SPMetadataFilterInterface. - */ -public class DefaultSPMetadataFilter implements SPMetadataFilter { - - @Override - public Map filter(Map metadata) { - - // Ex: ("client_name", "software_client_name"), property will read as "software_client_name" from metadata - // and put as "client_name" - Map propertiesVsMappingName = ImmutableMap.of( - "client_name", "client_name", - "software_client_name", "software_client_name", - "software_id", "software_id", - "logo_uri", "logo_uri", - "org_name", "org_name" - ); - - return propertiesVsMappingName.entrySet().stream() - .collect(Collectors.toMap( - Map.Entry::getKey, - e -> metadata.getOrDefault(e.getValue(), "") - )); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/DefaultTokenFilter.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/DefaultTokenFilter.java deleted file mode 100644 index fe85498d..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/DefaultTokenFilter.java +++ /dev/null @@ -1,88 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.token; - -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.json.JSONObject; - -import java.io.IOException; -import java.io.OutputStream; -import java.nio.charset.StandardCharsets; - -import javax.servlet.ServletException; -import javax.servlet.ServletRequest; -import javax.servlet.ServletResponse; -import javax.servlet.http.HttpServletResponse; -import javax.ws.rs.core.MediaType; - -/** - * Default token filter class to represent the abstract implementation. - */ -public class DefaultTokenFilter { - - private static final Log log = LogFactory.getLog(DefaultTokenFilter.class); - - /** - * Handle filter request. - * - * @param request - * @return ServletRequest - * @throws ServletException - */ - public ServletRequest handleFilterRequest(ServletRequest request) throws ServletException { - return request; - } - - /** - * Handle filter response. - * - * @param response - * @return ServletResponse - * @throws ServletException - */ - public ServletResponse handleFilterResponse(ServletResponse response) throws ServletException { - - return response; - } - - /** - * Respond when there is a failure in filter validation. - * - * @param response HTTP servlet response object - * @param status HTTP status code - * @param error error - * @param errorMessage error description - * @throws IOException - */ - public void handleValidationFailure(HttpServletResponse response, int status, String error, String errorMessage) - throws IOException { - - JSONObject errorJSON = new JSONObject(); - errorJSON.put(IdentityCommonConstants.OAUTH_ERROR, error); - errorJSON.put(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION, errorMessage); - - try (OutputStream outputStream = response.getOutputStream()) { - response.setStatus(status); - response.setContentType(MediaType.APPLICATION_JSON); - outputStream.write(errorJSON.toString().getBytes(StandardCharsets.UTF_8)); - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/TokenFilter.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/TokenFilter.java deleted file mode 100644 index f744ad05..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/TokenFilter.java +++ /dev/null @@ -1,298 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.token; - -import com.nimbusds.jwt.SignedJWT; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.CertificateUtils; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.token.util.TokenFilterException; -import com.wso2.openbanking.accelerator.identity.token.validators.OBIdentityFilterValidator; -import com.wso2.openbanking.accelerator.identity.token.wrapper.RequestWrapper; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonHelper; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.io.IOException; -import java.io.UnsupportedEncodingException; -import java.net.URLDecoder; -import java.nio.charset.StandardCharsets; -import java.security.cert.CertificateEncodingException; -import java.security.cert.X509Certificate; -import java.text.ParseException; -import java.util.ArrayList; -import java.util.Base64; -import java.util.List; -import java.util.Optional; - -import javax.servlet.Filter; -import javax.servlet.FilterChain; -import javax.servlet.FilterConfig; -import javax.servlet.ServletContext; -import javax.servlet.ServletException; -import javax.servlet.ServletRequest; -import javax.servlet.ServletResponse; -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; - -/** - * Filter engaged for /token request. - */ -public class TokenFilter implements Filter { - - private static final Log log = LogFactory.getLog(TokenFilter.class); - private static DefaultTokenFilter defaultTokenFilter; - private String clientId = null; - private static List validators = new ArrayList<>(); - - private static final String BASIC_AUTH_ERROR_MSG = "Unable to find client id in the request. " + - "Invalid Authorization header found."; - - @Generated(message = "Ignoring because it's a the init method") - @Override - public void init(FilterConfig filterConfig) { - - ServletContext context = filterConfig.getServletContext(); - context.log("TokenFilter initialized"); - } - - @Override - public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) - throws IOException, ServletException { - - try { - clientId = this.extractClientId(request); - } catch (TokenFilterException e) { - getDefaultTokenFilter().handleValidationFailure((HttpServletResponse) response, e.getErrorCode(), - e.getMessage(), e.getErrorDescription()); - return; - } - - try { - request = cleanClientCertificateAndAppendTransportHeader(request); - if (IdentityCommonUtil.getRegulatoryFromSPMetaData(clientId)) { - request = appendTransportHeader(request, response); - request = getDefaultTokenFilter().handleFilterRequest(request); - for (OBIdentityFilterValidator validator : getValidators()) { - validator.validate(request, clientId); - } - response = getDefaultTokenFilter().handleFilterResponse(response); - } - chain.doFilter(request, response); - } catch (TokenFilterException e) { - getDefaultTokenFilter().handleValidationFailure((HttpServletResponse) response, - e.getErrorCode(), e.getMessage(), e.getErrorDescription()); - } catch (CertificateEncodingException e) { - throw new ServletException("Certificate not valid", e); - } catch (OpenBankingException e) { - if (e.getMessage().contains("Error occurred while retrieving OAuth2 application data")) { - getDefaultTokenFilter().handleValidationFailure((HttpServletResponse) response, - HttpServletResponse.SC_INTERNAL_SERVER_ERROR, IdentityCommonConstants - .OAUTH2_INTERNAL_SERVER_ERROR, "OAuth2 application data retrieval failed." - + e.getMessage()); - } else { - getDefaultTokenFilter().handleValidationFailure((HttpServletResponse) response, - HttpServletResponse.SC_BAD_REQUEST, IdentityCommonConstants.OAUTH2_INVALID_REQUEST_MESSAGE, - "Service provider metadata retrieval failed. " + e.getMessage()); - } - } - } - - /** - * Append the transport header to the request. - * - * @param request - * @param response - * @return ServletRequest - * @throws ServletException - */ - private ServletRequest appendTransportHeader(ServletRequest request, ServletResponse response) throws - ServletException, IOException, CertificateEncodingException { - - if (request instanceof HttpServletRequest) { - Object certAttribute = request.getAttribute(IdentityCommonConstants.JAVAX_SERVLET_REQUEST_CERTIFICATE); - String x509Certificate = ((HttpServletRequest) request).getHeader(IdentityCommonUtil.getMTLSAuthHeader()); - if (new IdentityCommonHelper().isTransportCertAsHeaderEnabled() && x509Certificate != null) { - return request; - } else if (certAttribute != null) { - RequestWrapper requestWrapper = new RequestWrapper((HttpServletRequest) request); - X509Certificate certificate = IdentityCommonUtil.getCertificateFromAttribute(certAttribute); - requestWrapper.setHeader(IdentityCommonUtil.getMTLSAuthHeader(), - new IdentityCommonHelper().encodeCertificateContent(certificate)); - return requestWrapper; - } else { - getDefaultTokenFilter().handleValidationFailure((HttpServletResponse) response, - HttpServletResponse.SC_BAD_REQUEST, IdentityCommonConstants.OAUTH2_INVALID_REQUEST_MESSAGE, - "Transport certificate not found in the request"); - } - } else { - throw new ServletException("Error occurred when handling the request, passed request is not a " + - "HttpServletRequest"); - } - return request; - } - - /** - * Invoked after all execution of the filter has completed and the filter is being taken out of service. - */ - @Generated(message = "Ignoring because it's a clean up code") - @Override - public void destroy() { - // No special cleanup is required in this filter. - } - - /** - * @return Token filter - */ - @Generated(message = "Ignoring because the method is reading the configuration") - public DefaultTokenFilter getDefaultTokenFilter() { - - return defaultTokenFilter; - } - - /** - * Extracts the client id from the request parameter or from the assertion. - * - * @param request servlet request containing the request data - * @return clientId - * @throws ParseException - */ - private String extractClientId(ServletRequest request) throws TokenFilterException { - - try { - Optional signedObject = - Optional.ofNullable(request.getParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION)); - Optional clientIdAsReqParam = - Optional.ofNullable(request.getParameter(IdentityCommonConstants.CLIENT_ID)); - if (signedObject.isPresent()) { - SignedJWT signedJWT = SignedJWT.parse(signedObject.get()); - return signedJWT.getJWTClaimsSet().getIssuer(); - } else if (clientIdAsReqParam.isPresent()) { - return clientIdAsReqParam.get(); - } else if (((HttpServletRequest) request).getHeader("Authorization") != null) { - // This added condition will only affect the requests with Basic Authentication and the others will be - // handled by the above conditions as previously. - String authorizationHeader = ((HttpServletRequest) request).getHeader("Authorization"); - if (authorizationHeader.split(" ").length == 2) { - String authToken = ((HttpServletRequest) request).getHeader("Authorization").split(" ")[1]; - byte[] decodedBytes = Base64.getUrlDecoder().decode(authToken.getBytes(StandardCharsets.UTF_8)); - String decodedAuthToken = new String(decodedBytes, StandardCharsets.UTF_8); - if (decodedAuthToken.split(":").length == 2) { - return decodedAuthToken.split(":")[0]; - } else { - log.error(BASIC_AUTH_ERROR_MSG); - throw new TokenFilterException(HttpServletResponse.SC_BAD_REQUEST, - "Could not retrieve Client ID", BASIC_AUTH_ERROR_MSG); - } - } else { - log.error(BASIC_AUTH_ERROR_MSG); - throw new TokenFilterException(HttpServletResponse.SC_BAD_REQUEST, "Could not retrieve Client ID", - BASIC_AUTH_ERROR_MSG); - } - } else { - throw new TokenFilterException(HttpServletResponse.SC_BAD_REQUEST, IdentityCommonConstants - .OAUTH2_INVALID_REQUEST_MESSAGE, "Unable to find client id in the request"); - } - } catch (ParseException e) { - throw new TokenFilterException(HttpServletResponse.SC_UNAUTHORIZED, IdentityCommonConstants - .OAUTH2_INVALID_REQUEST_MESSAGE, "Error occurred while parsing the signed assertion", e); - } - } - - public static void setDefaultTokenFilter(DefaultTokenFilter tokenFilter) { - - defaultTokenFilter = tokenFilter; - } - - public static void setValidators(List validators) { - - TokenFilter.validators = validators; - } - - public List getValidators() { - - return validators; - } - - /** - * validates the transport header certificate and re-add to the header. - * - * @param request ServletRequest - * @return request - * @throws OpenBankingException - * @throws ServletException - */ - private ServletRequest cleanClientCertificateAndAppendTransportHeader(ServletRequest request) - throws ServletException, OpenBankingException { - - if (request instanceof HttpServletRequest) { - IdentityCommonHelper identityCommonHelper = new IdentityCommonHelper(); - if (identityCommonHelper.isTransportCertAsHeaderEnabled()) { - log.debug("Retrieving client transport certificate from header."); - String x509Certificate = ((HttpServletRequest) request) - .getHeader(IdentityCommonUtil.getMTLSAuthHeader()); - if (StringUtils.isNotEmpty(x509Certificate) && isClientCertificateEncoded()) { - try { - log.debug("Received encoded client certificate. URLDecoding cert."); - x509Certificate = URLDecoder.decode(x509Certificate, "UTF-8"); - } catch (UnsupportedEncodingException e) { - throw new OpenBankingException("Cannot decode the transport certificate passed through " + - "the request", e); - } - } - - try { - X509Certificate certificate = CertificateUtils.parseCertificate(x509Certificate); - if (certificate != null) { - RequestWrapper requestWrapper = new RequestWrapper((HttpServletRequest) request); - requestWrapper.setHeader(IdentityCommonUtil.getMTLSAuthHeader(), - new IdentityCommonHelper().encodeCertificateContent(certificate)); - return requestWrapper; - } - } catch (CertificateEncodingException e) { - throw new ServletException("Certificate not valid", e); - } catch (OpenBankingException e) { - // Ignore the error here, MTLSEnforcementValidator will validate the certificate - log.error("Invalid transport certificate received. Caused by, ", e); - } - } - } else { - throw new ServletException("Error occurred when handling the request, passed request is not a " + - "HttpServletRequest"); - } - return request; - } - - /** - * Get the clientCertificateEncode configuration. - * - * @return false if clientCertificateEncode configured as false, default true - */ - public boolean isClientCertificateEncoded() { - - Object isClientCertEncoded = IdentityExtensionsDataHolder.getInstance().getConfigurationMap() - .getOrDefault(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, true); - - return Boolean.parseBoolean(String.valueOf(isClientCertEncoded)); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/util/TokenFilterException.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/util/TokenFilterException.java deleted file mode 100644 index 008f5c71..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/util/TokenFilterException.java +++ /dev/null @@ -1,69 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.token.util; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; - -/** - * Token filter exception. - */ -public class TokenFilterException extends OpenBankingException { - - private String errorDescription; - private int errorCode; - - public TokenFilterException(int errorCode, String error, String errorDescription, Throwable e) { - - super(error, e); - this.errorDescription = errorDescription; - this.errorCode = errorCode; - } - - public TokenFilterException(int errorCode, String error, String errorDescription) { - - super(error); - this.errorDescription = errorDescription; - this.errorCode = errorCode; - } - - public TokenFilterException(String message, Throwable e) { - - super(message, e); - } - - public String getErrorDescription() { - - return errorDescription; - } - - public void setErrorDescription(String errorDescription) { - - this.errorDescription = errorDescription; - } - - public int getErrorCode() { - - return errorCode; - } - - public void setErrorCode(int errorCode) { - - this.errorCode = errorCode; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/ClientAuthenticatorValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/ClientAuthenticatorValidator.java deleted file mode 100644 index 0a3983b0..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/ClientAuthenticatorValidator.java +++ /dev/null @@ -1,126 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.token.validators; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.identity.token.util.TokenFilterException; -import com.wso2.openbanking.accelerator.identity.util.ClientAuthenticatorEnum; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonHelper; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import javax.servlet.ServletException; -import javax.servlet.ServletRequest; -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; - -/** - * Validates whether the registered client authentication method is invoked. - */ -public class ClientAuthenticatorValidator implements OBIdentityFilterValidator { - - private static final Log log = LogFactory.getLog(ClientAuthenticatorValidator.class); - - @Override - public void validate(ServletRequest request, String clientId) throws TokenFilterException, ServletException { - - if (request instanceof HttpServletRequest) { - String registeredClientAuthMethod = retrieveRegisteredAuthMethod(clientId); - - if (registeredClientAuthMethod.equals(IdentityCommonConstants.NOT_APPLICABLE)) { - return; - } - - // There can be multiple registered client auth methods - if (!(registeredClientAuthMethod.contains(retrieveRequestAuthMethod(request)))) { - throw new TokenFilterException(HttpServletResponse.SC_BAD_REQUEST, IdentityCommonConstants - .OAUTH2_INVALID_REQUEST_MESSAGE, "Request does not follow the registered token endpoint auth " + - "method " + registeredClientAuthMethod); - } - } else { - throw new ServletException("Error occurred during request validation, passed request is not a " + - "HttpServletRequest"); - } - } - - /** - * Get the authentication method that matches the request. - * - * @param request servlet request - * @return authentication method - */ - @Generated(message = "Excluding from code coverage because a the actual implementation test cases are coverd") - public String retrieveRequestAuthMethod(ServletRequest request) throws TokenFilterException { - - try { - if (isPrivateKeyJWTAuthentication(request)) { - log.debug("Validating request with JWT client authentication method"); - return ClientAuthenticatorEnum.PRIVATE_KEY_JWT.toString(); - } else if (new IdentityCommonHelper().isMTLSAuthentication(request)) { - log.debug("Validating request with MTLS client authentication method"); - return ClientAuthenticatorEnum.TLS_CLIENT_AUTH.toString(); - } - return "INVALID_AUTH"; - } catch (OpenBankingException e) { - throw new TokenFilterException(HttpServletResponse.SC_UNAUTHORIZED, IdentityCommonConstants - .OAUTH2_INVALID_REQUEST_MESSAGE, e.getMessage()); - } - } - - /** - * Validate whether the request follows the private key jwt authentication pattern. - * - * @param request servlet request - * @return whether request fallows PKJWT pattern - */ - public boolean isPrivateKeyJWTAuthentication(ServletRequest request) { - - String oauthJWTAssertionType = request.getParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION_TYPE); - String oauthJWTAssertion = request.getParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION); - return IdentityCommonConstants.OAUTH_JWT_BEARER_GRANT_TYPE.equals(oauthJWTAssertionType) && - StringUtils.isNotEmpty(oauthJWTAssertion); - } - - /** - * Retrieve client authentication method from sp metadata. - * - * @param clientId auth client ID - * @return the value of the client authentication method registered - * @throws TokenFilterException - */ - @Generated(message = "Excluding from code coverage because a service call is required for the method") - public String retrieveRegisteredAuthMethod(String clientId) throws TokenFilterException { - - try { - if (!(StringUtils.isNotEmpty(new IdentityCommonHelper().getCertificateContent(clientId)) - && IdentityCommonUtil.getRegulatoryFromSPMetaData(clientId))) { - return new IdentityCommonHelper().getAppPropertyFromSPMetaData(clientId, - IdentityCommonConstants.TOKEN_ENDPOINT_AUTH_METHOD); - } - } catch (OpenBankingException e) { - throw new TokenFilterException(HttpServletResponse.SC_UNAUTHORIZED, IdentityCommonConstants - .OAUTH2_INVALID_REQUEST_MESSAGE, "Client authentication method not registered", e); - } - return IdentityCommonConstants.NOT_APPLICABLE; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/MTLSCertificateValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/MTLSCertificateValidator.java deleted file mode 100644 index d1f45971..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/MTLSCertificateValidator.java +++ /dev/null @@ -1,63 +0,0 @@ -/* - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). All Rights Reserved. - * - * This software is the property of WSO2 LLC. and its suppliers, if any. - * Dissemination of any information or reproduction of any material contained - * herein in any form is strictly forbidden, unless permitted by WSO2 expressly. - * You may not alter or remove any copyright or other notice from copies of this content. - */ - -package com.wso2.openbanking.accelerator.identity.token.validators; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.CertificateUtils; -import com.wso2.openbanking.accelerator.identity.token.util.TokenFilterException; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.security.cert.X509Certificate; - -import javax.servlet.ServletException; -import javax.servlet.ServletRequest; -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; - -/** - * MTLS Certificate Validator. - * Validates the expiry status of the certificate. - */ -public class MTLSCertificateValidator implements OBIdentityFilterValidator { - - private static final Log log = LogFactory.getLog(MTLSCertificateValidator.class); - private static final String CERT_EXPIRED_ERROR = "Certificate with the serial number %s issued by the CA %s is " + - "expired"; - - @Override - public void validate(ServletRequest request, String clientId) throws TokenFilterException, ServletException { - - HttpServletRequest servletRequest = (HttpServletRequest) request; - String mtlsCertificate = servletRequest.getHeader(IdentityCommonUtil.getMTLSAuthHeader()); - // MTLSEnforcementValidator validates the presence of the certificate. - if (mtlsCertificate != null) { - try { - X509Certificate x509Certificate = CertificateUtils.parseCertificate(mtlsCertificate); - - if (CertificateUtils.isExpired(x509Certificate)) { - log.error(String.format(CERT_EXPIRED_ERROR, x509Certificate.getSerialNumber(), - x509Certificate.getIssuerDN().toString())); - throw new TokenFilterException(HttpServletResponse.SC_UNAUTHORIZED, - "Invalid mutual TLS request. Client certificate is expired", - String.format(CERT_EXPIRED_ERROR, x509Certificate.getSerialNumber(), - x509Certificate.getIssuerDN().toString())); - } - log.debug("Client certificate expiry validation completed successfully"); - } catch (OpenBankingException e) { - log.error("Invalid mutual TLS request. Client certificate is invalid", e); - throw new TokenFilterException(HttpServletResponse.SC_UNAUTHORIZED, IdentityCommonConstants - .OAUTH2_INVALID_CLIENT_MESSAGE, e.getMessage()); - } - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/MTLSEnforcementValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/MTLSEnforcementValidator.java deleted file mode 100644 index de5b0780..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/MTLSEnforcementValidator.java +++ /dev/null @@ -1,66 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.token.validators; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.CertificateUtils; -import com.wso2.openbanking.accelerator.identity.token.util.TokenFilterException; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import javax.servlet.ServletException; -import javax.servlet.ServletRequest; -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; - -/** - * Mutual TLS Enforcement Filter Validator. - * Enforces whether the Request is sent with MTLS cert as a header. - */ -public class MTLSEnforcementValidator implements OBIdentityFilterValidator { - - private static final Log log = LogFactory.getLog(MTLSEnforcementValidator.class); - - @Override - public void validate(ServletRequest request, String clientId) throws TokenFilterException, ServletException { - - if (request instanceof HttpServletRequest) { - - HttpServletRequest servletRequest = (HttpServletRequest) request; - String x509Certificate = servletRequest.getHeader(IdentityCommonUtil.getMTLSAuthHeader()); - try { - if (!(StringUtils.isNotEmpty(x509Certificate) && - CertificateUtils.parseCertificate(x509Certificate) != null)) { - throw new TokenFilterException(HttpServletResponse.SC_BAD_REQUEST, IdentityCommonConstants - .OAUTH2_INVALID_REQUEST_MESSAGE, "Transport certificate not passed through the " + - "request or the certificate is not valid"); - } - } catch (TokenFilterException e) { - throw new TokenFilterException(e.getErrorCode(), e.getMessage(), e.getErrorDescription()); - } catch (OpenBankingException e) { - throw new TokenFilterException(HttpServletResponse.SC_BAD_REQUEST, IdentityCommonConstants - .OAUTH2_INVALID_CLIENT_MESSAGE, "Invalid transport certificate. " + - e.getMessage(), e); - } - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/OBIdentityFilterValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/OBIdentityFilterValidator.java deleted file mode 100644 index 4cf41457..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/OBIdentityFilterValidator.java +++ /dev/null @@ -1,34 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.token.validators; - -import com.wso2.openbanking.accelerator.identity.token.util.TokenFilterException; - -import javax.servlet.ServletException; -import javax.servlet.ServletRequest; - -/** - * An interface which used to implement additional validations - * needed for the token endpoint in the Key Manager (oauth2/token). - */ -public interface OBIdentityFilterValidator { - - void validate(ServletRequest request, String clientId) throws TokenFilterException, ServletException; - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/SignatureAlgorithmEnforcementValidator.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/SignatureAlgorithmEnforcementValidator.java deleted file mode 100644 index a7065b3b..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/validators/SignatureAlgorithmEnforcementValidator.java +++ /dev/null @@ -1,108 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.token.validators; - -import com.nimbusds.jwt.SignedJWT; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.identity.token.util.TokenFilterException; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonHelper; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.text.ParseException; - -import javax.servlet.ServletRequest; -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; - -/** - * Filter validator to check if the client assertion is signed with the registered algorithm. - */ -public class SignatureAlgorithmEnforcementValidator implements OBIdentityFilterValidator { - - private static final Log log = LogFactory.getLog(SignatureAlgorithmEnforcementValidator.class); - - @Override - public void validate(ServletRequest request, String clientId) throws TokenFilterException { - - if (request instanceof HttpServletRequest) { - String signedObject = request.getParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION); - if (StringUtils.isNotEmpty(signedObject) && - StringUtils.isNotEmpty(getRegisteredSigningAlgorithm(clientId))) { - validateInboundSignatureAlgorithm(getRequestSigningAlgorithm(signedObject), - getRegisteredSigningAlgorithm(clientId)); - } - } - } - - /** - * Checks if the incoming signed request is signed with the registered algorithms during service provider creation. - * - * @param requestSigningAlgorithm the algorithm of signed message - * @param registeredSigningAlgorithm the algorithm registered during client authentication - */ - public void validateInboundSignatureAlgorithm(String requestSigningAlgorithm, String registeredSigningAlgorithm) - throws TokenFilterException { - - if (log.isDebugEnabled()) { - log.debug(String.format("Validating request algorithm %s against registered algorithm %s.", - requestSigningAlgorithm, registeredSigningAlgorithm)); - } - if (registeredSigningAlgorithm.equals(IdentityCommonConstants.NOT_APPLICABLE)) { - return; - } - if (!(StringUtils.isNotEmpty(requestSigningAlgorithm) && - requestSigningAlgorithm.equals(registeredSigningAlgorithm))) { - throw new TokenFilterException(HttpServletResponse.SC_UNAUTHORIZED, IdentityCommonConstants - .OAUTH2_INVALID_CLIENT_MESSAGE, "Registered algorithm does not match with the token " + - "signed algorithm"); - } - } - - @Generated(message = "Ignoring because it requires a service call") - public String getRegisteredSigningAlgorithm(String clientId) throws TokenFilterException { - - try { - if (!(StringUtils.isNotEmpty(new IdentityCommonHelper().getCertificateContent(clientId)) - && IdentityCommonUtil.getRegulatoryFromSPMetaData(clientId))) { - return new IdentityCommonHelper().getAppPropertyFromSPMetaData(clientId, - IdentityCommonConstants.TOKEN_ENDPOINT_AUTH_SIGNING_ALG); - } - } catch (OpenBankingException e) { - throw new TokenFilterException(HttpServletResponse.SC_UNAUTHORIZED, IdentityCommonConstants - .OAUTH2_INVALID_REQUEST_MESSAGE, "Token signing algorithm not registered", e); - } - return IdentityCommonConstants.NOT_APPLICABLE; - } - - public String getRequestSigningAlgorithm(String signedObject) throws TokenFilterException { - //retrieve algorithm from the signed JWT - try { - SignedJWT signedJWT = SignedJWT.parse(signedObject); - return signedJWT.getHeader().getAlgorithm().getName(); - } catch (ParseException e) { - throw new TokenFilterException(HttpServletResponse.SC_BAD_REQUEST, IdentityCommonConstants - .OAUTH2_INVALID_CLIENT_MESSAGE, "Error occurred while parsing the signed assertion", e); - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/wrapper/RequestWrapper.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/wrapper/RequestWrapper.java deleted file mode 100644 index 7515bd7d..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/wrapper/RequestWrapper.java +++ /dev/null @@ -1,130 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.token.wrapper; - -import java.io.ByteArrayOutputStream; -import java.io.IOException; -import java.util.Collections; -import java.util.Enumeration; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import javax.servlet.ReadListener; -import javax.servlet.ServletInputStream; -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletRequestWrapper; - -/** - * Request wrapper implementation. - */ -public class RequestWrapper extends HttpServletRequestWrapper { - - private Map headerMap = new HashMap<>(); - - /** - * Constructs a request object wrapping the given request. - * - * @param request - * @throws IllegalArgumentException if the request is null. - */ - private final ByteArrayOutputStream byteArrayOutputStream; - - public RequestWrapper(HttpServletRequest request) { - super(request); - byteArrayOutputStream = new ByteArrayOutputStream(); - } - - @Override - public ServletInputStream getInputStream() throws IOException { - final ServletInputStream originalInputStream = super.getInputStream(); - - return new ServletInputStream() { - @Override - public int read() throws IOException { - int data = originalInputStream.read(); - if (data != -1) { - byteArrayOutputStream.write(data); - } - return data; - } - - @Override - public boolean isFinished() { - return originalInputStream.isFinished(); - } - - @Override - public boolean isReady() { - return originalInputStream.isReady(); - } - - @Override - public void setReadListener(ReadListener readListener) { - originalInputStream.setReadListener(readListener); - } - }; - } - - public byte[] getCapturedRequest() { - return byteArrayOutputStream.toByteArray(); - } - - /** - * Set the header map to hold the header values. - * - * @param name - * @param value - */ - public void setHeader(String name, String value) { - - headerMap.put(name, value); - } - - @Override - public String getHeader(String name) { - - String headerValue = super.getHeader(name); - if (headerMap.containsKey(name)) { - headerValue = headerMap.get(name); - } - return headerValue; - } - - @Override - public Enumeration getHeaderNames() { - - List headerNames = Collections.list(super.getHeaderNames()); - for (Map.Entry entry: headerMap.entrySet()) { - // prevent adding duplicate entries to headerNames list - headerNames.remove(entry.getKey()); - } - headerNames.addAll(headerMap.keySet()); - return Collections.enumeration(headerNames); - } - - @Override - public Enumeration getHeaders(String name) { - - if (headerMap.containsKey(name)) { - return Collections.enumeration(Collections.singletonList(headerMap.get(name))); - } - return Collections.enumeration(Collections.list(super.getHeaders(name))); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/wrapper/ResponseWrapper.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/wrapper/ResponseWrapper.java deleted file mode 100644 index 916dbdcc..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/token/wrapper/ResponseWrapper.java +++ /dev/null @@ -1,84 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.token.wrapper; - -import net.sf.ehcache.constructs.web.filter.FilterServletOutputStream; - -import java.io.ByteArrayOutputStream; -import java.io.OutputStreamWriter; -import java.io.PrintWriter; -import java.nio.charset.StandardCharsets; - -import javax.servlet.ServletOutputStream; -import javax.servlet.http.HttpServletResponse; -import javax.servlet.http.HttpServletResponseWrapper; - -/** - * Response wrapper implementation. - */ -public class ResponseWrapper extends HttpServletResponseWrapper { - - private ByteArrayOutputStream output; - private int contentLength; - private String contentType; - - public ResponseWrapper(HttpServletResponse response) { - - super(response); - output = new ByteArrayOutputStream(); - } - - public byte[] getData() { - - return output.toByteArray(); - } - - public ServletOutputStream getOutputStream() { - - return new FilterServletOutputStream(output); - } - - public PrintWriter getWriter() { - - return new PrintWriter(new OutputStreamWriter(getOutputStream(), StandardCharsets.UTF_8), true); - } - - public int getContentLength() { - - return contentLength; - } - - public void setContentLength(int length) { - - this.contentLength = length; - super.setContentLength(length); - } - - public String getContentType() { - - return contentType; - } - - public void setContentType(String type) { - - this.contentType = type; - super.setContentType(type); - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/ClientAuthenticatorEnum.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/ClientAuthenticatorEnum.java deleted file mode 100644 index 5ea72436..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/ClientAuthenticatorEnum.java +++ /dev/null @@ -1,50 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.util; - -/** - * Enumerations for client authenticators. - */ -public enum ClientAuthenticatorEnum { - PRIVATE_KEY_JWT("private_key_jwt"), - TLS_CLIENT_AUTH("tls_client_auth"); - - private final String value; - - ClientAuthenticatorEnum(String value) { - - this.value = value; - } - - public static ClientAuthenticatorEnum fromValue(String text) { - - for (ClientAuthenticatorEnum authenticatorEnum : ClientAuthenticatorEnum.values()) { - if (String.valueOf(authenticatorEnum.value).equals(text)) { - return authenticatorEnum; - } - } - return null; - } - - @Override - public String toString() { - - return String.valueOf(value); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/HTTPClientUtils.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/HTTPClientUtils.java deleted file mode 100644 index 33eca0a0..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/HTTPClientUtils.java +++ /dev/null @@ -1,198 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.util; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.config.Registry; -import org.apache.http.config.RegistryBuilder; -import org.apache.http.conn.socket.ConnectionSocketFactory; -import org.apache.http.conn.socket.PlainConnectionSocketFactory; -import org.apache.http.conn.ssl.SSLConnectionSocketFactory; -import org.apache.http.conn.ssl.SSLContexts; -import org.apache.http.conn.ssl.SSLSocketFactory; -import org.apache.http.conn.ssl.TrustSelfSignedStrategy; -import org.apache.http.conn.ssl.X509HostnameVerifier; -import org.apache.http.impl.client.CloseableHttpClient; -import org.apache.http.impl.client.HttpClients; -import org.apache.http.impl.conn.PoolingHttpClientConnectionManager; -import org.wso2.carbon.base.ServerConfiguration; -import org.wso2.carbon.user.api.RealmConfiguration; -import org.wso2.carbon.user.core.UserStoreException; - - -import java.io.FileInputStream; -import java.io.IOException; -import java.nio.charset.StandardCharsets; -import java.security.KeyManagementException; -import java.security.KeyStore; -import java.security.KeyStoreException; -import java.security.NoSuchAlgorithmException; -import java.security.cert.CertificateException; -import java.util.Base64; - -import javax.net.ssl.SSLContext; - -/** - * HTTP Client Utility methods. - */ -public class HTTPClientUtils { - - public static final String ALLOW_ALL = "AllowAll"; - public static final String STRICT = "Strict"; - public static final String HOST_NAME_VERIFIER = "httpclient.hostnameVerifier"; - public static final String HTTP_PROTOCOL = "http"; - public static final String HTTPS_PROTOCOL = "https"; - private static final String[] SUPPORTED_HTTP_PROTOCOLS = {"TLSv1.2"}; - private static final Log log = LogFactory.getLog(HTTPClientUtils.class); - - /** - * Get closeable https client. - * - * @return Closeable https client - * @throws OpenBankingException OpenBankingException exception - */ - @Generated(message = "Unit testable components are covered") - public static CloseableHttpClient getHttpsClient() throws OpenBankingException { - - SSLConnectionSocketFactory sslsf = createSSLConnectionSocketFactory(); - - Registry socketFactoryRegistry = RegistryBuilder.create() - .register(HTTP_PROTOCOL, new PlainConnectionSocketFactory()) - .register(HTTPS_PROTOCOL, sslsf) - .build(); - - final PoolingHttpClientConnectionManager connectionManager = (socketFactoryRegistry != null) ? - new PoolingHttpClientConnectionManager(socketFactoryRegistry) : - new PoolingHttpClientConnectionManager(); - - return HttpClients.custom().setConnectionManager(connectionManager).build(); - } - - /** - * create a SSL Connection Socket Factory. - * - * @return SSLConnectionSocketFactory - * @throws OpenBankingException - */ - @Generated(message = "Ignoring because ServerConfiguration cannot be mocked") - private static SSLConnectionSocketFactory createSSLConnectionSocketFactory() - throws OpenBankingException { - - KeyStore trustStore = null; - - trustStore = loadKeyStore( - ServerConfiguration.getInstance().getFirstProperty("Security.TrustStore.Location"), - ServerConfiguration.getInstance().getFirstProperty("Security.TrustStore.Password")); - - // Trust own CA and all self-signed certs - SSLContext sslcontext = null; - try { - sslcontext = SSLContexts.custom().loadTrustMaterial(trustStore, new TrustSelfSignedStrategy()).build(); - } catch (NoSuchAlgorithmException | KeyManagementException | KeyStoreException e) { - throw new OpenBankingException("Unable to create the ssl context", e); - } - - // Allow TLSv1 protocol only - return new SSLConnectionSocketFactory(sslcontext, SUPPORTED_HTTP_PROTOCOLS, - null, getX509HostnameVerifier()); - - } - - /** - * Load the keystore when the location and password is provided. - * - * @param keyStoreLocation Location of the keystore - * @param keyStorePassword Keystore password - * @return Keystore as an object - * @throws OpenBankingException when failed to load Keystore from given details - */ - @SuppressFBWarnings("PATH_TRAVERSAL_IN") - // Suppressed content - new FileInputStream(keyStoreLocation) - // Suppression reason - False Positive : Keystore location is obtained from deployment.toml. So it can be marked - // as a trusted filepath - // Suppressed warning count - 1 - public static KeyStore loadKeyStore(String keyStoreLocation, String keyStorePassword) - throws OpenBankingException { - - KeyStore keyStore; - - try (FileInputStream inputStream = new FileInputStream(keyStoreLocation)) { - keyStore = KeyStore.getInstance(KeyStore.getDefaultType()); - keyStore.load(inputStream, keyStorePassword.toCharArray()); - return keyStore; - } catch (KeyStoreException e) { - throw new OpenBankingException("Error while retrieving aliases from keystore", e); - } catch (IOException | CertificateException | NoSuchAlgorithmException e) { - throw new OpenBankingException("Error while loading keystore", e); - } - } - - /** - * Get the Hostname Verifier property in set in system properties. - * - * @return X509HostnameVerifier - */ - public static X509HostnameVerifier getX509HostnameVerifier() { - - String hostnameVerifierOption = System.getProperty(HOST_NAME_VERIFIER); - X509HostnameVerifier hostnameVerifier; - - if (ALLOW_ALL.equalsIgnoreCase(hostnameVerifierOption)) { - hostnameVerifier = SSLSocketFactory.ALLOW_ALL_HOSTNAME_VERIFIER; - } else if (STRICT.equalsIgnoreCase(hostnameVerifierOption)) { - hostnameVerifier = SSLSocketFactory.STRICT_HOSTNAME_VERIFIER; - } else { - hostnameVerifier = SSLSocketFactory.BROWSER_COMPATIBLE_HOSTNAME_VERIFIER; - } - - if (log.isDebugEnabled()) { - log.debug(String.format("Proceeding with %s : %s", HOST_NAME_VERIFIER, - hostnameVerifierOption)); - } - return hostnameVerifier; - - } - - /** - * Get base 64 encoded credentials of basic authentication for protected consent APIs. - * - * @return basic auth - */ - @Generated(message = "Excluding from code coverage since it requires a service call") - public static String getBasicAuthCredentials() throws OpenBankingException { - RealmConfiguration realmConfiguration; - try { - realmConfiguration = IdentityExtensionsDataHolder.getInstance().getRealmService() - .getBootstrapRealm().getUserStoreManager().getRealmConfiguration(); - } catch (UserStoreException e) { - throw new OpenBankingException("Error while retrieving session data", e); - } - - String adminUsername = realmConfiguration.getAdminUserName(); - char[] adminPassword = realmConfiguration.getAdminPassword().toCharArray(); - - String credentials = adminUsername + ":" + String.valueOf(adminPassword); - return Base64.getEncoder().encodeToString(credentials.getBytes(StandardCharsets.UTF_8)); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/IdentityCommonConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/IdentityCommonConstants.java deleted file mode 100644 index 0b9c38ef..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/IdentityCommonConstants.java +++ /dev/null @@ -1,112 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.util; - -/** - * Class containing the constants for Open Banking Common module. - */ -public class IdentityCommonConstants { - - public static final String CLIENT_ID = "client_id"; - public static final String REQUEST_URI = "request_uri"; - public static final String REQUEST = "request"; - public static final String RESPONSE_TYPE = "response_type"; - public static final String REDIRECT_URI = "redirect_uri"; - public static final String CARBON_HOME = "carbon.home"; - public static final String REGULATORY_COMPLIANCE = "regulatory"; - public static final String TOKEN_ENDPOINT_AUTH_METHOD = "token_endpoint_auth_method"; - public static final String TOKEN_ENDPOINT_AUTH_SIGNING_ALG = "token_endpoint_auth_signing_alg"; - public static final String REQUEST_OBJECT_SIGNING_ALG = "request_object_signing_alg"; - public static final String CLIENT_ID_ERROR = "Client id not found"; - public static final String OAUTH_CLIENT_ID = "client_id"; - public static final String OAUTH_CLIENT_SECRET = "client_secret"; - public static final String AUTHORIZATION_HEADER = "authorization"; - public static final String OAUTH_JWT_ASSERTION = "client_assertion"; - public static final String OAUTH_JWT_ASSERTION_TYPE = "client_assertion_type"; - public static final String OAUTH_JWT_BEARER_GRANT_TYPE = "urn:ietf:params:oauth:client-assertion-type:jwt-bearer"; - public static final String BEGIN_CERT = "-----BEGIN CERTIFICATE-----"; - public static final String END_CERT = "-----END CERTIFICATE-----"; - public static final String MTLS_AUTH_HEADER = "MutualTLS.ClientCertificateHeader"; - public static final String X509 = "X.509"; - public static final String TOKEN_FILTER = "Identity.Filters.TokenFilter"; - public static final String TOKEN_VALIDATORS = "Identity.TokenFilterValidators.Validator"; - public static final String CLAIM_PROVIDER = "Identity.Extensions.ClaimProvider"; - public static final String INTROSPECTION_DATA_PROVIDER = "Identity.Extensions.IntrospectionDataProvider"; - public static final String SIGNING_CERT_KID = "Identity.SigningCertificateKid"; - public static final String OAUTH_ERROR = "error"; - public static final String OAUTH_ERROR_DESCRIPTION = "error_description"; - public static final String JAVAX_SERVLET_REQUEST_CERTIFICATE = "javax.servlet.request.X509Certificate"; - public static final String OB_CONSENT_ID_PREFIX = "OB_CONSENT_ID_"; - public static final String OB_PREFIX = "OB_"; - public static final String TIME_PREFIX = "TIME_"; - public static final String TLS_CERT = "tls_cert"; - public static final String CERT_PREFIX = "x5t#"; - public static final String CERTIFICATE_HEADER = "x-wso2-mutual-auth-cert"; - public static final String CERTIFICATE_HEADER_ATTRIBUTE = "x-wso2-mutual-auth-cert-attribute"; - public static final String SPACE_SEPARATOR = " "; - public static final String SCOPE = "scope"; - public static final String CONDITIONAL_COMMON_AUTH_SCRIPT_FILE_NAME = "common.auth.script.js"; - public static final String PRIMARY_AUTHENTICATOR_DISPLAYNAME = "SCA.PrimaryAuthenticator.DisplayName"; - public static final String PRIMARY_AUTHENTICATOR_NAME = "SCA.PrimaryAuthenticator.Name"; - public static final String IDENTITY_PROVIDER_NAME = "SCA.IdpName"; - public static final String IDENTITY_PROVIDER_STEP = "SCA.IdpStep"; - public static final String REQUEST_VALIDATOR = "Identity.Extensions.RequestObjectValidator"; - public static final String PUSH_AUTH_REQUEST_VALIDATOR = "Identity.Extensions.PushAuthRequestValidator"; - public static final String RESPONSE_HANDLER = "Identity.Extensions.ResponseTypeHandler"; - public static final String ENABLE_TRANSPORT_CERT_AS_HEADER = "Identity.ClientTransportCertAsHeaderEnabled"; - public static final String ENABLE_SUBJECT_AS_PPID = "Identity.EnableSubjectPPID"; - public static final String REMOVE_USER_STORE_DOMAIN_FROM_SUBJECT = - "Identity.TokenSubject.RemoveUserStoreDomainFromSubject"; - public static final String REMOVE_TENANT_DOMAIN_FROM_SUBJECT = - "Identity.TokenSubject.RemoveTenantDomainFromSubject"; - - public static final String AUTH_SERVLET_EXTENSION = "Identity.Extensions.AuthenticationWebApp.ServletExtension"; - public static final String CONSENT_ID_CLAIM_NAME = "Identity.ConsentIDClaimName"; - public static final String SP_ACCESS_TOKEN_INPUT_STREAM = "AccessTokenInputStream"; - public static final String INPUT_STREAM_VERSION = "1.0.0"; - public static final String ACCESS_TOKEN_ID = "accessTokenID"; - public static final String NOT_APPLICABLE = "N/A"; - public static final String CONSENT_JWT_PAYLOAD_VALIDATION = "Consent.Validation.JWTPayloadValidation"; - - public static final String S_HASH = "s_hash"; - public static final String CODE = "code"; - public static final String DCR_INTERNAL_SCOPE = "OB_DCR"; - public static final String OPENID_SCOPE = "openid"; - public static final String CLIENT_CREDENTIALS = "client_credentials"; - public static final String CARBON_SUPER = "carbon.super"; - public static final String REGISTRATION_ACCESS_TOKEN = "registration_access_token"; - public static final String REGISTRATION_CLIENT_URI = "registration_client_uri"; - public static final String PRIVATE_KEY = "pvt_key"; - public static final String ALG_ES256 = "ES256"; - public static final String ALG_PS256 = "PS256"; - public static final String DEFAULT_JWKS_URI = "software_jwks_endpoint"; - public static final String DEFAULT_REGISTRATION_CLIENT_URI = "https://localhost:8243/open-banking/0.1/register/"; - public static final String PAR_ENDPOINT = "/api/openbanking/push-authorization/par"; - - public static final String TLS_CERT_JWKS = "Identity.MutualTLS.TransportCertificateJWKS"; - public static final String CLIENT_CERTIFICATE_ENCODE = "Identity.MutualTLS.ClientCertificateEncode"; - public static final String DCR_MODIFY_RESPONSE = "DCR.ModifyResponse"; - public static final String DCR_SCOPE = "DCR.Scope"; - public static final String DCR_REGISTRATION_CLIENT_URI = "DCR.RegistrationClientURI"; - - //Error Constants - public static final String OAUTH2_INVALID_CLIENT_MESSAGE = "invalid_client"; - public static final String OAUTH2_INVALID_REQUEST_MESSAGE = "invalid_request"; - public static final String OAUTH2_INTERNAL_SERVER_ERROR = "server_error"; -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/IdentityCommonHelper.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/IdentityCommonHelper.java deleted file mode 100644 index 43d1863c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/IdentityCommonHelper.java +++ /dev/null @@ -1,271 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.util; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.CertificateUtils; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.common.util.ServiceProviderUtils; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.oltu.oauth2.common.message.types.GrantType; -import org.wso2.carbon.context.CarbonContext; -import org.wso2.carbon.context.PrivilegedCarbonContext; -import org.wso2.carbon.identity.application.common.IdentityApplicationManagementException; -import org.wso2.carbon.identity.application.common.model.ApplicationBasicInfo; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.application.common.model.ServiceProviderProperty; -import org.wso2.carbon.identity.application.common.util.IdentityApplicationConstants; -import org.wso2.carbon.identity.application.mgt.ApplicationManagementService; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.bean.OAuthClientAuthnContext; -import org.wso2.carbon.identity.oauth2.dao.OAuthTokenPersistenceFactory; -import org.wso2.carbon.identity.oauth2.dto.OAuthRevocationRequestDTO; -import org.wso2.carbon.identity.oauth2.dto.OAuthRevocationResponseDTO; -import org.wso2.carbon.user.api.RealmConfiguration; -import org.wso2.carbon.user.core.UserStoreException; -import org.wso2.carbon.utils.multitenancy.MultitenantUtils; - -import java.nio.charset.StandardCharsets; -import java.security.cert.CertificateEncodingException; -import java.security.cert.X509Certificate; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Base64; -import java.util.List; -import java.util.Optional; -import java.util.Set; -import java.util.stream.Collectors; - -import javax.annotation.Nullable; -import javax.servlet.ServletRequest; -import javax.servlet.http.HttpServletRequest; - -/** - * Identity common helper class. - */ -public class IdentityCommonHelper { - - private static final Log log = LogFactory.getLog(IdentityCommonHelper.class); - - /** - * Utility method get the application property from SP Meta Data. - * @param clientId ClientId of the application - * @return the service provider certificate - * @throws OpenBankingException - */ - @Generated(message = "Excluding from code coverage since it requires a service call") - public String getCertificateContent(String clientId) throws OpenBankingException { - - Optional serviceProvider; - try { - serviceProvider = Optional.ofNullable(IdentityExtensionsDataHolder.getInstance() - .getApplicationManagementService().getServiceProviderByClientId(clientId, - IdentityApplicationConstants.OAuth2.NAME, - ServiceProviderUtils.getSpTenantDomain(clientId))); - if (serviceProvider.isPresent()) { - return serviceProvider.get().getCertificateContent(); - } - } catch (IdentityApplicationManagementException e) { - log.error(String.format("Error occurred while retrieving OAuth2 application data for clientId %s", - clientId), e); - throw new OpenBankingException("Error occurred while retrieving OAuth2 application data for clientId" - , e); - } - return ""; - } - - /** - * Utility method get the application property from SP Meta Data. - * - * @param clientId ClientId of the application - * @param property Property of the application - * @return the property value from SP metadata - * @throws OpenBankingException - */ - @Generated(message = "Excluding from code coverage since it requires a service call") - public String getAppPropertyFromSPMetaData(String clientId, String property) throws OpenBankingException { - - String spProperty = null; - - if (StringUtils.isNotEmpty(clientId)) { - Optional serviceProvider; - try { - serviceProvider = Optional.ofNullable(IdentityExtensionsDataHolder.getInstance() - .getApplicationManagementService().getServiceProviderByClientId(clientId, - IdentityApplicationConstants.OAuth2.NAME, - ServiceProviderUtils.getSpTenantDomain(clientId))); - if (serviceProvider.isPresent()) { - spProperty = Arrays.stream(serviceProvider.get().getSpProperties()) - .collect(Collectors.toMap(ServiceProviderProperty::getName, - ServiceProviderProperty::getValue)) - .get(property); - } - } catch (IdentityApplicationManagementException e) { - log.error(String.format("Error occurred while retrieving OAuth2 application data for clientId %s", - clientId), e); - throw new OpenBankingException("Error occurred while retrieving OAuth2 application data for clientId" - , e); - } - } else { - log.error(IdentityCommonConstants.CLIENT_ID_ERROR); - throw new OpenBankingException(IdentityCommonConstants.CLIENT_ID_ERROR); - } - - return spProperty; - } - - /** - * Validate whether the request follows mtls authentication pattern. - * - * @param request servlet request - * @return whether request fallows MTLS pattern - */ - public boolean isMTLSAuthentication(ServletRequest request) throws - OpenBankingException { - - if (request instanceof HttpServletRequest) { - String oauthClientID = request.getParameter(IdentityCommonConstants.OAUTH_CLIENT_ID); - String oauthClientSecret = request.getParameter(IdentityCommonConstants.OAUTH_CLIENT_SECRET); - String oauthJWTAssertion = request.getParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION); - String oauthJWTAssertionType = request.getParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION_TYPE); - HttpServletRequest servletRequest = (HttpServletRequest) request; - String authorizationHeader = servletRequest.getHeader(IdentityCommonConstants.AUTHORIZATION_HEADER); - String x509Certificate = servletRequest.getHeader(IdentityCommonUtil.getMTLSAuthHeader()); - return (StringUtils.isNotEmpty(oauthClientID) && StringUtils.isEmpty(oauthClientSecret) && - StringUtils.isEmpty(oauthJWTAssertion) && StringUtils.isEmpty(oauthJWTAssertionType) && - StringUtils.isEmpty(authorizationHeader) && x509Certificate != null && - CertificateUtils.parseCertificate(x509Certificate) != null); - } else { - throw new OpenBankingException("Error occurred during request validation, passed request is not a " + - "HttpServletRequest"); - } - } - - /** - * Get the configured value of the transport cert as header enable. - * - * @return value of the transport cert as header enable - */ - public boolean isTransportCertAsHeaderEnabled() { - - Optional certAsHeader = - Optional.ofNullable(IdentityExtensionsDataHolder.getInstance().getConfigurationMap() - .get(IdentityCommonConstants.ENABLE_TRANSPORT_CERT_AS_HEADER)); - return certAsHeader.filter(isEnabled -> Boolean.parseBoolean(isEnabled.toString())).isPresent(); - } - - /** - * Retrieve all Service provider information. - * - * @return list of service providers - * @throws IdentityApplicationManagementException when get application basic info fails - * @throws UserStoreException when get realm configuration fails - */ - @Generated(message = "Excluding from code coverage since it requires a service call") - public List getAllServiceProviders() - throws IdentityApplicationManagementException, UserStoreException { - - ApplicationManagementService applicationManagementService = IdentityExtensionsDataHolder.getInstance() - .getApplicationManagementService(); - - List serviceProviderList = new ArrayList<>(); - if (applicationManagementService != null) { - - RealmConfiguration realmConfig = IdentityExtensionsDataHolder.getInstance().getRealmService() - .getBootstrapRealm().getUserStoreManager().getRealmConfiguration(); - - final String adminUsername = realmConfig.getAdminUserName(); - final String tenantDomain = MultitenantUtils.getTenantDomain(adminUsername); - final int totalApplicationCount = applicationManagementService - .getCountOfAllApplications(tenantDomain, adminUsername); - - ApplicationBasicInfo[] applicationBasicInfo = applicationManagementService - .getApplicationBasicInfo(tenantDomain, adminUsername, 0, totalApplicationCount); - // Set tenant domain before calling applicationManagementService - if (CarbonContext.getThreadLocalCarbonContext().getTenantDomain() == null) { - PrivilegedCarbonContext.getThreadLocalCarbonContext().setTenantDomain(tenantDomain); - } - if (applicationBasicInfo != null && applicationBasicInfo.length > 0) { - for (ApplicationBasicInfo basicInfo : applicationBasicInfo) { - serviceProviderList - .add(applicationManagementService.getServiceProvider(basicInfo.getApplicationId())); - } - } - } - return serviceProviderList; - } - - /** - * Encode the certificate content. - * @param certificate - * @return - * @throws CertificateEncodingException - */ - public String encodeCertificateContent(X509Certificate certificate) throws CertificateEncodingException { - if (certificate != null) { - byte[] encodedContent = certificate.getEncoded(); - return IdentityCommonConstants.BEGIN_CERT + new String(Base64.getEncoder().encode(encodedContent), - StandardCharsets.UTF_8) + IdentityCommonConstants.END_CERT; - } else { - return null; - } - } - - /** - * Revokes access tokens for the given client id. - * - * @param clientId consumer key of the application - * @throws IdentityOAuth2Exception when revoking access tokens fails - */ - @Generated(message = "Excluding from code coverage since it requires service calls") - public void revokeAccessTokensByClientId(@Nullable final String clientId) throws IdentityOAuth2Exception { - - if (StringUtils.isEmpty(clientId)) { - return; - } - - Set activeTokens = OAuthTokenPersistenceFactory.getInstance().getAccessTokenDAO() - .getActiveTokensByConsumerKey(clientId); - if (!activeTokens.isEmpty()) { - OAuthClientAuthnContext oAuthClientAuthnContext = new OAuthClientAuthnContext(); - oAuthClientAuthnContext.setAuthenticated(true); - oAuthClientAuthnContext.setClientId(clientId); - - OAuthRevocationRequestDTO revocationRequestDTO = new OAuthRevocationRequestDTO(); - revocationRequestDTO.setOauthClientAuthnContext(oAuthClientAuthnContext); - revocationRequestDTO.setConsumerKey(clientId); - revocationRequestDTO.setTokenType(GrantType.REFRESH_TOKEN.toString()); - - for (String accessToken : activeTokens) { - revocationRequestDTO.setToken(accessToken); - OAuthRevocationResponseDTO oAuthRevocationResponseDTO = IdentityExtensionsDataHolder.getInstance() - .getOAuth2Service().revokeTokenByOAuthClient(revocationRequestDTO); - - if (oAuthRevocationResponseDTO.isError()) { - throw new IdentityOAuth2Exception( - String.format("Error occurred while revoking access tokens for clientId: %s. Caused by, %s", - clientId, oAuthRevocationResponseDTO.getErrorMsg())); - } - } - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/IdentityCommonUtil.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/IdentityCommonUtil.java deleted file mode 100644 index 69bf3fa3..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/java/com/wso2/openbanking/accelerator/identity/util/IdentityCommonUtil.java +++ /dev/null @@ -1,436 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.util; - -import com.google.common.base.Charsets; -import com.nimbusds.jose.JOSEException; -import com.nimbusds.jose.JWSAlgorithm; -import com.nimbusds.jose.JWSHeader; -import com.nimbusds.jose.JWSSigner; -import com.nimbusds.jose.JWSVerifier; -import com.nimbusds.jose.crypto.RSASSASigner; -import com.nimbusds.jose.crypto.RSASSAVerifier; -import com.nimbusds.jwt.JWTClaimsSet; -import com.nimbusds.jwt.SignedJWT; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingRuntimeException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.identity.cache.IdentityCache; -import com.wso2.openbanking.accelerator.identity.cache.IdentityCacheKey; -import com.wso2.openbanking.accelerator.identity.dcr.validation.DCRCommonConstants; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.oltu.oauth2.common.exception.OAuthProblemException; -import org.wso2.carbon.core.util.KeyStoreManager; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.application.common.model.ServiceProviderProperty; -import org.wso2.carbon.identity.core.util.IdentityUtil; -import org.wso2.carbon.identity.oauth.cache.SessionDataCache; -import org.wso2.carbon.identity.oauth.cache.SessionDataCacheEntry; -import org.wso2.carbon.identity.oauth.cache.SessionDataCacheKey; -import org.wso2.carbon.identity.oauth.config.OAuthServerConfiguration; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.util.OAuth2Util; - -import java.io.FileInputStream; -import java.io.IOException; -import java.io.InputStream; -import java.nio.charset.StandardCharsets; -import java.security.Key; -import java.security.KeyStore; -import java.security.KeyStoreException; -import java.security.MessageDigest; -import java.security.NoSuchAlgorithmException; -import java.security.PublicKey; -import java.security.cert.Certificate; -import java.security.cert.CertificateException; -import java.security.cert.X509Certificate; -import java.security.interfaces.RSAPrivateKey; -import java.security.interfaces.RSAPublicKey; -import java.text.ParseException; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Base64; -import java.util.HashMap; -import java.util.LinkedList; -import java.util.List; -import java.util.Map; -import java.util.Optional; -import java.util.stream.Collectors; - -import javax.servlet.http.HttpServletRequest; - -/** - * Utility Class for Identity Open Banking. - */ -public class IdentityCommonUtil { - - private static final Log log = LogFactory.getLog(IdentityCommonUtil.class); - private static IdentityCache identityCache; - - /** - * Get the configured certificate header name. - * - * @return value of the cert header name configuration - */ - public static String getMTLSAuthHeader() { - - return Optional.ofNullable(IdentityUtil.getProperty(IdentityCommonConstants.MTLS_AUTH_HEADER)) - .orElse("CONFIG_NOT_FOUND"); - } - - /** - * Remove the internal scopes from the space delimited list of authorized scopes. - * - * @param scopes Authorized scopes of the token - * @return scopes by removing the internal scopes - */ - public static String[] removeInternalScopes(String[] scopes) { - - String consentIdClaim = IdentityExtensionsDataHolder.getInstance().getConfigurationMap() - .get(IdentityCommonConstants.CONSENT_ID_CLAIM_NAME).toString(); - - if (scopes != null && scopes.length > 0) { - List scopesList = new LinkedList<>(Arrays.asList(scopes)); - scopesList.removeIf(s -> s.startsWith(consentIdClaim)); - scopesList.removeIf(s -> s.startsWith(IdentityCommonConstants.OB_PREFIX)); - scopesList.removeIf(s -> s.startsWith(IdentityCommonConstants.TIME_PREFIX)); - scopesList.removeIf(s -> s.startsWith(IdentityCommonConstants.CERT_PREFIX)); - return scopesList.toArray(new String[scopesList.size()]); - } - return scopes; - } - - /** - * Cache regulatory property if exists. - * - * @param clientId clientId ClientId of the application - * @return the regulatory property from cache if exists or from sp metadata - * @throws OpenBankingException - */ - @Generated(message = "Excluding from code coverage since it requires a cache initialization/service call") - public static synchronized boolean getRegulatoryFromSPMetaData(String clientId) throws OpenBankingException { - - if (StringUtils.isNotEmpty(clientId)) { - // Skip My account and Console service providers with non opaque clientIds - if (clientId.equalsIgnoreCase("CONSOLE") || - clientId.equalsIgnoreCase("MY_ACCOUNT")) { - return false; - } - - if (identityCache == null) { - log.debug("Creating new Identity cache"); - identityCache = new IdentityCache(); - } - - IdentityCacheKey identityCacheKey = IdentityCacheKey.of(clientId - .concat("_").concat(OpenBankingConstants.REGULATORY)); - Object regulatoryProperty = null; - - regulatoryProperty = identityCache.getFromCacheOrRetrieve(identityCacheKey, - () -> new IdentityCommonHelper().getAppPropertyFromSPMetaData(clientId, - IdentityCommonConstants.REGULATORY_COMPLIANCE)); - - if (regulatoryProperty != null) { - return Boolean.parseBoolean(regulatoryProperty.toString()); - } else { - throw new OpenBankingException("Unable to retrieve regulatory property from sp metadata"); - } - } else { - throw new OpenBankingException(IdentityCommonConstants.CLIENT_ID_ERROR); - } - } - - public static ServiceProviderProperty getServiceProviderProperty(String spPropertyName, String spPropertyValue) { - - ServiceProviderProperty serviceProviderProperty = new ServiceProviderProperty(); - serviceProviderProperty.setValue(spPropertyValue); - serviceProviderProperty.setName(spPropertyName); - serviceProviderProperty.setDisplayName(spPropertyName); - return serviceProviderProperty; - } - - /** - * Sign a string body using the carbon default key pair. - * Skipped in unit tests since @KeystoreManager cannot be mocked - * - * @param body the body that needs to be signed as a string - * @return string value of the signed JWT - * @throws Exception error if the tenant is invalid - */ - public static String signJWTWithDefaultKey(String body) throws Exception { - KeyStoreManager keyStoreManager = KeyStoreManager.getInstance(-1234); - Key privateKey = keyStoreManager.getDefaultPrivateKey(); - return generateJWT(body, privateKey); - } - - /** - * Validate a JWT signature by providing the alias in the client truststore. - * Skipped in unit tests since @KeystoreManager cannot be mocked - * - * @param jwtString string value of the JWT to be validated - * @param alias alias in the trust store - * @return boolean value depicting whether the signature is valid - * @throws OpenBankingException error with message mentioning the cause - */ - public static boolean validateJWTSignatureWithPublicKey(String jwtString, String alias) - throws OpenBankingException { - - Certificate certificate; - try { - KeyStore trustStore = getTrustStore(); - certificate = trustStore.getCertificate(alias); - } catch (Exception e) { - throw new OpenBankingException("Error while retrieving certificate from truststore"); - } - - if (certificate == null) { - throw new OpenBankingException("Certificate not found for provided alias"); - } - PublicKey publicKey = certificate.getPublicKey(); - - try { - JWSVerifier verifier = new RSASSAVerifier((RSAPublicKey) publicKey); - return SignedJWT.parse(jwtString).verify(verifier); - } catch (JOSEException | ParseException e) { - throw new OpenBankingException("Error occurred while validating JWT signature"); - } - - } - - /** - * Util method to get the configured trust store by carbon config or cached instance. - * - * @return Keystore instance of the truststore - * @throws Exception Error when loading truststore or carbon truststore config unavailable - */ - public static KeyStore getTrustStore() throws Exception { - if (IdentityExtensionsDataHolder.getInstance().getTrustStore() == null) { - String trustStoreLocation = System.getProperty("javax.net.ssl.trustStore"); - String trustStorePassword = System.getProperty("javax.net.ssl.trustStorePassword"); - String trustStoreType = System.getProperty("javax.net.ssl.trustStoreType"); - - if (trustStoreLocation == null || trustStorePassword == null || trustStoreType == null) { - throw new Exception("Trust store config not available"); - } - - try (InputStream keyStoreStream = new FileInputStream(trustStoreLocation)) { - KeyStore keyStore = KeyStore.getInstance(trustStoreType); // or "PKCS12" - keyStore.load(keyStoreStream, trustStorePassword.toCharArray()); - IdentityExtensionsDataHolder.getInstance().setTrustStore(keyStore); - } catch (IOException | CertificateException | KeyStoreException | NoSuchAlgorithmException e) { - throw new Exception("Error while loading truststore.", e); - } - } - return IdentityExtensionsDataHolder.getInstance().getTrustStore(); - } - - /** - * Util method to generate JWT using a payload and a private key. RS256 is the algorithm used - * - * @param payload The payload body to be signed - * @param privateKey The private key for the JWT to be signed with - * @return String signed JWT - */ - public static String generateJWT(String payload, Key privateKey) { - - if (privateKey == null || payload == null) { - log.debug("Null value passed for payload or key. Cannot generate JWT"); - throw new OpenBankingRuntimeException("Payload and key cannot be null"); - } - - if (!(privateKey instanceof RSAPrivateKey)) { - throw new OpenBankingRuntimeException("Private key should be an instance of RSAPrivateKey"); - } - - JWSSigner signer = new RSASSASigner((RSAPrivateKey) privateKey); - JWSHeader.Builder headerBuilder = new JWSHeader.Builder(JWSAlgorithm.RS256); - - SignedJWT signedJWT = null; - try { - signedJWT = new SignedJWT(headerBuilder.build(), JWTClaimsSet.parse(payload)); - signedJWT.sign(signer); - } catch (ParseException | JOSEException e) { - throw new OpenBankingRuntimeException("Error occurred while signing JWT"); - } - return signedJWT.serialize(); - } - - /** - * Util method to generate SP meta data using service provider. - * - * @param serviceProvider The service provider - * @return SP meta data as a Map - */ - public static Map getSpMetaData(ServiceProvider serviceProvider) { - - Map originalData = Arrays.stream(serviceProvider.getSpProperties()) - .collect(Collectors.toMap(ServiceProviderProperty::getName, ServiceProviderProperty::getValue)); - - Map spMetaDataMap = new HashMap<>(); - - for (Map.Entry data : originalData.entrySet()) { - - if (data.getValue().contains(DCRCommonConstants.ARRAY_ELEMENT_SEPERATOR)) { - ArrayList dataList = new ArrayList<>(Arrays.asList(data.getValue() - .split(DCRCommonConstants.ARRAY_ELEMENT_SEPERATOR))); - spMetaDataMap.put(data.getKey(), dataList); - } else { - spMetaDataMap.put(data.getKey(), data.getValue()); - } - } - return spMetaDataMap; - } - - /** - * Method to obtain Hash Value for a given String, default algorithm SHA256withRSA. - * - * @param value String value that required to be Hashed - * @return Hashed String - * @throws IdentityOAuth2Exception - */ - public static String getHashValue(String value, String digestAlgorithm) throws IdentityOAuth2Exception { - - if (digestAlgorithm == null) { - JWSAlgorithm digAlg = OAuth2Util.mapSignatureAlgorithmForJWSAlgorithm( - OAuthServerConfiguration.getInstance().getIdTokenSignatureAlgorithm()); - digestAlgorithm = OAuth2Util.mapDigestAlgorithm(digAlg); - } - MessageDigest md; - try { - md = MessageDigest.getInstance(digestAlgorithm); - } catch (NoSuchAlgorithmException e) { - throw new IdentityOAuth2Exception("Error creating the hash value. Invalid Digest Algorithm: " + - digestAlgorithm); - } - //generating hash value - md.update(value.getBytes(Charsets.UTF_8)); - byte[] digest = md.digest(); - int leftHalfBytes = digest.length / 2; - byte[] leftmost = new byte[leftHalfBytes]; - System.arraycopy(digest, 0, leftmost, 0, leftHalfBytes); - - return Base64.getUrlEncoder().withoutPadding().encodeToString(leftmost) - .replace("\n", "").replace("\r", ""); - } - - /** - * This method returns the configuration value on whether the JWT payload validation needs to be performed in the - * consent validation endpoint. - * @return config value - */ - public static boolean getConsentJWTPayloadValidatorConfigEnabled() { - return Boolean.parseBoolean(String.valueOf(IdentityExtensionsDataHolder.getInstance().getConfigurationMap() - .getOrDefault(IdentityCommonConstants.CONSENT_JWT_PAYLOAD_VALIDATION, true))); - } - - /** - * This method returns the configured JWK URI value of the transport certificate. - * @return - */ - public static String getJWKURITransportCert() { - return String.valueOf(IdentityExtensionsDataHolder.getInstance().getConfigurationMap() - .getOrDefault(IdentityCommonConstants.TLS_CERT_JWKS, IdentityCommonConstants.DEFAULT_JWKS_URI)); - } - - /** - * This method will return the configured DCR scope. - * @return - */ - public static String getDCRScope() { - return String.valueOf(IdentityExtensionsDataHolder.getInstance().getConfigurationMap() - .getOrDefault(IdentityCommonConstants.DCR_SCOPE, IdentityCommonConstants.DCR_INTERNAL_SCOPE)); - } - - public static Boolean getDCRModifyResponseConfig() { - return Boolean.parseBoolean(String.valueOf(IdentityExtensionsDataHolder.getInstance().getConfigurationMap() - .getOrDefault(IdentityCommonConstants.DCR_MODIFY_RESPONSE, "false"))); - } - - /** - * Retrieve certificate from servlet request attribute. - * @param certObject certificate Object. - * @return X509Certificate certificate. - */ - public static X509Certificate getCertificateFromAttribute(Object certObject) { - - if (certObject instanceof X509Certificate[]) { - X509Certificate[] cert = (X509Certificate[]) certObject; - return cert[0]; - } else if (certObject instanceof X509Certificate) { - return (X509Certificate) certObject; - } - return null; - } - - /** - * Method to decode request object and retrieve values. - * - * @param request HTTP Servlet request. - * @param key key to retrieve. - * @return value. - */ - public static String decodeRequestObjectAndGetKey(HttpServletRequest request, String key) - throws OAuthProblemException { - - if (request.getParameterMap().containsKey(IdentityCommonConstants.REQUEST_URI) && - request.getParameter(IdentityCommonConstants.REQUEST_URI) != null) { - - // Consider as PAR request - String[] requestUri = request.getParameter(IdentityCommonConstants.REQUEST_URI).split(":"); - String requestUriRef = requestUri[requestUri.length - 1]; - SessionDataCacheEntry valueFromCache = SessionDataCache.getInstance() - .getValueFromCache(new SessionDataCacheKey(requestUriRef)); - if (valueFromCache != null) { - String essentialClaims = valueFromCache.getoAuth2Parameters().getEssentialClaims(); - if (essentialClaims != null) { - String[] essentialClaimsWithExpireTime = essentialClaims.split(":"); - essentialClaims = essentialClaimsWithExpireTime[0]; - essentialClaims = essentialClaims.split("\\.")[1]; - byte[] requestObject; - try { - requestObject = Base64.getDecoder().decode(essentialClaims); - } catch (IllegalArgumentException e) { - - // Decode if the requestObject is base64-url encoded. - requestObject = Base64.getUrlDecoder().decode(essentialClaims); - } - org.json.JSONObject - requestObjectVal = - new org.json.JSONObject(new String(requestObject, StandardCharsets.UTF_8)); - return requestObjectVal.has(key) ? requestObjectVal.getString(key) : null; - } - } else { - throw OAuthProblemException.error("invalid_request_uri") - .description("Provided request URI is not valid"); - } - } - return null; - - } - - public static OAuthProblemException handleOAuthProblemException(String errorCode, String message, String state) { - - return OAuthProblemException.error(errorCode).description(message).state(state); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/resources/findbugs-exclude.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/resources/findbugs-exclude.xml deleted file mode 100644 index c4f8e532..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/resources/findbugs-exclude.xml +++ /dev/null @@ -1,21 +0,0 @@ - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/resources/findbugs-include.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/resources/findbugs-include.xml deleted file mode 100644 index 649d044e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,23 +0,0 @@ - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/HTTPClientUtilsTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/HTTPClientUtilsTest.java deleted file mode 100644 index fda5597c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/HTTPClientUtilsTest.java +++ /dev/null @@ -1,70 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.util.HTTPClientUtils; -import org.apache.http.conn.ssl.SSLSocketFactory; -import org.testng.Assert; -import org.testng.annotations.Test; - -import java.io.File; - -/** - * Test for HTTP client utils. - */ -public class HTTPClientUtilsTest { - - String path = "src/test/resources"; - File file = new File(path); - String absolutePathForTestResources = file.getAbsolutePath(); - - @Test - public void testLoadKeystore() throws OpenBankingException { - - Assert.assertNotNull(HTTPClientUtils.loadKeyStore(absolutePathForTestResources + "/wso2carbon.jks", - "wso2carbon")); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testLoadInvalidKeystore() throws OpenBankingException { - - HTTPClientUtils.loadKeyStore(absolutePathForTestResources + "/wso2carbon2.jks", - "wso2carbon"); - } - - @Test - public void testHostNameVerifier() throws OpenBankingException { - - Assert.assertEquals(HTTPClientUtils.getX509HostnameVerifier(), - SSLSocketFactory.BROWSER_COMPATIBLE_HOSTNAME_VERIFIER); - - System.setProperty(HTTPClientUtils.HOST_NAME_VERIFIER, - String.valueOf(SSLSocketFactory.ALLOW_ALL_HOSTNAME_VERIFIER)); - - Assert.assertEquals(HTTPClientUtils.getX509HostnameVerifier(), - SSLSocketFactory.BROWSER_COMPATIBLE_HOSTNAME_VERIFIER); - - System.setProperty(HTTPClientUtils.HOST_NAME_VERIFIER, - String.valueOf(SSLSocketFactory.STRICT_HOSTNAME_VERIFIER)); - - Assert.assertEquals(HTTPClientUtils.getX509HostnameVerifier(), - SSLSocketFactory.STRICT_HOSTNAME_VERIFIER); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthUtilsTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthUtilsTest.java deleted file mode 100644 index f3b1b2dd..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthUtilsTest.java +++ /dev/null @@ -1,194 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.app2app.utils.App2AppAuthUtils; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedUser; -import org.wso2.carbon.identity.application.authenticator.push.device.handler.exception.PushDeviceHandlerClientException; -import org.wso2.carbon.identity.application.authenticator.push.device.handler.exception.PushDeviceHandlerServerException; -import org.wso2.carbon.identity.application.authenticator.push.device.handler.impl.DeviceHandlerImpl; -import org.wso2.carbon.identity.application.authenticator.push.device.handler.model.Device; -import org.wso2.carbon.identity.core.util.IdentityTenantUtil; -import org.wso2.carbon.user.api.UserRealm; -import org.wso2.carbon.user.api.UserStoreException; -import org.wso2.carbon.user.core.common.AbstractUserStoreManager; -import org.wso2.carbon.user.core.service.RealmService; - -import java.util.ArrayList; -import java.util.List; - -/** - * Test class for Unit Testing App2AppAuthUtils. - */ -@PrepareForTest({AuthenticatedUser.class, IdentityTenantUtil.class, IdentityExtensionsDataHolder.class}) -@PowerMockIgnore({"javax.net.ssl.*", "jdk.internal.reflect.*"}) -public class App2AppAuthUtilsTest { - - @Test - public void testGetAuthenticatedUserFromSubjectIdentifier() { - - PowerMockito.mockStatic(AuthenticatedUser.class); - // Prepare test data - String subjectIdentifier = "admin@wso2.com"; - // Mock the AuthenticatedUser class - AuthenticatedUser authenticatedUserMock = Mockito.mock(AuthenticatedUser.class); - // Mock the behavior of AuthenticatedUser.createLocalAuthenticatedUserFromSubjectIdentifier() - Mockito.when(AuthenticatedUser.createLocalAuthenticatedUserFromSubjectIdentifier(subjectIdentifier)) - .thenReturn(authenticatedUserMock); - // Call the method under test - AuthenticatedUser user = App2AppAuthUtils.getAuthenticatedUserFromSubjectIdentifier(subjectIdentifier); - // Verify the result - Assert.assertNotNull(user, "Authenticated user should not be null"); - Assert.assertEquals(user, authenticatedUserMock, "Returned user should match the mocked user"); - } - - @Test - public void testGetUserRealm() throws UserStoreException { - - // Mock the AuthenticatedUser - AuthenticatedUser authenticatedUserMock = Mockito.mock(AuthenticatedUser.class); - Mockito.when(authenticatedUserMock.getTenantDomain()).thenReturn("testTenantDomain"); - // Mock IdentityTenantUtil - PowerMockito.mockStatic(IdentityTenantUtil.class); - Mockito.when(IdentityTenantUtil.getTenantId(Mockito.anyString())).thenReturn(1234); - // Mock RealmService and UserRealm - RealmService realmServiceMock = Mockito.mock(RealmService.class); - UserRealm userRealmMock = Mockito.mock(UserRealm.class); - Mockito.when(realmServiceMock.getTenantUserRealm(1234)).thenReturn(userRealmMock); - // Mock IdentityExtensionsDataHolder - IdentityExtensionsDataHolder dataHolderMock = Mockito.mock(IdentityExtensionsDataHolder.class); - Mockito.when(dataHolderMock.getRealmService()).thenReturn(realmServiceMock); - PowerMockito.mockStatic(IdentityExtensionsDataHolder.class); - Mockito.when(IdentityExtensionsDataHolder.getInstance()).thenReturn(dataHolderMock); - // Call the method under test - UserRealm userRealm = App2AppAuthUtils.getUserRealm(authenticatedUserMock); - // Verify the result - Assert.assertEquals(userRealm, userRealmMock, "UserRealm should match the mocked UserRealm"); - } - - @Test - public void testGetUserRealmWhenUserIsNull() throws UserStoreException { - - // Call the method under test - UserRealm userRealm = App2AppAuthUtils.getUserRealm(null); - // Verify the result - Assert.assertNull(userRealm, "UserRealm should be null when the input is null."); - } - - @Test - public void testGetUserIdFromUsername() throws UserStoreException, OpenBankingException { - - // Prepare test data - String username = "admin@wso2.com"; - String userIDMock = "354cd9f4-ae85-4ce9-8c42-dc1111ac8acf"; - // Mock the UserRealm - UserRealm userRealmMock = Mockito.mock(UserRealm.class); - // Mock the AbstractUserStoreManager - AbstractUserStoreManager userStoreManagerMock = Mockito.mock(AbstractUserStoreManager.class); - Mockito.when(userStoreManagerMock.getUserIDFromUserName(username)).thenReturn(userIDMock); - // Mock the RealmService - Mockito.when(userRealmMock.getUserStoreManager()).thenReturn(userStoreManagerMock); - // Call the method under test - String userId = App2AppAuthUtils.getUserIdFromUsername(username, userRealmMock); - // Verify the result - Assert.assertNotNull(userId, "User ID should not be null"); - Assert.assertEquals(userId, userIDMock , - "User ID should match the expected value"); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testGetUserIdFromUsernameWhenRealmNull() throws UserStoreException, OpenBankingException { - - // Prepare test data - String username = "admin@wso2.com"; - // Mock the UserRealm - UserRealm userRealmMock = null; - // Call the method under test - String userId = App2AppAuthUtils.getUserIdFromUsername(username, userRealmMock); - } - - @Test - public void testGetPublicKey() throws PushDeviceHandlerServerException, PushDeviceHandlerClientException, - OpenBankingException { - - // Prepare test data - String deviceID = "testDeviceID"; - String invalidDeviceId = "invalidDeviceID"; - String userID = "testUserID"; - String publicKey = "testPublicKey"; - // Mock DeviceHandlerImpl and Device - DeviceHandlerImpl deviceHandlerMock = Mockito.mock(DeviceHandlerImpl.class); - Device deviceMockI = Mockito.mock(Device.class); - Device deviceMockII = Mockito.mock(Device.class); - Mockito.when(deviceMockI.getPublicKey()).thenReturn(publicKey); - Mockito.when(deviceMockI.getDeviceId()).thenReturn(deviceID); - Mockito.when(deviceMockII.getPublicKey()).thenReturn(publicKey); - Mockito.when(deviceMockII.getDeviceId()).thenReturn(invalidDeviceId); - // Mock DeviceHandlerImpl.listDevices() to return a list with the mock device - List deviceList = new ArrayList<>(); - deviceList.add(deviceMockI); - deviceList.add(deviceMockII); - Mockito.when(deviceHandlerMock.listDevices(userID)).thenReturn(deviceList); - Mockito.when(deviceHandlerMock.getPublicKey(deviceID)).thenReturn(publicKey); - // Call the method under test - String result = App2AppAuthUtils.getPublicKey(deviceID, userID, deviceHandlerMock); - // Verify the result - Assert.assertEquals(result, publicKey, "Public key should match"); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void testGetPublicKeyInvalidDeviceID() throws PushDeviceHandlerServerException, - PushDeviceHandlerClientException, OpenBankingException { - - // Prepare test data - String deviceID = "testDeviceID"; - String invalidDeviceId = "invalidDeviceID"; - String userID = "testUserID"; - String publicKey = "testPublicKey"; - // Mock DeviceHandlerImpl and Device - DeviceHandlerImpl deviceHandlerMock = Mockito.mock(DeviceHandlerImpl.class); - Device deviceMock = Mockito.mock(Device.class); - Mockito.when(deviceMock.getPublicKey()).thenReturn(publicKey); - Mockito.when(deviceMock.getDeviceId()).thenReturn(invalidDeviceId); - // Mock DeviceHandlerImpl.listDevices() to return a list with the mock device - List deviceList = new ArrayList<>(); - deviceList.add(deviceMock); - Mockito.when(deviceHandlerMock.listDevices(userID)).thenReturn(deviceList); - Mockito.when(deviceHandlerMock.getPublicKey(userID)).thenReturn(publicKey); - // Call the method under test - String result = App2AppAuthUtils.getPublicKey(deviceID, userID, deviceHandlerMock); - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthValidationTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthValidationTest.java deleted file mode 100644 index 36e24abd..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthValidationTest.java +++ /dev/null @@ -1,179 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app; - -import com.nimbusds.jose.JOSEException; -import com.nimbusds.jwt.SignedJWT; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.identity.app2app.cache.JTICache; -import com.wso2.openbanking.accelerator.identity.app2app.exception.JWTValidationException; -import com.wso2.openbanking.accelerator.identity.app2app.model.DeviceVerificationToken; -import com.wso2.openbanking.accelerator.identity.app2app.testutils.App2AppUtilsTestJWTDataProvider; -import com.wso2.openbanking.accelerator.identity.app2app.utils.App2AppAuthUtils; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.testng.IObjectFactory; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; - -import java.security.NoSuchAlgorithmException; -import java.security.spec.InvalidKeySpecException; -import java.text.ParseException; -import java.util.Date; - -/** - * Test class for unit testing App2AppAuthValidations. - */ -@PrepareForTest({JTICache.class, JWTUtils.class}) -@PowerMockIgnore({"javax.net.ssl.*", "jdk.internal.reflect.*"}) -public class App2AppAuthValidationTest { - - @Test(dataProviderClass = App2AppUtilsTestJWTDataProvider.class, - dataProvider = "ValidJWTProvider") - public void validationTest(String jwtString, String publicKey, String requestObject) throws ParseException, - OpenBankingException, JOSEException, NoSuchAlgorithmException, InvalidKeySpecException { - - //Mocking JTICache and JWTUtils - PowerMockito.mockStatic(JTICache.class); - PowerMockito.mockStatic(JWTUtils.class); - Mockito.when(JTICache.getJtiDataFromCache(Mockito.anyString())).thenReturn(null); - Mockito.when(JWTUtils.isValidSignature(Mockito.any(SignedJWT.class), Mockito.anyString())) - .thenReturn(true); - Mockito.when(JWTUtils.isValidExpiryTime(Mockito.any(Date.class), Mockito.any(long.class))) - .thenReturn(true); - Mockito.when(JWTUtils.isValidNotValidBeforeTime(Mockito.any(Date.class), Mockito.any(long.class))) - .thenReturn(true); - //Creating a new device verification token using signed jwt - SignedJWT signedJWT = SignedJWT.parse(jwtString); - DeviceVerificationToken deviceVerificationToken = new DeviceVerificationToken(signedJWT); - deviceVerificationToken.setPublicKey(publicKey); - deviceVerificationToken.setRequestObject(requestObject); - // Call the method under test - App2AppAuthUtils.validateToken(deviceVerificationToken); - } - - @Test(expectedExceptions = JWTValidationException.class, - dataProviderClass = App2AppUtilsTestJWTDataProvider.class, - dataProvider = "ValidJWTProvider") - public void validationTestJTIReplayed(String jwtString, String publicKey, String requestObject) throws - ParseException, OpenBankingException, JOSEException, NoSuchAlgorithmException, InvalidKeySpecException { - - //Mocking JTICache and JWTUtils - PowerMockito.mockStatic(JTICache.class); - PowerMockito.mockStatic(JWTUtils.class); - Mockito.when(JTICache.getJtiDataFromCache(Mockito.anyString())).thenReturn("NotNullJTI"); - Mockito.when(JWTUtils.isValidSignature(Mockito.any(SignedJWT.class), Mockito.anyString())) - .thenReturn(true); - Mockito.when(JWTUtils.isValidExpiryTime(Mockito.any(Date.class), Mockito.any(long.class))) - .thenReturn(true); - Mockito.when(JWTUtils.isValidNotValidBeforeTime(Mockito.any(Date.class), Mockito.any(long.class))) - .thenReturn(true); - //Creating a new device verification token using signed jwt - SignedJWT signedJWT = SignedJWT.parse(jwtString); - DeviceVerificationToken deviceVerificationToken = new DeviceVerificationToken(signedJWT); - deviceVerificationToken.setPublicKey(publicKey); - deviceVerificationToken.setRequestObject(requestObject); - // Call the method under test - App2AppAuthUtils.validateToken(deviceVerificationToken); - } - - @Test(expectedExceptions = JWTValidationException.class, - dataProviderClass = App2AppUtilsTestJWTDataProvider.class, - dataProvider = "ValidJWTProvider") - public void validationTestJWTExpired(String jwtString, String publicKey, String requestObject) throws - ParseException, OpenBankingException, JOSEException, NoSuchAlgorithmException, InvalidKeySpecException { - - //Mocking JTICache and JWTUtils - PowerMockito.mockStatic(JTICache.class); - PowerMockito.mockStatic(JWTUtils.class); - Mockito.when(JTICache.getJtiDataFromCache(Mockito.anyString())).thenReturn(null); - Mockito.when(JWTUtils.isValidSignature(Mockito.any(SignedJWT.class), Mockito.anyString())) - .thenReturn(true); - Mockito.when(JWTUtils.isValidExpiryTime(Mockito.any(Date.class), Mockito.any(long.class))) - .thenReturn(false); - Mockito.when(JWTUtils.isValidNotValidBeforeTime(Mockito.any(Date.class), Mockito.any(long.class))) - .thenReturn(true); - //Creating a new device verification token using signed jwt - SignedJWT signedJWT = SignedJWT.parse(jwtString); - DeviceVerificationToken deviceVerificationToken = new DeviceVerificationToken(signedJWT); - deviceVerificationToken.setPublicKey(publicKey); - deviceVerificationToken.setRequestObject(requestObject); - // Call the method under test - App2AppAuthUtils.validateToken(deviceVerificationToken); - } - - @Test(expectedExceptions = JWTValidationException.class, - dataProviderClass = App2AppUtilsTestJWTDataProvider.class, - dataProvider = "ValidJWTProvider") - public void validationTestJWTNotActive(String jwtString, String publicKey, String requestObject) throws - ParseException, OpenBankingException, JOSEException, NoSuchAlgorithmException, InvalidKeySpecException { - - //Mocking JTICache and JWTUtils - PowerMockito.mockStatic(JTICache.class); - PowerMockito.mockStatic(JWTUtils.class); - Mockito.when(JTICache.getJtiDataFromCache(Mockito.anyString())).thenReturn(null); - Mockito.when(JWTUtils.isValidSignature(Mockito.any(SignedJWT.class), Mockito.anyString())). - thenReturn(true); - Mockito.when(JWTUtils.isValidExpiryTime(Mockito.any(Date.class), Mockito.any(long.class))) - .thenReturn(true); - Mockito.when(JWTUtils.isValidNotValidBeforeTime(Mockito.any(Date.class), Mockito.any(long.class))) - .thenReturn(false); - //Creating a new device verification token using signed jwt - SignedJWT signedJWT = SignedJWT.parse(jwtString); - DeviceVerificationToken deviceVerificationToken = new DeviceVerificationToken(signedJWT); - deviceVerificationToken.setPublicKey(publicKey); - deviceVerificationToken.setRequestObject(requestObject); - // Call the method under test - App2AppAuthUtils.validateToken(deviceVerificationToken); - } - - @Test(expectedExceptions = JWTValidationException.class, - dataProviderClass = App2AppUtilsTestJWTDataProvider.class, - dataProvider = "invalidDigestProvider") - public void validationTestInvalidDigest(String jwtString, String publicKey, String requestObject) throws - ParseException, OpenBankingException, JOSEException, NoSuchAlgorithmException, InvalidKeySpecException { - - //Mocking JTICache and JWTUtils - PowerMockito.mockStatic(JTICache.class); - PowerMockito.mockStatic(JWTUtils.class); - Mockito.when(JTICache.getJtiDataFromCache(Mockito.anyString())).thenReturn(null); - Mockito.when(JWTUtils.isValidSignature(Mockito.any(SignedJWT.class), Mockito.anyString())). - thenReturn(true); - Mockito.when(JWTUtils.isValidExpiryTime(Mockito.any(Date.class), Mockito.any(long.class))) - .thenReturn(true); - Mockito.when(JWTUtils.isValidNotValidBeforeTime(Mockito.any(Date.class), Mockito.any(long.class))) - .thenReturn(true); - //Creating a new device verification token using signed jwt - SignedJWT signedJWT = SignedJWT.parse(jwtString); - DeviceVerificationToken deviceVerificationToken = new DeviceVerificationToken(signedJWT); - deviceVerificationToken.setPublicKey(publicKey); - deviceVerificationToken.setRequestObject(requestObject); - // Call the method under test - App2AppAuthUtils.validateToken(deviceVerificationToken); - } - @ObjectFactory - public IObjectFactory getObjectFactory() { - - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthenticatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthenticatorTest.java deleted file mode 100644 index 812b4e08..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/App2AppAuthenticatorTest.java +++ /dev/null @@ -1,281 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.identity.app2app.testutils.App2AppAuthenticatorTestDataProvider; -import com.wso2.openbanking.accelerator.identity.app2app.utils.App2AppAuthUtils; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.application.authentication.framework.context.AuthenticationContext; -import org.wso2.carbon.identity.application.authentication.framework.exception.AuthenticationFailedException; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedUser; -import org.wso2.carbon.identity.application.authenticator.push.device.handler.exception.PushDeviceHandlerClientException; -import org.wso2.carbon.identity.application.authenticator.push.device.handler.exception.PushDeviceHandlerServerException; -import org.wso2.carbon.user.api.UserStoreException; - -import java.text.ParseException; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; - -import static org.testng.Assert.assertEquals; - -/** - * Test class for unit testing App2AppAuthenticator. - */ -@PrepareForTest({App2AppAuthUtils.class, JWTUtils.class}) -@PowerMockIgnore({"javax.net.ssl.*", "jdk.internal.reflect.*"}) -public class App2AppAuthenticatorTest { - - private HttpServletRequest mockRequest; - private HttpServletResponse mockResponse; - - private AuthenticationContext mockAuthenticationContext; - private App2AppAuthenticator app2AppAuthenticator; - - @BeforeTest - public void setup() { - - // setting the authenticator for testing - app2AppAuthenticator = new App2AppAuthenticator(); - //Mocking the behaviour of request, response and authenticationContext - mockRequest = Mockito.mock(HttpServletRequest.class); - mockResponse = Mockito.mock(HttpServletResponse.class); - mockAuthenticationContext = Mockito.mock(AuthenticationContext.class); - } - - @Test - public void testGetName() { - - String expectedName = App2AppAuthenticatorConstants.AUTHENTICATOR_NAME; - String actualName = app2AppAuthenticator.getName(); - // Invoke the method under test - assertEquals(actualName, expectedName, "Expected and actual names should match."); - } - - @Test - public void testGetFriendlyName() { - - String expectedFriendlyName = App2AppAuthenticatorConstants.AUTHENTICATOR_FRIENDLY_NAME; - String actualFriendlyName = app2AppAuthenticator.getFriendlyName(); - // Invoke the method under test - assertEquals(actualFriendlyName, expectedFriendlyName, - "Expected and actual friendly names should match"); - } - - @Test(dataProviderClass = App2AppAuthenticatorTestDataProvider.class , - dataProvider = "app_auth_identifier_provider") - public void canHandleTestCase(String secret, String expected) { - - // Set up mock behavior for HttpServletRequest - Mockito.when(mockRequest.getParameter(App2AppAuthenticatorConstants.DEVICE_VERIFICATION_TOKEN_IDENTIFIER)) - .thenReturn(secret); - // Invoke the method under test - assertEquals(Boolean.valueOf(expected).booleanValue(), app2AppAuthenticator.canHandle(mockRequest), - "Invalid can handle response for the request."); - } - - @Test(expectedExceptions = AuthenticationFailedException.class) - public void initiateAuthenticationRequest() throws AuthenticationFailedException { - - // Invoke the method under test - app2AppAuthenticator.initiateAuthenticationRequest(mockRequest, mockResponse, mockAuthenticationContext); - } - - @Test(dataProviderClass = App2AppAuthenticatorTestDataProvider.class, - dataProvider = "sessionDataKeyProvider") - public void getContextIdentifierTest(String sessionDataKey) { - - // Set up mock behavior for HttpServletRequest - Mockito.when(mockRequest.getParameter(App2AppAuthenticatorConstants.SESSION_DATA_KEY)) - .thenReturn(sessionDataKey); - // Invoke the method under test - String output = app2AppAuthenticator.getContextIdentifier(mockRequest); - assertEquals(sessionDataKey, output); - } - - @Test(dataProviderClass = App2AppAuthenticatorTestDataProvider.class, - dataProvider = "AppAuthIdentifierProvider") - public void testProcessAuthenticationResponse_success(String jwtString) { - - PowerMockito.mockStatic(App2AppAuthUtils.class); - // Set up mock behavior for HttpServletRequest - Mockito.when(mockRequest.getParameter(App2AppAuthenticatorConstants.DEVICE_VERIFICATION_TOKEN_IDENTIFIER)) - .thenReturn(jwtString); - // Mock the authenticated user - AuthenticatedUser authenticatedUserMock = Mockito.mock(AuthenticatedUser.class); - // Mock the behavior of App2AppAuthUtils.getAuthenticatedUserFromSubjectIdentifier() to return a mocked user - Mockito.when(App2AppAuthUtils.getAuthenticatedUserFromSubjectIdentifier(Mockito.anyString())) - .thenReturn(authenticatedUserMock); - - try { - app2AppAuthenticator.processAuthenticationResponse(mockRequest, mockResponse, mockAuthenticationContext); - // Verify that the authentication context subject is set (or any other verification) - Mockito.verify(mockAuthenticationContext).setSubject(authenticatedUserMock); - } catch (Exception e) { - // If any unexpected exception occurs, fail the test - Assert.fail("Unexpected exception occurred: " + e.getMessage()); - } - } - @Test(expectedExceptions = AuthenticationFailedException.class, - dataProviderClass = App2AppAuthenticatorTestDataProvider.class, - dataProvider = "AppAuthIdentifierProvider" - ) - public void testProcessAuthenticationResponse_IllegalArgumentException(String jwtString) - throws AuthenticationFailedException { - - PowerMockito.mockStatic(App2AppAuthUtils.class); - // Mock the behavior of HttpServletRequest to return a value for login hint - Mockito.when(mockRequest.getParameter(App2AppAuthenticatorConstants.DEVICE_VERIFICATION_TOKEN_IDENTIFIER)) - .thenReturn(jwtString); - // Mock App2AppAuthUtils.getAuthenticatedUserFromSubjectIdentifier to throw IllegalArgumentException - Mockito.when(App2AppAuthUtils.getAuthenticatedUserFromSubjectIdentifier(Mockito.anyString())) - .thenThrow(new IllegalArgumentException("Failed to create Local Authenticated User from the given " + - "subject identifier. Invalid argument. authenticatedSubjectIdentifier : ")); - // Invoke the method under test - app2AppAuthenticator.processAuthenticationResponse(mockRequest, mockResponse, mockAuthenticationContext); - } - - @Test(expectedExceptions = AuthenticationFailedException.class, - dataProviderClass = App2AppAuthenticatorTestDataProvider.class, - dataProvider = "AppAuthIdentifierProvider" - ) - public void testProcessAuthenticationResponse_ParseException(String jwtString) - throws AuthenticationFailedException, ParseException { - - PowerMockito.mockStatic(JWTUtils.class); - // Mock the behavior of HttpServletRequest to return a value for login hint - Mockito.when(mockRequest.getParameter(App2AppAuthenticatorConstants.DEVICE_VERIFICATION_TOKEN_IDENTIFIER)) - .thenReturn(jwtString); - // Mock App2AppAuthUtils.getAuthenticatedUserFromSubjectIdentifier to throw IllegalArgumentException - Mockito.when(JWTUtils.getSignedJWT(Mockito.anyString())) - .thenThrow(new ParseException("JWT Not parsable.", 1)); - // Invoke the method under test - app2AppAuthenticator.processAuthenticationResponse(mockRequest, mockResponse, mockAuthenticationContext); - } - - @Test(expectedExceptions = AuthenticationFailedException.class, - dataProviderClass = App2AppAuthenticatorTestDataProvider.class, - dataProvider = "AppAuthIdentifierProvider" - ) - public void testProcessAuthenticationResponse_UserStoreException(String jwtString) - throws AuthenticationFailedException, UserStoreException { - - PowerMockito.mockStatic(App2AppAuthUtils.class); - // Mock the behavior of HttpServletRequest to return a value for login hint - Mockito.when(mockRequest.getParameter(App2AppAuthenticatorConstants.DEVICE_VERIFICATION_TOKEN_IDENTIFIER)) - .thenReturn(jwtString); - // Mock the behavior of App2AppAuthUtils.getAuthenticatedUserFromSubjectIdentifier() to return a mock user - AuthenticatedUser authenticatedUserMock = Mockito.mock(AuthenticatedUser.class); - Mockito.when(App2AppAuthUtils.getAuthenticatedUserFromSubjectIdentifier(Mockito.anyString())) - .thenReturn(authenticatedUserMock); - // Mock the behavior of getPublicKeyByDeviceID() to throw UserStoreException - Mockito.when(App2AppAuthUtils.getUserRealm(Mockito.any(AuthenticatedUser.class))) - .thenThrow(new UserStoreException(App2AppAuthenticatorConstants.USER_STORE_EXCEPTION_MESSAGE)); - // Invoke the method under test - app2AppAuthenticator.processAuthenticationResponse(mockRequest, mockResponse, mockAuthenticationContext); - } - - @Test(expectedExceptions = AuthenticationFailedException.class, - dataProviderClass = App2AppAuthenticatorTestDataProvider.class, - dataProvider = "AppAuthIdentifierProvider" - ) - public void testProcessAuthenticationResponse_PushDeviceHandlerServerException(String jwtString) - throws AuthenticationFailedException, OpenBankingException, PushDeviceHandlerServerException, - PushDeviceHandlerClientException { - - PowerMockito.mockStatic(App2AppAuthUtils.class); - // Mock the behavior of HttpServletRequest to return a value for login hint - Mockito.when(mockRequest.getParameter(App2AppAuthenticatorConstants.DEVICE_VERIFICATION_TOKEN_IDENTIFIER)) - .thenReturn(jwtString); - // Mock the behavior of App2AppAuthUtils.getAuthenticatedUserFromSubjectIdentifier() to return a mock user - AuthenticatedUser authenticatedUserMock = Mockito.mock(AuthenticatedUser.class); - Mockito.when(App2AppAuthUtils.getAuthenticatedUserFromSubjectIdentifier(Mockito.anyString())) - .thenReturn(authenticatedUserMock); - // Mock the behavior of getPublicKeyByDeviceID() to throw UserStoreException - Mockito.when(App2AppAuthUtils.getPublicKey(Mockito.anyString(), Mockito.anyString(), Mockito.any())) - .thenThrow(new PushDeviceHandlerServerException( - App2AppAuthenticatorConstants.PUSH_DEVICE_HANDLER_SERVER_EXCEPTION_MESSAGE)); - // Invoke the method under test - app2AppAuthenticator.processAuthenticationResponse(mockRequest, mockResponse, mockAuthenticationContext); - } - - @Test(expectedExceptions = AuthenticationFailedException.class, - dataProviderClass = App2AppAuthenticatorTestDataProvider.class, - dataProvider = "AppAuthIdentifierProvider" - ) - public void testProcessAuthenticationResponse_PushDeviceHandlerClientException(String jwtString) - throws AuthenticationFailedException, OpenBankingException, PushDeviceHandlerServerException, - PushDeviceHandlerClientException { - - PowerMockito.mockStatic(App2AppAuthUtils.class); - // Mock the behavior of HttpServletRequest to return a value for login hint - Mockito.when(mockRequest.getParameter(App2AppAuthenticatorConstants.DEVICE_VERIFICATION_TOKEN_IDENTIFIER)) - .thenReturn(jwtString); - // Mock the behavior of App2AppAuthUtils.getAuthenticatedUserFromSubjectIdentifier() to return a mock user - AuthenticatedUser authenticatedUserMock = Mockito.mock(AuthenticatedUser.class); - Mockito.when(App2AppAuthUtils.getAuthenticatedUserFromSubjectIdentifier(Mockito.anyString())) - .thenReturn(authenticatedUserMock); - // Mock the behavior of getPublicKeyByDeviceID() to throw UserStoreException - Mockito.when(App2AppAuthUtils.getPublicKey(Mockito.anyString(), Mockito.anyString(), Mockito.any())) - .thenThrow(new PushDeviceHandlerClientException( - App2AppAuthenticatorConstants.PUSH_DEVICE_HANDLER_CLIENT_EXCEPTION_MESSAGE)); - // Invoke the method under test - app2AppAuthenticator.processAuthenticationResponse(mockRequest, mockResponse, mockAuthenticationContext); - } - - @Test(expectedExceptions = AuthenticationFailedException.class, - dataProviderClass = App2AppAuthenticatorTestDataProvider.class, - dataProvider = "AppAuthIdentifierProvider" - ) - public void testProcessAuthenticationResponse_OpenBankingException(String jwtString) - throws AuthenticationFailedException, OpenBankingException, PushDeviceHandlerServerException, - PushDeviceHandlerClientException { - - PowerMockito.mockStatic(App2AppAuthUtils.class); - // Mock the behavior of HttpServletRequest to return a value for login hint - Mockito.when(mockRequest.getParameter(App2AppAuthenticatorConstants.DEVICE_VERIFICATION_TOKEN_IDENTIFIER)) - .thenReturn(jwtString); - // Mock the behavior of App2AppAuthUtils.getAuthenticatedUserFromSubjectIdentifier() to return a mock user - AuthenticatedUser authenticatedUserMock = Mockito.mock(AuthenticatedUser.class); - Mockito.when(App2AppAuthUtils.getAuthenticatedUserFromSubjectIdentifier(Mockito.anyString())) - .thenReturn(authenticatedUserMock); - // Mock the behavior of getPublicKeyByDeviceID() to throw UserStoreException - Mockito.when(App2AppAuthUtils.getPublicKey(Mockito.anyString(), Mockito.anyString(), Mockito.any())) - .thenThrow(new OpenBankingException( - App2AppAuthenticatorConstants.OPEN_BANKING_EXCEPTION_MESSAGE)); - // Invoke the method under test - app2AppAuthenticator.processAuthenticationResponse(mockRequest, mockResponse, mockAuthenticationContext); - } - @ObjectFactory - public IObjectFactory getObjectFactory() { - - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/testutils/App2AppAuthenticatorTestDataProvider.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/testutils/App2AppAuthenticatorTestDataProvider.java deleted file mode 100644 index ac39da2e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/testutils/App2AppAuthenticatorTestDataProvider.java +++ /dev/null @@ -1,62 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app.testutils; - -import org.testng.annotations.DataProvider; - -/** - * Data Provider class for testing App2AppAuthenticator. - */ -public class App2AppAuthenticatorTestDataProvider { - private static final String validAppAuthIdentifier = - "eyJhbGciOiJSUzI1NiIsInR5cCI6IkpXVCJ9.eyJkaWQiOiI1NTBmNDQ1My05NTQ3LTRlNGYtYmUwNi04ZGIyZWVkNTYzYjMiLCJsb" + - "2dpbl9oaW50IjoiYWRtaW5Ad3NvMi5jb20iLCJpYXQiOjE3MTYyNjQ5NTUsImp0aSI6IjA1NDU1Zjc1LTkwMmUtNDFhNi04ZDg4LWV" + - "jZTUwZDM2OTc2NSIsImRpZ2VzdCI6IlNIQS0yNTY9RWtIOGZQZ1oyVFkyWEduczhjNVZ2Y2U4aDNEQjgzVit3NDd6SGl5WWZpUT0iL" + - "CJleHAiOjE3MTYyNjY3NTUsIm5iZiI6MTcxNjI2NDk1NX0.C0OGMkkaosP2FSLFtqmCgRhrCG7nCJCDLsikkbFWwc5NdzxCFyYUQVI" + - "Zx4HIRQdabg5K8Ox-WYeqwdhajaKs5Uk63tz5UjlPzX0IKsklXgnWUxdMwfrYsu-znTce0Tc-Ph0h8a8jXF2CKTOfWxwuQvgevSqJe" + - "-K6zrbJmO8imu4"; - @DataProvider(name = "app_auth_identifier_provider") - public Object[][] getAppAuthIdentifier() { - - return new String[][]{ - {validAppAuthIdentifier, "true"}, - {null, "false"}, - {"", "false"}, - }; - } - - @DataProvider(name = "sessionDataKeyProvider") - public Object[][] getSessionDataKey() { - - return new String[][]{ - {null}, - {""}, - {"550e8400-e29b-41d4-a716-446655440000"}, - {"aaaaaaa-bbbb-Cccc-dddd-eeeeeeeeeeeee"} - }; - } - - @DataProvider(name = "AppAuthIdentifierProvider") - public Object[][] getJWT() { - return new String[][]{ - {validAppAuthIdentifier}, - }; - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/testutils/App2AppUtilsTestJWTDataProvider.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/testutils/App2AppUtilsTestJWTDataProvider.java deleted file mode 100644 index ffbc0f39..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/app2app/testutils/App2AppUtilsTestJWTDataProvider.java +++ /dev/null @@ -1,83 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.app2app.testutils; - -import org.testng.annotations.DataProvider; - -/** - * JWT Data provider for App2AppAuthValidation Testing. - */ -public class App2AppUtilsTestJWTDataProvider { - - private final String validPublicKey = - "MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQDLyl7YvRhy57IbxuhV4n7OZw0mmnnXNsDJmL4YQNXy2bRCs59pJb+TYO" + - "HsR1xCsq3WH7bX1Ik/EI3weQd2zcxNbtDAUSXSy7jRBuFm1Sk52lASBbmdeOstiqlsg9ptIp/o7u1366cRjn32cXhhsR0y" + - "/spUGy8IiXz9rJfP5bEgHQIDAQ"; - private final String validAppAuthIdentifier = - "eyJhbGciOiJSUzI1NiIsInR5cCI6IkpXVCJ9.eyJkaWQiOiI1NTBmNDQ1My05NTQ3LTRlNGYtYmUwNi04ZGIyZWVkNTYzYjMiLCJsb" + - "2dpbl9oaW50IjoiYWRtaW5Ad3NvMi5jb20iLCJpYXQiOjE3MTYyNjQ5NTUsImp0aSI6IjA1NDU1Zjc1LTkwMmUtNDFhNi04ZDg4LWV" + - "jZTUwZDM2OTc2NSIsImRpZ2VzdCI6IlNIQS0yNTY9RWtIOGZQZ1oyVFkyWEduczhjNVZ2Y2U4aDNEQjgzVit3NDd6SGl5WWZpUT0iL" + - "CJleHAiOjE3MTYyNjY3NTUsIm5iZiI6MTcxNjI2NDk1NX0.C0OGMkkaosP2FSLFtqmCgRhrCG7nCJCDLsikkbFWwc5NdzxCFyYUQVI" + - "Zx4HIRQdabg5K8Ox-WYeqwdhajaKs5Uk63tz5UjlPzX0IKsklXgnWUxdMwfrYsu-znTce0Tc-Ph0h8a8jXF2CKTOfWxwuQvgevSqJe" + - "-K6zrbJmO8imu4"; - private final String appAuthIdentifierMissingDigest = - "eyJhbGciOiJSUzI1NiIsInR5cCI6IkpXVCJ9.eyJkaWQiOiI1NTBmNDQ1My05NTQ3LTRlNGYtYmUwNi04ZGIyZWVkNTYzYjMiLCJsb" + - "2dpbl9oaW50IjoiYWRtaW5Ad3NvMi5jb20iLCJpYXQiOjE3MTYyNjcyMDMsImp0aSI6ImZkNDhmOWMzLTYyZDMtNDUzZS04MWY2LTF" + - "kMGE4ZDIzM2YzZiIsImV4cCI6MTcxNjI2OTAwMywibmJmIjoxNzE2MjY3MjAzfQ.C_G5-_McCMTz6D01XpPVfrdGlPLaKli9cqWL5K" + - "nd5ntlDq5ww7J769EJdCGt-S5sfgg5hrPRhyIWK2MJwavGTMzsp1vGdUQXQkT7z68_20k82Lms67tQLIM1VUCDc9rqz5Pule5bVqbY" + - "oZFmFlHU0Hcmvy166J6c9HlySyMC994"; - private final String appAuthIdentifierInvalidDigest = - "eyJhbGciOiJSUzI1NiIsInR5cCI6IkpXVCJ9.eyJkaWQiOiI1NTBmNDQ1My05NTQ3LTRlNGYtYmUwNi04ZGIyZWVkNTYzYjMiLCJsb" + - "2dpbl9oaW50IjoiYWRtaW5Ad3NvMi5jb20iLCJpYXQiOjE3MTYyNjc0MjYsImp0aSI6IjYyM2ZhZDY3LTc0ZDMtNDk4OS04YTc1LTE" + - "2OWYxNDQzOGUwZiIsImRpZ2VzdCI6IlNIQS0yNTY9WUJlc3lUWnhIMWtBVitMTTNKMzZDdzQrVXlQYWlKS0VydVhsdGxsbS9DRT0iL" + - "CJleHAiOjE3MTYyNjkyMjYsIm5iZiI6MTcxNjI2NzQyNn0.hsuj0osE-o_hyOif7eUvVFIfJpmzF2bDqeINj2Qq2XMQ1Lbnf7LgYMG" + - "POzmtMi1Jp9Ivwl_3Wt35PcCVko2LI2TIoG-JB8MMeWc1okwwdWGP8Rz5TWCnaXiPGeeFw4PjuV3JMbWeTFafqUFtJUX7pU-8q_hiQ" + - "zxK1mGjRTjDXRA"; - private final String validRequestObject = - "eyJraWQiOiI3ZUo4U19aZ3ZsWXhGQUZTZ2hWOXhNSlJPdmsiLCJhbGciOiJQUzI1NiIsInR5cCI6IkpXVCJ9.eyJtYXhfYWdlIjo4N" + - "jQwMCwiYXVkIjoiaHR0cHM6Ly8xOTIuMTY4LjQzLjQ5Ojk0NDYvb2F1dGgyL3Rva2VuIiwic2NvcGUiOiJhY2NvdW50cyBvcGVuaWQ" + - "iLCJpc3MiOiI2RWZaSTVOUnByTm9tZlFQWElQZjFSN0ZsNUVhIiwiY2xhaW1zIjp7ImlkX3Rva2VuIjp7ImFjciI6eyJ2YWx1ZXMiO" + - "lsidXJuOm9wZW5iYW5raW5nOnBzZDI6c2NhIiwidXJuOm9wZW5iYW5raW5nOnBzZDI6Y2EiXSwiZXNzZW50aWFsIjp0cnVlfSwib3B" + - "lbmJhbmtpbmdfaW50ZW50X2lkIjp7InZhbHVlIjoiZTkyNmE2MzItYzlkMy00MmEwLWEyM2YtMWEwMWZhNDAwOWU3IiwiZXNzZW50a" + - "WFsIjp0cnVlfX0sInVzZXJpbmZvIjp7Im9wZW5iYW5raW5nX2ludGVudF9pZCI6eyJ2YWx1ZSI6ImU5MjZhNjMyLWM5ZDMtNDJhMC1" + - "hMjNmLTFhMDFmYTQwMDllNyIsImVzc2VudGlhbCI6dHJ1ZX19fSwicmVzcG9uc2VfdHlwZSI6ImNvZGUgaWRfdG9rZW4iLCJyZWRpc" + - "mVjdF91cmkiOiJodHRwczovL3d3dy5tb2NrY29tcGFueS5jb20vcmVkaXJlY3RzL3JlZGlyZWN0MSIsInN0YXRlIjoiWVdsemNEb3p" + - "NVFE0IiwiZXhwIjoxODA3MjMzNDc4LCJub25jZSI6Im4tMFM2X1d6QTJNbCIsImNsaWVudF9pZCI6IjZFZlpJNU5ScHJOb21mUVBYS" + - "VBmMVI3Rmw1RWEifQ.nKapNc1N5AHxil-xbVpSXrDRsGYkn1YHe1jURxZMVRluDWnyRmjVce9AJ5lCl338Jg0EsU4CNmLwOSu7zmtl" + - "DCFz4fCIHLj1Q8A-C5I9cWE-nAlV1HnCR_3V7cTU4YE13ZIH7bMCqOPfBX_fpDkJeDXoSnRHQtipMPqIwNfmv7Kf4SjPpZ7kT5zmDn" + - "cHsUqotpPVoPka_-Nal0KL_-PknC31pKECcxakOFNTeAeiODZN5JIyKGFtq10jQaJi7YvDKsGg1l3rv1gUdJ4s5eXqmnxJUu4J6ocY" + - "h26Nz3l_Xc1p7XIm2HPhvSW3DpbNpE8Ej0kJkI9FgWz77QACkiO4Hg"; - - @DataProvider(name = "ValidJWTProvider") - public Object[][] getDigest() { - return new String[][]{ - {validAppAuthIdentifier, validPublicKey, null}, - {validAppAuthIdentifier, validPublicKey, validRequestObject} - }; - } - - @DataProvider(name = "invalidDigestProvider") - public Object[][] getInvalidDigest() { - return new String[][]{ - {appAuthIdentifierMissingDigest, validPublicKey, validRequestObject}, - {appAuthIdentifierInvalidDigest, validPublicKey, validRequestObject} - }; - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/application/listener/ApplicationManagementListenerTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/application/listener/ApplicationManagementListenerTest.java deleted file mode 100644 index b041f815..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/application/listener/ApplicationManagementListenerTest.java +++ /dev/null @@ -1,302 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.application.listener; - -import com.wso2.openbanking.accelerator.common.config.TextFileReader; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.dcr.validation.DCRCommonConstants; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.listener.application.ApplicationUpdaterImpl; -import com.wso2.openbanking.accelerator.identity.listener.application.OBApplicationManagementListener; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.mockito.InjectMocks; -import org.mockito.Mockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; -import org.wso2.carbon.context.CarbonContext; -import org.wso2.carbon.identity.application.common.IdentityApplicationManagementException; -import org.wso2.carbon.identity.application.common.model.IdentityProvider; -import org.wso2.carbon.identity.application.common.model.LocalAndOutboundAuthenticationConfig; -import org.wso2.carbon.identity.application.common.model.RequestPathAuthenticatorConfig; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.application.common.model.ServiceProviderProperty; -import org.wso2.carbon.identity.application.mgt.ApplicationManagementService; -import org.wso2.carbon.identity.common.testng.WithCarbonHome; -import org.wso2.carbon.identity.oauth.IdentityOAuthAdminException; -import org.wso2.carbon.identity.oauth.OAuthAdminServiceImpl; -import org.wso2.carbon.identity.oauth.dto.OAuthConsumerAppDTO; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import static org.mockito.Matchers.anyObject; -import static org.mockito.Matchers.anyString; -import static org.powermock.api.mockito.PowerMockito.mock; -import static org.powermock.api.mockito.PowerMockito.mockStatic; -import static org.powermock.api.mockito.PowerMockito.when; - -/** - * Test for application management listener. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@WithCarbonHome -@PrepareForTest({CarbonContext.class}) -public class ApplicationManagementListenerTest extends PowerMockTestCase { - - private static final Log log = LogFactory.getLog(ApplicationManagementListenerTest.class); - - ServiceProvider serviceProvider; - - IdentityExtensionsDataHolder identityExtensionsDataHolder; - - OAuthAdminServiceImpl oAuthAdminService; - - ApplicationManagementService applicationManagementService; - - ApplicationUpdaterImpl applicationUpdater; - - @InjectMocks - OBApplicationManagementListener applicationManagementListener = new OBApplicationManagementListener(); - - @InjectMocks - ApplicationUpdaterImpl applicationUpdaterImpl = new ApplicationUpdaterImpl(); - - - @BeforeClass - public void beforeClass() { - - identityExtensionsDataHolder = Mockito.mock(IdentityExtensionsDataHolder.class); - oAuthAdminService = Mockito.mock(OAuthAdminServiceImpl.class);; - applicationManagementService = Mockito.mock(ApplicationManagementService.class);; - applicationUpdater = Mockito.mock(ApplicationUpdaterImpl.class);; - - Map confMap = new HashMap<>(); - List regulatoryIssuers = new ArrayList<>(); - regulatoryIssuers.add("OpenBanking Ltd"); - regulatoryIssuers.add("CDR"); - confMap.put(IdentityCommonConstants.PRIMARY_AUTHENTICATOR_DISPLAYNAME, "Basic"); - confMap.put(IdentityCommonConstants.PRIMARY_AUTHENTICATOR_NAME, "BasicAuthenticator"); - confMap.put(IdentityCommonConstants.IDENTITY_PROVIDER_NAME, "SMSAuthentication"); - confMap.put(IdentityCommonConstants.IDENTITY_PROVIDER_STEP, "2"); - confMap.put(DCRCommonConstants.REGULATORY_ISSUERS, regulatoryIssuers); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(confMap); - IdentityExtensionsDataHolder.getInstance().setApplicationManagementService(applicationManagementService); - IdentityExtensionsDataHolder.getInstance().setOauthAdminService(oAuthAdminService); - serviceProvider = new ServiceProvider(); - ServiceProviderProperty serviceProviderProperty = new ServiceProviderProperty(); - serviceProviderProperty.setDisplayName(DCRCommonConstants.SOFTWARE_STATEMENT); - serviceProviderProperty.setName(DCRCommonConstants.SOFTWARE_STATEMENT); - serviceProviderProperty.setValue("dfdfdfd"); - ServiceProviderProperty[] spPropertyArray = new ServiceProviderProperty[1]; - spPropertyArray[0] = serviceProviderProperty; - serviceProvider.setSpProperties(spPropertyArray); - serviceProvider.setApplicationName("testApp"); - - TextFileReader textFileReader = TextFileReader.getInstance(); - textFileReader.setDirectoryPath("src/test/resources"); - } - - @Test - public void testPostApplicationCreation() throws IdentityApplicationManagementException, - IdentityOAuthAdminException { - - when(identityExtensionsDataHolder.getOauthAdminService()).thenReturn(oAuthAdminService); - when(identityExtensionsDataHolder.getApplicationManagementService()). - thenReturn(applicationManagementService); - - when(oAuthAdminService.getOAuthApplicationDataByAppName(anyString())) - .thenReturn(new OAuthConsumerAppDTO()); - - when(identityExtensionsDataHolder.getAbstractApplicationUpdater()) - .thenReturn(applicationUpdater); - - boolean isSuccess = applicationManagementListener.doPostCreateApplication(serviceProvider, - "carbon@super", "admin"); - Assert.assertTrue(isSuccess); - - } - - @Test - public void testPreUpdateApplicationCreation() throws IdentityOAuthAdminException, - IdentityApplicationManagementException { - - OAuthConsumerAppDTO oAuthConsumerAppDTO = new OAuthConsumerAppDTO(); - oAuthConsumerAppDTO.setApplicationName("testApp"); - OAuthConsumerAppDTO[] oAuthConsumerAppDTOS = new OAuthConsumerAppDTO[1]; - oAuthConsumerAppDTOS[0] = oAuthConsumerAppDTO; - - Map confMap = new HashMap<>(); - List regulatoryIssuers = new ArrayList<>(); - regulatoryIssuers.add("OpenBanking Ltd"); - regulatoryIssuers.add("CDR"); - confMap.put(DCRCommonConstants.REGULATORY_ISSUERS, regulatoryIssuers); - - when(identityExtensionsDataHolder.getOauthAdminService()).thenReturn(oAuthAdminService); - when(identityExtensionsDataHolder.getConfigurationMap()).thenReturn(confMap); - when(oAuthAdminService.getAllOAuthApplicationData()).thenReturn(oAuthConsumerAppDTOS); - applicationManagementListener.doPreUpdateApplication(serviceProvider, "carbon@super", - "admin"); - } - - @Test - public void testPostGetApplicationCreation() throws IdentityApplicationManagementException { - - applicationManagementListener.doPostGetServiceProvider(serviceProvider, "appName", - "carbon@super"); - - } - - @Test - public void testPreDeleteApplicationCreation() throws IdentityApplicationManagementException { - - applicationManagementListener.doPreDeleteApplication("appName", "carbon@super", - "admin"); - - } - - @Test - public void testSetAuthenticators() throws OpenBankingException, IdentityApplicationManagementException { - - IdentityProvider[] federatedIdPs = new IdentityProvider[1]; - IdentityProvider identityProvider = new IdentityProvider(); - identityProvider.setIdentityProviderName("SMSAuthentication"); - federatedIdPs[0] = identityProvider; - - Map confMap = IdentityExtensionsDataHolder.getInstance().getConfigurationMap(); - confMap.put(IdentityCommonConstants.IDENTITY_PROVIDER_STEP, "2"); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(confMap); - - Mockito.when(identityExtensionsDataHolder.getApplicationManagementService()). - thenReturn(applicationManagementService); - - LocalAndOutboundAuthenticationConfig localAndOutboundAuthenticationConfig = - new LocalAndOutboundAuthenticationConfig(); - - Mockito.when(applicationManagementService.getAllIdentityProviders(anyString())). - thenReturn(federatedIdPs); - applicationUpdaterImpl.setAuthenticators(true, "carbon@super", serviceProvider, - localAndOutboundAuthenticationConfig); - Assert.assertNotNull(localAndOutboundAuthenticationConfig.getAuthenticationSteps()); - - } - - @Test - public void testSetAuthenticatorsWithFederatedIdp() throws OpenBankingException, - IdentityApplicationManagementException { - - IdentityProvider[] federatedIdPs = new IdentityProvider[1]; - IdentityProvider identityProvider = new IdentityProvider(); - identityProvider.setIdentityProviderName("SMSAuthentication"); - federatedIdPs[0] = identityProvider; - - Map confMap = IdentityExtensionsDataHolder.getInstance().getConfigurationMap(); - confMap.put(IdentityCommonConstants.IDENTITY_PROVIDER_STEP, "1"); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(confMap); - - Mockito.when(identityExtensionsDataHolder.getApplicationManagementService()). - thenReturn(applicationManagementService); - - LocalAndOutboundAuthenticationConfig localAndOutboundAuthenticationConfig = - new LocalAndOutboundAuthenticationConfig(); - - Mockito.when(applicationManagementService.getAllIdentityProviders(anyString())). - thenReturn(federatedIdPs); - applicationUpdaterImpl.setAuthenticators(true, "carbon@super", serviceProvider, - localAndOutboundAuthenticationConfig); - Assert.assertNotNull(localAndOutboundAuthenticationConfig.getAuthenticationSteps()); - - } - - @Test - public void testSetConditionalAuthScript() throws OpenBankingException { - - LocalAndOutboundAuthenticationConfig localAndOutboundAuthenticationConfig = - new LocalAndOutboundAuthenticationConfig(); - - applicationUpdaterImpl.setConditionalAuthScript(true, serviceProvider, - localAndOutboundAuthenticationConfig); - - Assert.assertNotNull(localAndOutboundAuthenticationConfig.getAuthenticationScriptConfig()); - - } - - @Test - public void testDoPreUpdateApplicationOnAppUpdate() throws OpenBankingException, IdentityOAuthAdminException, - IdentityApplicationManagementException { - - System.setProperty("carbon.home", "/"); - mockStatic(CarbonContext.class); - CarbonContext carbonContext = mock(CarbonContext.class); - when(CarbonContext.getThreadLocalCarbonContext()).thenReturn(carbonContext); - when(CarbonContext.getThreadLocalCarbonContext().getUsername()).thenReturn("admin"); - - LocalAndOutboundAuthenticationConfig localAndOutboundAuthenticationConfig = - new LocalAndOutboundAuthenticationConfig(); - when(identityExtensionsDataHolder.getOauthAdminService()).thenReturn(oAuthAdminService); - when(identityExtensionsDataHolder.getApplicationManagementService()). - thenReturn(applicationManagementService); - OAuthConsumerAppDTO oAuthConsumerAppDTO = new OAuthConsumerAppDTO(); - Mockito.doNothing().when(oAuthAdminService).updateConsumerApplication(anyObject()); - List spProperties = new ArrayList<>(Arrays.asList - (serviceProvider.getSpProperties())); - serviceProvider.setSpProperties(spProperties.toArray(new ServiceProviderProperty[0])); - serviceProvider.setRequestPathAuthenticatorConfigs(new RequestPathAuthenticatorConfig[0]); - Mockito.when(applicationManagementService.getServiceProvider(serviceProvider.getApplicationID())) - .thenReturn(serviceProvider); - applicationUpdaterImpl.doPreUpdateApplication(true, oAuthConsumerAppDTO, serviceProvider, - localAndOutboundAuthenticationConfig, "carbon@super", "admin"); - } - - @Test - public void testDoPreUpdateApplicationOnAppCreate() throws OpenBankingException, IdentityOAuthAdminException { - - System.setProperty("carbon.home", "/"); - mockStatic(CarbonContext.class); - CarbonContext carbonContext = mock(CarbonContext.class); - when(CarbonContext.getThreadLocalCarbonContext()).thenReturn(carbonContext); - when(CarbonContext.getThreadLocalCarbonContext().getUsername()).thenReturn("admin"); - - LocalAndOutboundAuthenticationConfig localAndOutboundAuthenticationConfig = - new LocalAndOutboundAuthenticationConfig(); - when(identityExtensionsDataHolder.getOauthAdminService()).thenReturn(oAuthAdminService); - when(identityExtensionsDataHolder.getApplicationManagementService()). - thenReturn(applicationManagementService); - OAuthConsumerAppDTO oAuthConsumerAppDTO = new OAuthConsumerAppDTO(); - Mockito.doNothing().when(oAuthAdminService).updateConsumerApplication(anyObject()); - List spProperties = new ArrayList<>(Arrays.asList - (serviceProvider.getSpProperties())); - spProperties.add(IdentityCommonUtil.getServiceProviderProperty("AppCreateRequest", - "true")); - serviceProvider.setSpProperties(spProperties.toArray(new ServiceProviderProperty[0])); - serviceProvider.setRequestPathAuthenticatorConfigs(new RequestPathAuthenticatorConfig[0]); - applicationUpdaterImpl.doPreUpdateApplication(true, oAuthConsumerAppDTO, serviceProvider, - localAndOutboundAuthenticationConfig, "carbon@super", "admin"); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/adaptive/function/OpenBankingAuthenticationWorkerFunctionImplTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/adaptive/function/OpenBankingAuthenticationWorkerFunctionImplTest.java deleted file mode 100644 index 3dd58d42..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/adaptive/function/OpenBankingAuthenticationWorkerFunctionImplTest.java +++ /dev/null @@ -1,69 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.adaptive.function; - -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import org.json.JSONObject; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.application.authentication.framework.config.model.graph.js.JsAuthenticationContext; -import org.wso2.carbon.identity.application.authentication.framework.context.AuthenticationContext; - -import java.util.HashMap; - -import static org.testng.Assert.assertEquals; -import static org.testng.Assert.assertTrue; - -/** - * Unit test class for OpenBankingAuthenticationWorkerFunctionImpl class. - */ -public class OpenBankingAuthenticationWorkerFunctionImplTest { - - private JSONObject customJSON; - private OpenBankingAuthenticationWorkerFunction workerFunction; - private AuthenticationContext authenticationContext; - private JsAuthenticationContext jsAuthenticationContext; - - @BeforeTest - void beforeClass() { - - customJSON = new JSONObject(); - customJSON.put("custom", "object"); - IdentityExtensionsDataHolder.getInstance().addWorker((context, properties) -> - customJSON, "customHandlerName"); - workerFunction = new OpenBankingAuthenticationWorkerFunctionImpl(); - authenticationContext = new AuthenticationContext(); - jsAuthenticationContext = new JsAuthenticationContext(authenticationContext); - } - - @Test - public void testHandleInvokeWithExistingWorker() { - - assertEquals(workerFunction.handleRequest(jsAuthenticationContext, - new HashMap<>(), "customHandlerName"), customJSON); - } - - @Test - public void testHandleInvokeWithInvalidWorker() { - - JSONObject jsonObject = workerFunction.handleRequest(jsAuthenticationContext, - new HashMap<>(), "invalidHandlerName"); - assertTrue(jsonObject.has("Error")); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/authz/request/OBOAuthAuthzRequestTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/authz/request/OBOAuthAuthzRequestTest.java deleted file mode 100644 index 01864c22..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/authz/request/OBOAuthAuthzRequestTest.java +++ /dev/null @@ -1,328 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.authz.request; - -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.oltu.oauth2.as.validator.CodeValidator; -import org.apache.oltu.oauth2.common.OAuth; -import org.apache.oltu.oauth2.common.exception.OAuthProblemException; -import org.apache.oltu.oauth2.common.exception.OAuthSystemException; -import org.apache.oltu.oauth2.common.validators.OAuthValidator; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.oauth.common.CodeTokenResponseValidator; -import org.wso2.carbon.identity.oauth.config.OAuthServerConfiguration; - -import java.nio.charset.StandardCharsets; -import java.util.Base64; -import java.util.Collections; -import java.util.HashMap; -import java.util.HashSet; -import java.util.Hashtable; -import java.util.Map; - -import javax.servlet.http.HttpServletRequest; - -import static org.mockito.Matchers.any; -import static org.mockito.Mockito.mock; -import static org.mockito.Mockito.when; -import static org.testng.Assert.assertEquals; - -/** - * Test for OB authorization request for request_uri support. - */ -@PrepareForTest({IdentityCommonUtil.class, OAuthServerConfiguration.class}) -@PowerMockIgnore("jdk.internal.reflect.*") -public class OBOAuthAuthzRequestTest extends PowerMockTestCase { - - private OBOAuthAuthzRequest obOAuthAuthzRequest; - private HttpServletRequest mockRequest; - private static final String STATE = "abc"; - private static final String SAMPLE_REQUEST_URI = "urn:ietf:params:oauth:request_uri:abc"; - - @BeforeMethod - public void beforeMethod() throws OAuthProblemException { - - // Mock HttpServletRequest - mockRequest = mock(HttpServletRequest.class); - when(mockRequest.getMethod()).thenReturn("POST"); - when(mockRequest.getParameter(IdentityCommonConstants.CLIENT_ID)).thenReturn("1234567"); - - // Mock IdentityCommonUtil - PowerMockito.mockStatic(IdentityCommonUtil.class); - PowerMockito.when(IdentityCommonUtil.handleOAuthProblemException(any(), any(), any())) - .thenReturn(OAuthProblemException.error("invalid_request").description("Error message").state(STATE)); - PowerMockito.when(IdentityCommonUtil.decodeRequestObjectAndGetKey(mockRequest, OAuth.OAUTH_STATE)) - .thenReturn(STATE); - - // Mock OAuthServerConfiguration - OAuthServerConfiguration oAuthServerConfigurationMock = PowerMockito.mock(OAuthServerConfiguration.class); - PowerMockito.mockStatic(OAuthServerConfiguration.class); - PowerMockito.when(OAuthServerConfiguration.getInstance()).thenReturn(oAuthServerConfigurationMock); - - // Mock supported response type validators - Map>> - supportedResponseTypeValidators = new Hashtable<>(); - supportedResponseTypeValidators.put("code", CodeValidator.class); - supportedResponseTypeValidators.put("code id_token", CodeTokenResponseValidator.class); - - PowerMockito.when(oAuthServerConfigurationMock.getSupportedResponseTypeValidators()) - .thenReturn(supportedResponseTypeValidators); - } - - @Test - public void testInitValidatorForCodeResponseType() throws OAuthProblemException, OAuthSystemException { - - // Mock - Map mockParameterMap = new HashMap<>(); - mockParameterMap.put(IdentityCommonConstants.RESPONSE_TYPE, new String[]{"code"}); - mockParameterMap.put(IdentityCommonConstants.REQUEST_URI, - new String[]{SAMPLE_REQUEST_URI}); - - when(mockRequest.getParameterMap()).thenReturn(mockParameterMap); - when(mockRequest.getParameter(IdentityCommonConstants.RESPONSE_TYPE)).thenReturn("code"); - - PowerMockito.when(IdentityCommonUtil.decodeRequestObjectAndGetKey(mockRequest, OAuth.OAUTH_RESPONSE_TYPE)) - .thenReturn("code"); - - obOAuthAuthzRequest = new OBOAuthAuthzRequest(mockRequest); - - // Assertion - assertEquals(obOAuthAuthzRequest.initValidator().getClass(), CodeValidator.class); - } - - @Test - public void testInitValidatorForHybridResponseType() throws OAuthProblemException, OAuthSystemException { - - // Mock - Map mockParameterMap = new HashMap<>(); - mockParameterMap.put(IdentityCommonConstants.RESPONSE_TYPE, new String[]{"code id_token"}); - mockParameterMap.put(IdentityCommonConstants.REQUEST_URI, - new String[]{SAMPLE_REQUEST_URI}); - - when(mockRequest.getParameterMap()).thenReturn(mockParameterMap); - when(mockRequest.getParameter(IdentityCommonConstants.RESPONSE_TYPE)).thenReturn("code id_token"); - when(mockRequest.getParameter(IdentityCommonConstants.REDIRECT_URI)).thenReturn("abc.com"); - when(mockRequest.getParameter(IdentityCommonConstants.SCOPE)).thenReturn("openid"); - - PowerMockito.when(IdentityCommonUtil.decodeRequestObjectAndGetKey(mockRequest, OAuth.OAUTH_RESPONSE_TYPE)) - .thenReturn("code id_token"); - - obOAuthAuthzRequest = new OBOAuthAuthzRequest(mockRequest); - - // Assertion - assertEquals(obOAuthAuthzRequest.initValidator().getClass(), CodeTokenResponseValidator.class); - } - - @Test(expectedExceptions = OAuthProblemException.class) - public void testInitValidatorWithoutResponseTypeParam() throws OAuthProblemException, OAuthSystemException { - - // Mock - Map mockParameterMap = new HashMap<>(); - mockParameterMap.put(IdentityCommonConstants.REQUEST_URI, - new String[]{SAMPLE_REQUEST_URI}); - - when(mockRequest.getParameterMap()).thenReturn(mockParameterMap); - - PowerMockito.when(IdentityCommonUtil.decodeRequestObjectAndGetKey(mockRequest, OAuth.OAUTH_RESPONSE_TYPE)) - .thenReturn(null); - - obOAuthAuthzRequest = new OBOAuthAuthzRequest(mockRequest); - } - - @Test(expectedExceptions = OAuthProblemException.class) - public void testInitValidatorWithUnsupportedResponseTypeParam() throws OAuthProblemException, OAuthSystemException { - - // Mock - Map mockParameterMap = new HashMap<>(); - mockParameterMap.put(IdentityCommonConstants.REQUEST_URI, - new String[]{SAMPLE_REQUEST_URI}); - - when(mockRequest.getParameterMap()).thenReturn(mockParameterMap); - - PowerMockito.when(IdentityCommonUtil.decodeRequestObjectAndGetKey(mockRequest, OAuth.OAUTH_RESPONSE_TYPE)) - .thenReturn("unsupported"); - - obOAuthAuthzRequest = new OBOAuthAuthzRequest(mockRequest); - } - - @Test - public void testValidGetScopesFromRequestURI() throws OAuthProblemException, OAuthSystemException { - - // Mock - Map mockParameterMap = new HashMap<>(); - mockParameterMap.put(IdentityCommonConstants.RESPONSE_TYPE, new String[]{"code id_token"}); - mockParameterMap.put(IdentityCommonConstants.REQUEST_URI, - new String[]{SAMPLE_REQUEST_URI}); - - when(mockRequest.getParameterMap()).thenReturn(mockParameterMap); - when(mockRequest.getParameter(IdentityCommonConstants.RESPONSE_TYPE)).thenReturn("code id_token"); - when(mockRequest.getParameter(IdentityCommonConstants.REDIRECT_URI)).thenReturn("abc.com"); - when(mockRequest.getParameter(IdentityCommonConstants.SCOPE)).thenReturn("openid"); - when(mockRequest.getParameter(IdentityCommonConstants.REQUEST_URI)).thenReturn(SAMPLE_REQUEST_URI); - - PowerMockito.when(IdentityCommonUtil.decodeRequestObjectAndGetKey(mockRequest, OAuth.OAUTH_SCOPE)) - .thenReturn("openid"); - - obOAuthAuthzRequest = new OBOAuthAuthzRequest(mockRequest); - - // Assertion - assertEquals(obOAuthAuthzRequest.getScopes(), new HashSet<>(Collections.singletonList("openid"))); - } - - @Test - public void testValidGetScopesFromRequest() throws OAuthProblemException, OAuthSystemException { - - // Mock - Map mockParameterMap = new HashMap<>(); - mockParameterMap.put(IdentityCommonConstants.RESPONSE_TYPE, new String[]{"code id_token"}); - - when(mockRequest.getParameterMap()).thenReturn(mockParameterMap); - when(mockRequest.getParameter(IdentityCommonConstants.RESPONSE_TYPE)).thenReturn("code id_token"); - when(mockRequest.getParameter(IdentityCommonConstants.REDIRECT_URI)).thenReturn("abc.com"); - when(mockRequest.getParameter(IdentityCommonConstants.SCOPE)).thenReturn("openid"); - - obOAuthAuthzRequest = new OBOAuthAuthzRequest(mockRequest); - - // Assertion - assertEquals(obOAuthAuthzRequest.getScopes(), new HashSet<>(Collections.singletonList("openid"))); - } - - @Test - public void testValidGetResponseTypeFromRequestURI() throws OAuthProblemException, OAuthSystemException { - - // Mock - Map mockParameterMap = new HashMap<>(); - mockParameterMap.put(IdentityCommonConstants.RESPONSE_TYPE, new String[]{"code id_token"}); - mockParameterMap.put(IdentityCommonConstants.REQUEST_URI, - new String[]{SAMPLE_REQUEST_URI}); - - when(mockRequest.getParameterMap()).thenReturn(mockParameterMap); - when(mockRequest.getParameter(IdentityCommonConstants.RESPONSE_TYPE)).thenReturn("code id_token"); - when(mockRequest.getParameter(IdentityCommonConstants.REDIRECT_URI)).thenReturn("abc.com"); - when(mockRequest.getParameter(IdentityCommonConstants.SCOPE)).thenReturn("openid"); - when(mockRequest.getParameter(IdentityCommonConstants.REQUEST_URI)).thenReturn(SAMPLE_REQUEST_URI); - - PowerMockito.when(IdentityCommonUtil.decodeRequestObjectAndGetKey(mockRequest, OAuth.OAUTH_RESPONSE_TYPE)) - .thenReturn("code id_token"); - - obOAuthAuthzRequest = new OBOAuthAuthzRequest(mockRequest); - - // Assertion - assertEquals(obOAuthAuthzRequest.getResponseType(), "code id_token"); - } - - @Test - public void testValidGetResponseTypeFromRequest() throws OAuthProblemException, OAuthSystemException { - - // Mock - Map mockParameterMap = new HashMap<>(); - mockParameterMap.put(IdentityCommonConstants.RESPONSE_TYPE, new String[]{"code id_token"}); - - when(mockRequest.getParameterMap()).thenReturn(mockParameterMap); - when(mockRequest.getParameter(IdentityCommonConstants.RESPONSE_TYPE)).thenReturn("code id_token"); - when(mockRequest.getParameter(IdentityCommonConstants.REDIRECT_URI)).thenReturn("abc.com"); - when(mockRequest.getParameter(IdentityCommonConstants.SCOPE)).thenReturn("openid"); - - obOAuthAuthzRequest = new OBOAuthAuthzRequest(mockRequest); - - // Assertion - assertEquals(obOAuthAuthzRequest.getResponseType(), "code id_token"); - } - - @Test - public void testValidGetStateFromRequestURI() throws OAuthProblemException, OAuthSystemException { - - // Mock - Map mockParameterMap = new HashMap<>(); - mockParameterMap.put(IdentityCommonConstants.RESPONSE_TYPE, new String[]{"code id_token"}); - mockParameterMap.put(IdentityCommonConstants.REQUEST_URI, - new String[]{SAMPLE_REQUEST_URI}); - - when(mockRequest.getParameterMap()).thenReturn(mockParameterMap); - when(mockRequest.getParameter(IdentityCommonConstants.RESPONSE_TYPE)).thenReturn("code id_token"); - when(mockRequest.getParameter(IdentityCommonConstants.REDIRECT_URI)).thenReturn("abc.com"); - when(mockRequest.getParameter(IdentityCommonConstants.SCOPE)).thenReturn("openid"); - when(mockRequest.getParameter(IdentityCommonConstants.REQUEST_URI)).thenReturn(SAMPLE_REQUEST_URI); - - PowerMockito.when(IdentityCommonUtil.decodeRequestObjectAndGetKey(mockRequest, OAuth.OAUTH_STATE)) - .thenReturn("abc"); - - obOAuthAuthzRequest = new OBOAuthAuthzRequest(mockRequest); - - // Assertion - assertEquals(obOAuthAuthzRequest.getState(), "abc"); - } - - @Test - public void testInvalidGetStateFromRequestURI() throws OAuthProblemException, OAuthSystemException { - - Map mockParameterMap = new HashMap<>(); - mockParameterMap.put(IdentityCommonConstants.RESPONSE_TYPE, new String[]{"code id_token"}); - mockParameterMap.put(IdentityCommonConstants.REQUEST_URI, - new String[]{SAMPLE_REQUEST_URI}); - - when(mockRequest.getParameterMap()).thenReturn(mockParameterMap); - when(mockRequest.getParameter(IdentityCommonConstants.RESPONSE_TYPE)).thenReturn("code id_token"); - when(mockRequest.getParameter(IdentityCommonConstants.REDIRECT_URI)).thenReturn("abc.com"); - when(mockRequest.getParameter(IdentityCommonConstants.SCOPE)).thenReturn("openid"); - when(mockRequest.getParameter(IdentityCommonConstants.REQUEST_URI)).thenReturn(SAMPLE_REQUEST_URI); - - // Simulate an exception being thrown when decoding the state - PowerMockito.when(IdentityCommonUtil.decodeRequestObjectAndGetKey(mockRequest, OAuth.OAUTH_STATE)) - .thenThrow(OAuthProblemException.error("invalid_request").description("Invalid state").state("abc")); - - obOAuthAuthzRequest = new OBOAuthAuthzRequest(mockRequest); - - assertEquals(obOAuthAuthzRequest.getState(), null); - } - - @Test - public void testValidGetScopesFromRequest_WhenRequestURIIsAbsent() throws OAuthProblemException, - OAuthSystemException { - - Map mockParameterMap = new HashMap<>(); - mockParameterMap.put(IdentityCommonConstants.RESPONSE_TYPE, new String[]{"code id_token"}); - mockParameterMap.put(IdentityCommonConstants.SCOPE, new String[]{"openid"}); - mockParameterMap.put(IdentityCommonConstants.REQUEST, - new String[]{Base64.getEncoder().encodeToString( - "{\"scope\": \"openid\", \"redirect_uri\": \"http://example.com\"}".getBytes( - StandardCharsets.UTF_8))}); - mockParameterMap.put(IdentityCommonConstants.REDIRECT_URI, new String[]{"http://example.com"}); - - when(mockRequest.getParameterMap()).thenReturn(mockParameterMap); - when(mockRequest.getParameter(IdentityCommonConstants.RESPONSE_TYPE)).thenReturn("code id_token"); - when(mockRequest.getParameter(IdentityCommonConstants.SCOPE)).thenReturn("openid"); - when(mockRequest.getParameter(IdentityCommonConstants.REQUEST)).thenReturn( - Base64.getEncoder().encodeToString( - "{\"scope\": \"openid\", \"redirect_uri\": \"http://example.com\"}".getBytes( - StandardCharsets.UTF_8))); - when(mockRequest.getParameter(IdentityCommonConstants.REDIRECT_URI)).thenReturn("http://example.com"); - - obOAuthAuthzRequest = new OBOAuthAuthzRequest(mockRequest); - - assertEquals(obOAuthAuthzRequest.getScopes(), new HashSet<>(Collections.singletonList("openid"))); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/DefaultOBRequestObjectValidatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/DefaultOBRequestObjectValidatorTest.java deleted file mode 100644 index 477c3cba..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/DefaultOBRequestObjectValidatorTest.java +++ /dev/null @@ -1,161 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator; - -import com.nimbusds.jose.JOSEObject; -import com.nimbusds.jose.JWSAlgorithm; -import com.nimbusds.jwt.PlainJWT; -import com.nimbusds.jwt.SignedJWT; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.validator.OpenBankingValidator; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator.models.OBRequestObject; -import com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator.models.ValidationResponse; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.oauth2.RequestObjectException; -import org.wso2.carbon.identity.openidconnect.model.RequestObject; - -import java.text.ParseException; -import java.util.Collections; - -import static org.mockito.Mockito.anyString; -import static org.mockito.Mockito.doReturn; -import static org.mockito.Mockito.mock; - -/** - * Test for Default Open Banking object validator. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({OpenBankingValidator.class, IdentityExtensionsDataHolder.class}) -public class DefaultOBRequestObjectValidatorTest extends PowerMockTestCase { - - private static final String CLIENT_ID_1 = "2X0n9WSNmPiq3XTB8dtC0Shs5r8a"; - private static final String CLIENT_ID_2 = "owjqxsPTQ7zJIFeZRib5b03ufxsa"; - private static ConsentCoreServiceImpl consentCoreServiceMock; - - @BeforeClass - public void initTest() { - - consentCoreServiceMock = PowerMockito.mock(ConsentCoreServiceImpl.class); - } - - @BeforeMethod - private void mockStaticClasses() throws ConsentManagementException { - - PowerMockito.mockStatic(IdentityExtensionsDataHolder.class); - IdentityExtensionsDataHolder mock = PowerMockito.mock(IdentityExtensionsDataHolder.class); - PowerMockito.when(IdentityExtensionsDataHolder.getInstance()).thenReturn(mock); - PowerMockito.when(IdentityExtensionsDataHolder.getInstance().getConsentCoreService()) - .thenReturn(consentCoreServiceMock); - } - - @Test - public void testValidateOBConstraintsWithValidRequestObject() throws Exception { - // mock - DetailedConsentResource consentResourceMock = mock(DetailedConsentResource.class); - doReturn(CLIENT_ID_1).when(consentResourceMock).getClientID(); - - doReturn(consentResourceMock).when(consentCoreServiceMock).getDetailedConsent(anyString()); - - OpenBankingValidator openBankingValidatorMock = mock(OpenBankingValidator.class); - doReturn("").when(openBankingValidatorMock).getFirstViolation(Mockito.anyObject()); - - PowerMockito.mockStatic(OpenBankingValidator.class); - PowerMockito.when(OpenBankingValidator.getInstance()).thenReturn(openBankingValidatorMock); - - // act - DefaultOBRequestObjectValidator uut = new DefaultOBRequestObjectValidator(); - - OBRequestObject obRequestObject = getObRequestObject(ReqObjectTestDataProvider.VALID_REQUEST); - ValidationResponse validationResponse = uut.validateOBConstraints(obRequestObject, Collections.emptyMap()); - - // assert - Assert.assertTrue(validationResponse.isValid()); - } - - @Test - public void testValidateOBConstraintsWhenNoClientId() throws Exception { - // mock - DetailedConsentResource consentResourceMock = mock(DetailedConsentResource.class); - doReturn(null).when(consentResourceMock).getClientID(); - - ConsentCoreServiceImpl consentCoreServiceMock = mock(ConsentCoreServiceImpl.class); - doReturn(consentResourceMock).when(consentCoreServiceMock).getDetailedConsent(anyString()); - - OpenBankingValidator openBankingValidatorMock = mock(OpenBankingValidator.class); - doReturn("").when(openBankingValidatorMock).getFirstViolation(Mockito.anyObject()); - - PowerMockito.mockStatic(OpenBankingValidator.class); - PowerMockito.when(OpenBankingValidator.getInstance()).thenReturn(openBankingValidatorMock); - - // act - DefaultOBRequestObjectValidator uut = new DefaultOBRequestObjectValidator(); - - OBRequestObject obRequestObject = getObRequestObject(ReqObjectTestDataProvider.NO_CLIENT_ID_REQUEST); - ValidationResponse validationResponse = uut.validateOBConstraints(obRequestObject, Collections.emptyMap()); - - // assert - Assert.assertFalse(validationResponse.isValid()); - Assert.assertEquals(validationResponse.getViolationMessage(), - "Client id or scope cannot be empty"); - } - - @Test - public void testValidateOBConstraintsWhenOBRequestObjectHasErrors() throws Exception { - // mock - OpenBankingValidator openBankingValidatorMock = mock(OpenBankingValidator.class); - doReturn("dummy-error").when(openBankingValidatorMock).getFirstViolation(Mockito.anyObject()); - - PowerMockito.mockStatic(OpenBankingValidator.class); - PowerMockito.when(OpenBankingValidator.getInstance()).thenReturn(openBankingValidatorMock); - - // act - DefaultOBRequestObjectValidator uut = new DefaultOBRequestObjectValidator(); - - OBRequestObject obRequestObject = getObRequestObject(ReqObjectTestDataProvider.REQUEST_STRING); - ValidationResponse validationResponse = uut.validateOBConstraints(obRequestObject, Collections.emptyMap()); - - // assert - Assert.assertFalse(validationResponse.isValid()); - Assert.assertEquals(validationResponse.getViolationMessage(), "dummy-error"); - } - - private OBRequestObject getObRequestObject(String request) throws ParseException, RequestObjectException { - - RequestObject requestObject = new RequestObject(); - JOSEObject jwt = JOSEObject.parse(request); - if (jwt.getHeader().getAlgorithm() == null || jwt.getHeader().getAlgorithm().equals(JWSAlgorithm.NONE)) { - requestObject.setPlainJWT(PlainJWT.parse(request)); - } else { - requestObject.setSignedJWT(SignedJWT.parse(request)); - } - return new OBRequestObject<>(requestObject); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/ReqObjectTestDataProvider.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/ReqObjectTestDataProvider.java deleted file mode 100644 index 3ca3c069..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/ReqObjectTestDataProvider.java +++ /dev/null @@ -1,133 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator; - -import com.nimbusds.jose.JOSEObject; -import com.nimbusds.jose.JWSAlgorithm; -import com.nimbusds.jwt.PlainJWT; -import com.nimbusds.jwt.SignedJWT; -import org.testng.annotations.DataProvider; -import org.wso2.carbon.identity.oauth2.RequestObjectException; -import org.wso2.carbon.identity.oauth2.model.OAuth2Parameters; -import org.wso2.carbon.identity.openidconnect.model.RequestObject; - -import java.text.ParseException; - -/** - * Request object test data provider. - */ -public class ReqObjectTestDataProvider { - - public static final String VALID_REQUEST = "eyJraWQiOiJEd01LZFdNbWo3UFdpbnZvcWZReVhWenlaNlEiLCJ0eXAiOiJKV1" + - "QiLCJhbGciOiJQUzI1NiJ9.eyJhdWQiOiJodHRwczovL2xvY2FsaG9zdDo5NDQ2L29hdXRoMi90b2tlbiIsIm1heF9hZ2UiOjg2NDAw" + - "LCJjcml0IjpbImI2NCIsImh0dHA6Ly9vcGVuYmFua2luZy5vcmcudWsvaWF0IiwiaHR0cDovL29wZW5iYW5raW5nLm9yZy51ay9pc3M" + - "iLCJodHRwOi8vb3BlbmJhbmtpbmcub3JnLnVrL3RhbiJdLCJzY29wZSI6ImFjY291bnRzIG9wZW5pZCIsImV4cCI6MTk1NDcwODcxMC" + - "wiY2xhaW1zIjp7ImlkX3Rva2VuIjp7ImFjciI6eyJ2YWx1ZXMiOlsidXJuOm9wZW5iYW5raW5nOnBzZDI6Y2EiLCJ1cm46b3BlbmJhb" + - "mtpbmc6cHNkMjpzY2EiXSwiZXNzZW50aWFsIjp0cnVlfSwib3BlbmJhbmtpbmdfaW50ZW50X2lkIjp7InZhbHVlIjoiOTRmODU3M2Mt" + - "NjA4MC00MDI1LThkOWItMDhlM2U5MjAwZGU3IiwiZXNzZW50aWFsIjp0cnVlfX0sInVzZXJpbmZvIjp7Im9wZW5iYW5raW5nX2ludGV" + - "udF9pZCI6eyJ2YWx1ZSI6Ijk0Zjg1NzNjLTYwODAtNDAyNS04ZDliLTA4ZTNlOTIwMGRlNyIsImVzc2VudGlhbCI6dHJ1ZX19fSwiaX" + - "NzIjoiMlgwbjlXU05tUGlxM1hUQjhkdEMwU2hzNXI4YSIsInJlc3BvbnNlX3R5cGUiOiJjb2RlIGlkX3Rva2VuIiwicmVkaXJlY3Rfd" + - "XJpIjoiaHR0cHM6Ly93d3cuZ29vZ2xlLmNvbS9yZWRpcmVjdHMvcmVkaXJlY3QxIiwic3RhdGUiOiIwcE4wTkJUSGN2Iiwibm9uY2Ui" + - "OiJqQlhoT21PS0NCIiwiY2xpZW50X2lkIjoiMlgwbjlXU05tUGlxM1hUQjhkdEMwU2hzNXI4YSJ9.BjQIMnlqevJgj1zL25hmfKA9Ab" + - "7qm2GvrkZAkxbTMNOxkxtKxQSt9QkLfycITEeM9YdGV5rh2FMdXxyex-WtO_H0G9zlKtsDyrsUw3_HWaBDd-61Hz6n65Few_f6bwtIg" + - "HtW8oDeKpylUu01OsYtB_s4nnDw42ZCKv7zGzkTDkyoxoM2_b_AUqh-F3PNY9Arru5m1-FGDYi_zl4iQ3d3um_NYnhPhmC2wz2R9yms" + - "flXBn9bd-d6nPKl_ftGnqmiBua7oKMd-3CMCFW9Uxig82PHbwHcuy1hYqa_7JoE58Zkr6baGur3YtgJ2381_8t5v19DJvjqhaodabfe" + - "uNWR3GA"; - - public static final String SCOPES_INVALID_REQ = "eyJhbGciOiJQUzI1NiIsInR5cCI6IkpXVCIsImtpZCI6IkR3TUtkV01tajdQ" + - "V2ludm9xZlF5WFZ6eVo2USJ9.eyJhdWQiOiJodHRwczovL2xvY2FsaG9zdDo4MjQzL3Rva2VuIiwicmVzcG9uc2VfdHlwZSI6ImNvZG" + - "UgaWRfdG9rZW4iLCJjbGllbnRfaWQiOiJpVEtPZnVxejQ2WTFIVlkyQkYwWjdKTTE4QXdhIiwicmVkaXJlY3RfdXJpIjoiaHR0cHM6L" + - "y8xMC4xMTAuNS4yMzI6ODAwMC90ZXN0L2EvYXBwMS9jYWxsYmFjayIsInNjb3BlIjoib3BlbmlkIGJhbms6YWNjb3VudHMuYmFzaWM6" + - "cmVhZCBiYW5rOmFjY291bnRzLmRldGFpbDpyZWFkIGJhbms6dHJhbnNhY3Rpb25zOnJlYWQiLCJzdGF0ZSI6ImFmMGlmanNsZGtqIiw" + - "ibm9uY2UiOiJuLTBTNl9XekEyTWoiLCJjbGFpbXMiOnsic2hhcmluZ19kdXJhdGlvbiI6IjcyMDAiLCJpZF90b2tlbiI6eyJhY3IiOn" + - "siZXNzZW50aWFsIjp0cnVlLCJ2YWx1ZXMiOlsidXJuOmNkcy5hdTpjZHI6MyJdfSwib3BlbmJhbmtpbmdfaW50ZW50X2lkIjp7InZhb" + - "HVlIjoiZDMwMmI4ODgtNzM3Zi00YzQ5LTk4ZmUtNmVkZGU4OTk2ZDZlIiwiZXNzZW50aWFsIjp0cnVlfX0sInVzZXJpbmZvIjp7Imdp" + - "dmVuX25hbWUiOm51bGwsImZhbWlseV9uYW1lIjpudWxsfX19.dmkbejbZLg22Pe81rgt9-TS_7ynT4f1HGpUNugryL7K0-xWSwroqQL" + - "mqLyx442nahZP1nd_1r5LScfer5-lslKid7WTh_gV-v9GvBe6U5xDIKxxgHXBepz7nAUZnOkmyZF9As6JrKxfVa36F-iKN-ZUyIfv0F" + - "bVs5rejAWXqNQzYOlkSwMlOZWDjeCpozKwMYR4FyyecJ-l6gRF11hbRTxo-Uj8U40x3hHH7R7vDVLT6eK5VTwbmFYROXHy4iBQJNIcL" + - "sOBF9lJnSvw8_rJEUteFBqtrlDpWVdQ8fZcHj_9mYMSCZlTKFMvh22ILXJ13EoxrhqPpuFxZMMx7oEPNjA"; - - public static final String REQUEST_STRING = "eyJhbGciOiJQUzI1NiIsInR5cCI6IkpXVCIsImtpZCI6IkR3TUtkV01tajdQV2ludm9x" + - "ZlF5WFZ6eVo2USJ9.eyJhdWQiOiJodHRwczovL2xvY2FsaG9zdDo4MjQzL3Rva2VuIiwicmVzcG9uc2VfdHlwZSI6ImNvZGUgaWR" + - "fdG9rZW4iLCJjbGllbnRfaWQiOiJpVEtPZnVxejQ2WTFIVlkyQkYwWjdKTTE4QXdhIiwicmVkaXJlY3RfdXJpIjoiaHR0cHM6Ly8" + - "xMC4xMTAuNS4yMzI6ODAwMC90ZXN0L2EvYXBwMS9jYWxsYmFjayIsInNjb3BlIjoib3BlbmlkIGJhbms6YWNjb3VudHMuYmFzaWM6" + - "cmVhZCBiYW5rOmFjY291bnRzLmRldGFpbDpyZWFkIGJhbms6dHJhbnNhY3Rpb25zOnJlYWQiLCJzdGF0ZSI6ImFmMGlmanNsZGtqIi" + - "wibm9uY2UiOiJuLTBTNl9XekEyTWoiLCJjbGFpbXMiOnsic2hhcmluZ19kdXJhdGlvbiI6IjcyMDAiLCJpZF90b2tlbiI6eyJhY3Ii" + - "OnsiZXNzZW50aWFsIjp0cnVlLCJ2YWx1ZXMiOlsidXJuOmNkcy5hdTpjZHI6MyJdfX0sInVzZXJpbmZvIjp7ImdpdmVuX25hbWUiOm" + - "51bGwsImZhbWlseV9uYW1lIjpudWxsfX19.cnKvzjgiDWJ2JeRGL8ncTKB_pCxEynNHn6kzHSPBBXYRJ5e-WvPocTkvaDnwu1qSr" + - "5lsJnFCNgYuNickzoIaTl9wUvl0rnK15iGVe0rSOCWIJ53eVphaV9uYtRfVHTN4HL4ecgdsREHhu6MyjYcqdgAeuv4g0robZGf" + - "DDVCLr2Xb77f8yAr42xc6fBccAFnvZX33zVOHtFaY3S3j9RbQqRZjUxLycIgdVXGypRc2ESVKqJ9WgGxKG6fCUt2rDgqsobVj" + - "8ekRAMyP2fGmYLoRAyycJ8JwU9uoRhGL6nqM6_uOYNG5a6xOsSs8i1Yvn4s7G6FUKQ_bmm4Gx2aJptzVA"; - - public static final String NO_CLIENT_ID_REQUEST = "eyJhbGciOiJQUzI1NiIsInR5cCI6IkpXVCIsImtpZCI6IkR3TUtkV01tajdQV" + - "2ludm9xZlF5WFZ6eVo2USJ9.eyJhdWQiOiJodHRwczovL2xvY2FsaG9zdDo4MjQzL3Rva2VuIiwicmVzcG9uc2VfdHlwZSI6ImNvZGU" + - "gaWRfdG9rZW4iLCJjbGllbnRfaWQiOiIiLCJyZWRpcmVjdF91cmkiOiJodHRwczovLzEwLjExMC41LjIzMjo4MDAwL3Rlc3QvYS9hcH" + - "AxL2NhbGxiYWNrIiwic2NvcGUiOiJvcGVuaWQgYmFuazphY2NvdW50cy5iYXNpYzpyZWFkIGJhbms6YWNjb3VudHMuZGV0YWlsOnJlY" + - "WQgYmFuazp0cmFuc2FjdGlvbnM6cmVhZCIsInN0YXRlIjoiYWYwaWZqc2xka2oiLCJub25jZSI6Im4tMFM2X1d6QTJNaiIsImNsYWlt" + - "cyI6eyJzaGFyaW5nX2R1cmF0aW9uIjoiNzIwMCIsImlkX3Rva2VuIjp7ImFjciI6eyJlc3NlbnRpYWwiOnRydWUsInZhbHVlcyI6WyJ" + - "1cm46Y2RzLmF1OmNkcjozIl19fSwidXNlcmluZm8iOnsiZ2l2ZW5fbmFtZSI6bnVsbCwiZmFtaWx5X25hbWUiOm51bGx9fX0.BE444R" + - "orL9NH6THQkpr6HzylwqoxGod_OD8aIWLUgOJnS9FFQAQh6RAE-k4cXbriPS40EKC4pNIkA0CuDr3zmaZQsTrcn09W_gL9aWxIQ50cI" + - "2RPkqXzl3W5EEtA-LBOpiqyToVvZfvfsVqZqWF8nNQW85_rToR6yiB-LWWMvlshkF0XDP6qRiFuQhsRDAa6Ro1r3hdnHBBMeoBSGRTe" + - "0LE6jCVq_P6YrJvIbpZnMHL1wsgkJQzXoeu8HLN7kgwaYuUQesWIAWrFR26Ca7kcmRCmVqtj4Z2arhF2QQWtKUurrDYKcSxrQwZHRsQ" + - "Zh76z6dULjhUfz7JQ5JIvtqooAA"; - - - @DataProvider(name = "dp-checkValidRequestObject") - public Object[][] dpCheckValidRequestObject() throws ParseException, RequestObjectException { - - RequestObject requestObject = new RequestObject(); - JOSEObject jwt = JOSEObject.parse(REQUEST_STRING); - if (jwt.getHeader().getAlgorithm() == null || jwt.getHeader().getAlgorithm().equals(JWSAlgorithm.NONE)) { - requestObject.setPlainJWT(PlainJWT.parse(REQUEST_STRING)); - } else { - requestObject.setSignedJWT(SignedJWT.parse(REQUEST_STRING)); - } - - return new Object[][]{ - {requestObject, new OAuth2Parameters()}, - {requestObject, null}, - }; - } - - @DataProvider(name = "dp-checkIncorrectRequestObject") - public Object[][] dpCheckIncorrectRequestObject() throws ParseException, RequestObjectException { - - RequestObject requestObject = new RequestObject(); - JOSEObject jwt = JOSEObject.parse(REQUEST_STRING); - if (jwt.getHeader().getAlgorithm() == null || jwt.getHeader().getAlgorithm().equals(JWSAlgorithm.NONE)) { - requestObject.setPlainJWT(PlainJWT.parse(REQUEST_STRING)); - } else { - requestObject.setSignedJWT(SignedJWT.parse(REQUEST_STRING)); - } - - return new Object[][]{ - {new RequestObject(), new OAuth2Parameters()}, - }; - } - - @DataProvider(name = "dp-checkInValidRequestObject") - public Object[][] dpCheckInValidRequestObject() { - - return new Object[][]{ - {null, new OAuth2Parameters()}, - {null, null}, - }; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/RequestObjectValidatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/RequestObjectValidatorTest.java deleted file mode 100644 index 8e908510..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/request/validator/RequestObjectValidatorTest.java +++ /dev/null @@ -1,141 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator; - -import com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator.models.OBRequestObject; -import com.wso2.openbanking.accelerator.identity.auth.extensions.request.validator.models.ValidationResponse; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.oauth2.RequestObjectException; -import org.wso2.carbon.identity.oauth2.model.OAuth2Parameters; -import org.wso2.carbon.identity.openidconnect.model.RequestObject; - -import static org.mockito.Matchers.anyMap; -import static org.mockito.Matchers.anyObject; -import static org.mockito.Mockito.doReturn; -import static org.mockito.Mockito.mock; -import static org.mockito.Mockito.spy; -import static org.mockito.Mockito.when; -import static org.testng.AssertJUnit.assertEquals; -import static org.testng.AssertJUnit.assertTrue; - -/** - * Test for request object validator. - */ -public class RequestObjectValidatorTest { - - @Test(dataProvider = "dp-checkValidRequestObject", dataProviderClass = ReqObjectTestDataProvider.class) - public void checkValidRequestObject(RequestObject requestObject, - OAuth2Parameters oAuth2Parameters) throws Exception { - - // Mock - OBRequestObjectValidator obRequestObjectValidator = mock(OBRequestObjectValidator.class); - when(obRequestObjectValidator.validateOBConstraints(anyObject(), anyMap())) - .thenReturn(new ValidationResponse(true)); - - OBRequestObjectValidationExtension uut = spy(new OBRequestObjectValidationExtension()); - doReturn(true).when(uut).isRegulatory(anyObject()); - doReturn(true).when(uut).validateIAMConstraints(anyObject(), anyObject()); - doReturn("accounts payments").when(uut).getAllowedScopes(anyObject()); - - // Assign - OBRequestObjectValidationExtension.obDefaultRequestObjectValidator = obRequestObjectValidator; - - // Act - boolean result = uut.validateRequestObject(requestObject, oAuth2Parameters); - - // Assert - assertTrue("Valid request object should pass", result); - } - - - @Test(dataProvider = "dp-checkIncorrectRequestObject", dataProviderClass = ReqObjectTestDataProvider.class, - expectedExceptions = org.wso2.carbon.identity.oauth2.RequestObjectException.class) - public void checkIncorrectRequestObject(RequestObject requestObject, - OAuth2Parameters oAuth2Parameters) throws Exception { - - // Mock - // OBRequestObjectValidator obRequestObjectValidator = mock(OBRequestObjectValidator.class); - //when(obRequestObjectValidator.validateOBConstraints(anyObject())).thenReturn(new ValidationResponse(true)); - - OBRequestObjectValidationExtension uut = spy(new OBRequestObjectValidationExtension()); - doReturn(true).when(uut).validateIAMConstraints(anyObject(), anyObject()); - - // Assign - OBRequestObjectValidationExtension.obDefaultRequestObjectValidator = new OBRequestObjectValidator(); - - // Act - boolean result = uut.validateRequestObject(requestObject, oAuth2Parameters); - - // Assert - assertTrue("InValid request object should throw exception", result); - } - - - @Test(dataProvider = "dp-checkInValidRequestObject", dataProviderClass = ReqObjectTestDataProvider.class, - expectedExceptions = org.wso2.carbon.identity.oauth2.RequestObjectException.class) - public void checkInValidRequestObject(RequestObject requestObject, - OAuth2Parameters oAuth2Parameters) throws Exception { - - // Mock - OBRequestObjectValidator obRequestObjectValidator = mock(OBRequestObjectValidator.class); - when(obRequestObjectValidator.validateOBConstraints(anyObject(), anyMap())) - .thenReturn(new ValidationResponse(true)); - - OBRequestObjectValidationExtension uut = spy(new OBRequestObjectValidationExtension()); - doReturn(true).when(uut).isRegulatory(anyObject()); - doReturn(true).when(uut).validateIAMConstraints(anyObject(), anyObject()); - doReturn("accounts payments").when(uut).getAllowedScopes(anyObject()); - - // Assign - OBRequestObjectValidationExtension.obDefaultRequestObjectValidator = obRequestObjectValidator; - - // Act - boolean result = uut.validateRequestObject(requestObject, oAuth2Parameters); - - // Assert - assertTrue("InValid request object should throw exception", result); - } - - @Test(dataProvider = "dp-checkValidRequestObject", dataProviderClass = ReqObjectTestDataProvider.class) - public void checkChildClassCreation(RequestObject requestObject, - OAuth2Parameters oAuth2Parameters) throws RequestObjectException { - - class UKRequestObject extends OBRequestObject { - public UKRequestObject(OBRequestObject childObject) { - super(childObject); - } - } - - OBRequestObject obRequestObject = new OBRequestObject(requestObject); - UKRequestObject ukRequestObject = new UKRequestObject(obRequestObject); - - // Assert - assertEquals("Inheritance should be preserved in toolkit child classes", - 8, ukRequestObject.getClaimsSet().getClaims().size()); - - // Assert - assertEquals(3, ukRequestObject.getRequestedClaims().size()); - - // Assert - assertEquals("code id_token", ukRequestObject.getClaim("response_type")); - - // Assert - assertEquals("code id_token", ukRequestObject.getClaimValue("response_type")); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/ResponseTypeHandlerTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/ResponseTypeHandlerTest.java deleted file mode 100644 index a6df365f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/handler/ResponseTypeHandlerTest.java +++ /dev/null @@ -1,132 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.response.handler; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.auth.extensions.response.handler.impl.OBDefaultResponseTypeHandlerImpl; -import org.mockito.ArgumentCaptor; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.authz.OAuthAuthzReqMessageContext; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AuthorizeReqDTO; - -import static org.mockito.Matchers.anyObject; -import static org.mockito.Mockito.doReturn; -import static org.mockito.Mockito.doThrow; -import static org.mockito.Mockito.mock; -import static org.mockito.Mockito.spy; -import static org.mockito.Mockito.verify; -import static org.mockito.Mockito.when; -import static org.testng.Assert.fail; -import static org.testng.AssertJUnit.assertEquals; - -/** - * test for response type handler. - */ -public class ResponseTypeHandlerTest { - - @Test - public void checkValidHybridResponseTypeHandling() throws IdentityOAuth2Exception, OpenBankingException { - - // Mock - OBResponseTypeHandler obResponseTypeHandler = mock(OBDefaultResponseTypeHandlerImpl.class); - when(obResponseTypeHandler.updateRefreshTokenValidityPeriod(anyObject())).thenReturn(999L); - when(obResponseTypeHandler.updateApprovedScopes(anyObject())).thenReturn(new String[]{"Asd", "addd"}); - - OBHybridResponseTypeHandlerExtension uut = spy(new OBHybridResponseTypeHandlerExtension()); - doReturn(null).when(uut).issueCode(anyObject()); - doReturn(true).when(uut).isRegulatory(anyObject()); - - ArgumentCaptor argument = - ArgumentCaptor.forClass(OAuthAuthzReqMessageContext.class); - - // Assign - OBHybridResponseTypeHandlerExtension.obResponseTypeHandler = obResponseTypeHandler; - - // Act - uut.issue(new OAuthAuthzReqMessageContext(new OAuth2AuthorizeReqDTO())); - - // Assert - verify(uut).issueCode(argument.capture()); - assertEquals(999L, argument.getValue().getRefreshTokenvalidityPeriod()); - assertEquals(2, argument.getValue().getApprovedScope().length); - - } - - @Test - public void checkValidCodeResponseTypeHandling() throws IdentityOAuth2Exception, OpenBankingException { - - // Mock - OBResponseTypeHandler obResponseTypeHandler = mock(OBDefaultResponseTypeHandlerImpl.class); - when(obResponseTypeHandler.updateRefreshTokenValidityPeriod(anyObject())).thenReturn(109L); - when(obResponseTypeHandler.updateApprovedScopes(anyObject())).thenReturn(new String[]{"Asd", "addd", "rr"}); - - OBCodeResponseTypeHandlerExtension uut = spy(new OBCodeResponseTypeHandlerExtension()); - doReturn(null).when(uut).issueCode(anyObject()); - doReturn(true).when(uut).isRegulatory(anyObject()); - - ArgumentCaptor argument = - ArgumentCaptor.forClass(OAuthAuthzReqMessageContext.class); - - // Assign - OBCodeResponseTypeHandlerExtension.obResponseTypeHandler = obResponseTypeHandler; - - // Act - uut.issue(new OAuthAuthzReqMessageContext(new OAuth2AuthorizeReqDTO())); - - // Assert - verify(uut).issueCode(argument.capture()); - assertEquals(109L, argument.getValue().getRefreshTokenvalidityPeriod()); - assertEquals(3, argument.getValue().getApprovedScope().length); - - } - - @Test - public void checkHandlerLogic() { - - OAuthAuthzReqMessageContext mock = mock(OAuthAuthzReqMessageContext.class); - when(mock.getRefreshTokenvalidityPeriod()).thenReturn(6666L); - when(mock.getApprovedScope()).thenReturn(new String[]{"1"}); - - OBResponseTypeHandler uut = new OBDefaultResponseTypeHandlerImpl(); - - assertEquals(6666L, uut.updateRefreshTokenValidityPeriod(mock)); - - } - - @Test - public void checkExceptionHandling_WhenIsRegulatoryThrowsOpenBankingException() throws Exception { - - OAuthAuthzReqMessageContext mockAuthzReqMsgCtx = mock(OAuthAuthzReqMessageContext.class); - OAuth2AuthorizeReqDTO mockAuthorizeReqDTO = mock(OAuth2AuthorizeReqDTO.class); - when(mockAuthzReqMsgCtx.getAuthorizationReqDTO()).thenReturn(mockAuthorizeReqDTO); - when(mockAuthorizeReqDTO.getConsumerKey()).thenReturn("dummyClientId"); - OBCodeResponseTypeHandlerExtension uut = spy(new OBCodeResponseTypeHandlerExtension()); - doThrow(new OpenBankingException("Simulated isRegulatory exception")) - .when(uut).isRegulatory("dummyClientId"); - - try { - uut.issue(mockAuthzReqMsgCtx); - fail("Expected IdentityOAuth2Exception was not thrown."); - } catch (IdentityOAuth2Exception e) { - // Verify that the IdentityOAuth2Exception is thrown with the expected message - assertEquals("Error while reading regulatory property", e.getMessage()); - } - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/validator/OBCodeResponseTypeValidatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/validator/OBCodeResponseTypeValidatorTest.java deleted file mode 100644 index 2c8e2804..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/validator/OBCodeResponseTypeValidatorTest.java +++ /dev/null @@ -1,60 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.response.validator; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.oltu.oauth2.common.exception.OAuthProblemException; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.annotations.Test; - - -import javax.servlet.http.HttpServletRequest; - -import static org.mockito.Mockito.spy; -import static org.powermock.api.mockito.PowerMockito.mock; -import static org.powermock.api.mockito.PowerMockito.when; - -/** - * OBCodeResponseTypeValidator Test class. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({IdentityCommonUtil.class}) -public class OBCodeResponseTypeValidatorTest extends PowerMockTestCase { - - @Test - public void checkValidCodeResponseTypeValidation() throws OAuthProblemException, OpenBankingException { - - // Mock - HttpServletRequest httpServletRequestMock = mock(HttpServletRequest.class); - when(httpServletRequestMock.getParameter("response_type")).thenReturn("code"); - when(httpServletRequestMock.getParameter("client_id")).thenReturn("1234567654321"); - - PowerMockito.mockStatic(IdentityCommonUtil.class); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - - OBCodeResponseTypeValidator uut = spy(new OBCodeResponseTypeValidator()); - - // Act - uut.validateRequiredParameters(httpServletRequestMock); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/validator/OBHybridResponseTypeValidatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/validator/OBHybridResponseTypeValidatorTest.java deleted file mode 100644 index cf9c667f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/auth/extensions/response/validator/OBHybridResponseTypeValidatorTest.java +++ /dev/null @@ -1,103 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.auth.extensions.response.validator; - -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.oltu.oauth2.common.OAuth; -import org.apache.oltu.oauth2.common.exception.OAuthProblemException; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.annotations.Test; - -import javax.servlet.http.HttpServletRequest; - -import static org.mockito.Mockito.spy; -import static org.powermock.api.mockito.PowerMockito.mock; -import static org.powermock.api.mockito.PowerMockito.when; - -/** - * OBHybridResponseTypeValidator Test class. - */ -@PrepareForTest({IdentityCommonUtil.class}) -@PowerMockIgnore("jdk.internal.reflect.*") -public class OBHybridResponseTypeValidatorTest extends PowerMockTestCase { - - @Test - public void checkValidHybridResponseTypeValidationWithRequestURI() throws OAuthProblemException { - - // Mock - HttpServletRequest httpServletRequestMock = mock(HttpServletRequest.class); - when(httpServletRequestMock.getParameter(IdentityCommonConstants.REQUEST_URI)).thenReturn("test"); - when(httpServletRequestMock.getParameter(IdentityCommonConstants.CLIENT_ID)).thenReturn("1234567"); - - PowerMockito.mockStatic(IdentityCommonUtil.class); - PowerMockito.when(IdentityCommonUtil.decodeRequestObjectAndGetKey(httpServletRequestMock, OAuth.OAUTH_SCOPE)) - .thenReturn("openid"); - - OBHybridResponseTypeValidator uut = spy(new OBHybridResponseTypeValidator()); - - // Act - uut.validateRequiredParameters(httpServletRequestMock); - } - - @Test - public void checkValidHybridResponseTypeValidationWithoutRequestURI() throws OAuthProblemException { - - // Mock - HttpServletRequest httpServletRequestMock = mock(HttpServletRequest.class); - when(httpServletRequestMock.getParameter(OAuth.OAUTH_SCOPE)).thenReturn("openid"); - when(httpServletRequestMock.getParameter(IdentityCommonConstants.CLIENT_ID)).thenReturn("1234567"); - when(httpServletRequestMock.getParameter(IdentityCommonConstants.RESPONSE_TYPE)).thenReturn("code id_token"); - when(httpServletRequestMock.getParameter(IdentityCommonConstants.REDIRECT_URI)).thenReturn("abc.com"); - - OBHybridResponseTypeValidator uut = spy(new OBHybridResponseTypeValidator()); - - // Act - uut.validateRequiredParameters(httpServletRequestMock); - } - - @Test - public void testValidateMethod() throws OAuthProblemException { - - // Mock - HttpServletRequest httpServletRequestMock = mock(HttpServletRequest.class); - when(httpServletRequestMock.getMethod()).thenReturn("POST"); - - OBHybridResponseTypeValidator uut = spy(new OBHybridResponseTypeValidator()); - - // Act - uut.validateMethod(httpServletRequestMock); - } - - @Test(expectedExceptions = OAuthProblemException.class) - public void testInvalidMethodScenario() throws OAuthProblemException { - - // Mock - HttpServletRequest httpServletRequestMock = mock(HttpServletRequest.class); - when(httpServletRequestMock.getMethod()).thenReturn("PUT"); - - OBHybridResponseTypeValidator uut = spy(new OBHybridResponseTypeValidator()); - - // Act - uut.validateMethod(httpServletRequestMock); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/authenticator/OBIdentifierAuthenticatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/authenticator/OBIdentifierAuthenticatorTest.java deleted file mode 100644 index 22e30e43..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/authenticator/OBIdentifierAuthenticatorTest.java +++ /dev/null @@ -1,1226 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.authenticator; - -import com.wso2.openbanking.accelerator.common.exception.OBThrottlerException; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.authenticator.util.OBIdentifierAuthenticatorTestData; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.util.HTTPClientUtils; -import com.wso2.openbanking.accelerator.throttler.service.OBThrottleService; -import org.apache.http.HttpEntity; -import org.apache.http.StatusLine; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpGet; -import org.apache.http.impl.client.CloseableHttpClient; -import org.mockito.Mockito; -import org.mockito.invocation.InvocationOnMock; -import org.mockito.stubbing.Answer; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeTest; -import org.testng.annotations.DataProvider; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.application.authentication.framework.AuthenticatorFlowStatus; -import org.wso2.carbon.identity.application.authentication.framework.config.ConfigurationFacade; -import org.wso2.carbon.identity.application.authentication.framework.config.builder.FileBasedConfigurationBuilder; -import org.wso2.carbon.identity.application.authentication.framework.config.model.ApplicationConfig; -import org.wso2.carbon.identity.application.authentication.framework.config.model.AuthenticatorConfig; -import org.wso2.carbon.identity.application.authentication.framework.config.model.SequenceConfig; -import org.wso2.carbon.identity.application.authentication.framework.context.AuthenticationContext; -import org.wso2.carbon.identity.application.authentication.framework.exception.AuthenticationFailedException; -import org.wso2.carbon.identity.application.authentication.framework.exception.InvalidCredentialsException; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedIdPData; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedUser; -import org.wso2.carbon.identity.application.authenticator.basicauth.BasicAuthenticator; -import org.wso2.carbon.identity.application.authenticator.basicauth.BasicAuthenticatorConstants; -import org.wso2.carbon.identity.core.model.IdentityErrorMsgContext; -import org.wso2.carbon.identity.core.util.IdentityTenantUtil; -import org.wso2.carbon.identity.core.util.IdentityUtil; -import org.wso2.carbon.user.api.RealmConfiguration; -import org.wso2.carbon.user.api.UserRealm; -import org.wso2.carbon.user.core.UserStoreException; -import org.wso2.carbon.user.core.UserStoreManager; -import org.wso2.carbon.user.core.service.RealmService; -import org.wso2.carbon.utils.multitenancy.MultitenantUtils; - -import java.io.BufferedReader; -import java.io.IOException; -import java.io.InputStream; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Map; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; - -import static org.mockito.Matchers.anyBoolean; -import static org.mockito.Matchers.anyString; -import static org.powermock.api.mockito.PowerMockito.doAnswer; -import static org.powermock.api.mockito.PowerMockito.mock; -import static org.powermock.api.mockito.PowerMockito.mockStatic; -import static org.powermock.api.mockito.PowerMockito.when; -import static org.testng.Assert.assertEquals; -import static org.testng.Assert.assertNotNull; - -/** - * Unit test cases for the OB Identifier Authenticator. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({FileBasedConfigurationBuilder.class, IdentityUtil.class, IdentityExtensionsDataHolder.class, - IdentityTenantUtil.class, MultitenantUtils.class, - AuthenticatedUser.class, HTTPClientUtils.class}) -public class OBIdentifierAuthenticatorTest { - - private HttpServletRequest mockRequest; - private HttpServletResponse mockResponse; - private AuthenticationContext mockAuthnCtxt; - private Map dummyIdpData = new HashMap<>(); - private ArrayList dummyAuthenticatorList = new ArrayList<>(); - private AuthenticatorConfig mockAuthConfig; - private BasicAuthenticator mockBasiAuthenticator; - private AuthenticatedIdPData mockAuthIdpData; - private Map dummyAuthParams = new HashMap<>(); - private FileBasedConfigurationBuilder mockFileBasedConfigurationBuilder; - private IdentityErrorMsgContext mockIdentityErrorMsgContext; - private IdentityExtensionsDataHolder identityExtensionsDataHolder; - private OBThrottleService obThrottleService; - private SequenceConfig mockSequenceConfig; - private ApplicationConfig mockApplicationConfig; - private RealmService mockRealmService; - private UserRealm mockUserRealm; - private org.wso2.carbon.user.core.UserRealm mockUserCoreRealm; - private UserStoreManager mockUserStoreManager; - private AuthenticatedUser authenticatedUser; - private Boolean isUserTenantDomainMismatch = true; - private String redirect; - private String dummyUserName = "dummyUserName"; - private String dummyQueryParam = "dummyQueryParams"; - private String dummyLoginPage = "dummyLoginPageurl"; - private String dummyString = "dummyString"; - private int dummyTenantId = -1234; - private String dummyVal = "dummyVal"; - private String dummySessionDataKey = "dummySessionDataKey"; - private OBIdentifierAuthenticator obIdentifierAuthenticator; - - @BeforeTest - public void setup() { - - obIdentifierAuthenticator = new OBIdentifierAuthenticator(); - mockBasiAuthenticator = new BasicAuthenticator(); - obThrottleService = mock(OBThrottleService.class); - identityExtensionsDataHolder = mock(IdentityExtensionsDataHolder.class); - } - - @DataProvider(name = "UsernameAndPasswordProvider") - public Object[][] getWrongUsernameAndPassword() { - - return new String[][]{ - {"admin@wso2.com", null, "true"}, - {null, "continue", "true"}, - {null, null, "false"}, - {"admin@wso2.com", "reset", "true"}, - {"", "", "true"} - }; - } - - @Test(dataProvider = "UsernameAndPasswordProvider") - public void canHandleTestCase(String userName, String identifierConsent, String expected) { - - mockRequest = mock(HttpServletRequest.class); - when(mockRequest.getParameter(BasicAuthenticatorConstants.USER_NAME)).thenReturn(userName); - when(mockRequest.getParameter("identifier_consent")).thenReturn(identifierConsent); - assertEquals(Boolean.valueOf(expected).booleanValue(), obIdentifierAuthenticator.canHandle(mockRequest), - "Invalid can handle response for the request."); - } - - @Test - public void processSuccessTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - when(mockAuthnCtxt.isLogoutRequest()).thenReturn(true); - assertEquals(obIdentifierAuthenticator.process(mockRequest, mockResponse, mockAuthnCtxt), - AuthenticatorFlowStatus.SUCCESS_COMPLETED); - } - - @Test - public void processSeamlessPromptFalseTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - - dummyIdpData.put("LOCAL", OBIdentifierAuthenticatorTestData.getSampleAuthenticatedIdPData()); - dummyAuthenticatorList.add(new AuthenticatorConfig()); - dummyAuthParams.put("promptConfirmationWindow", "false"); - - when(mockAuthnCtxt.isLogoutRequest()).thenReturn(false); - when(mockAuthnCtxt.getPreviousAuthenticatedIdPs()).thenReturn(dummyIdpData); - when(mockAuthnCtxt.getAuthenticatorParams(obIdentifierAuthenticator.getName())).thenReturn(dummyAuthParams); - assertEquals(obIdentifierAuthenticator.process(mockRequest, mockResponse, mockAuthnCtxt), - AuthenticatorFlowStatus.SUCCESS_COMPLETED); - } - - @Test - public void processSeamlessPromptTrueWithConsentContinueTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - - dummyIdpData.put("LOCAL", OBIdentifierAuthenticatorTestData.getSampleAuthenticatedIdPData()); - dummyAuthenticatorList.add(new AuthenticatorConfig()); - dummyAuthParams.put("promptConfirmationWindow", "true"); - - when(mockAuthnCtxt.isLogoutRequest()).thenReturn(false); - when(mockAuthnCtxt.getPreviousAuthenticatedIdPs()).thenReturn(dummyIdpData); - when(mockAuthnCtxt.getAuthenticatorParams(obIdentifierAuthenticator.getName())).thenReturn(dummyAuthParams); - when(mockRequest.getParameter(Mockito.anyString())).thenReturn("continue"); - assertEquals(obIdentifierAuthenticator.process(mockRequest, mockResponse, mockAuthnCtxt), - AuthenticatorFlowStatus.SUCCESS_COMPLETED); - } - - @Test - public void processSeamlessPromptTrueWithConsentNullTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - - dummyIdpData.put("LOCAL", OBIdentifierAuthenticatorTestData.getSampleAuthenticatedIdPData()); - dummyAuthenticatorList.add(new AuthenticatorConfig()); - dummyAuthParams.put("promptConfirmationWindow", "true"); - - when(mockAuthnCtxt.isLogoutRequest()).thenReturn(false); - when(mockAuthnCtxt.getPreviousAuthenticatedIdPs()).thenReturn(dummyIdpData); - when(mockAuthnCtxt.getAuthenticatorParams(obIdentifierAuthenticator.getName())).thenReturn(dummyAuthParams); - - mockStatic(FileBasedConfigurationBuilder.class); - mockFileBasedConfigurationBuilder = mock(FileBasedConfigurationBuilder.class); - when(FileBasedConfigurationBuilder.getInstance()).thenReturn(mockFileBasedConfigurationBuilder); - when(FileBasedConfigurationBuilder.getInstance().getAuthenticatorBean(anyString())).thenReturn(null); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - isUserTenantDomainMismatch = (Boolean) invocation.getArguments()[1]; - return null; - } - }).when(mockAuthnCtxt).setProperty(anyString(), anyBoolean()); - - when(ConfigurationFacade.getInstance().getAuthenticationEndpointURL()).thenReturn(dummyLoginPage); - when(mockAuthnCtxt.getContextIdIncludedQueryParams()).thenReturn(dummyQueryParam); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - redirect = (String) invocation.getArguments()[0]; - return null; - } - }).when(mockResponse).sendRedirect(anyString()); - - assertEquals(obIdentifierAuthenticator.process(mockRequest, mockResponse, mockAuthnCtxt), - AuthenticatorFlowStatus.INCOMPLETE); - } - - @Test(expectedExceptions = AuthenticationFailedException.class) - public void processSeamlessPromptTrueWithExceptionTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - - dummyIdpData.put("LOCAL", OBIdentifierAuthenticatorTestData.getSampleAuthenticatedIdPData()); - dummyAuthenticatorList.add(new AuthenticatorConfig()); - dummyAuthParams.put("promptConfirmationWindow", "true"); - - when(mockAuthnCtxt.isLogoutRequest()).thenReturn(false); - when(mockAuthnCtxt.getPreviousAuthenticatedIdPs()).thenReturn(dummyIdpData); - when(mockAuthnCtxt.getAuthenticatorParams(obIdentifierAuthenticator.getName())).thenReturn(dummyAuthParams); - - mockStatic(FileBasedConfigurationBuilder.class); - mockFileBasedConfigurationBuilder = mock(FileBasedConfigurationBuilder.class); - when(FileBasedConfigurationBuilder.getInstance()).thenReturn(mockFileBasedConfigurationBuilder); - when(FileBasedConfigurationBuilder.getInstance().getAuthenticatorBean(anyString())).thenReturn(null); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - isUserTenantDomainMismatch = (Boolean) invocation.getArguments()[1]; - return null; - } - }).when(mockAuthnCtxt).setProperty(anyString(), anyBoolean()); - - when(ConfigurationFacade.getInstance().getAuthenticationEndpointURL()).thenReturn(dummyLoginPage); - when(mockAuthnCtxt.getContextIdIncludedQueryParams()).thenReturn(dummyQueryParam); - Mockito.doThrow(IOException.class).when(mockResponse).sendRedirect(anyString()); - - obIdentifierAuthenticator.process(mockRequest, mockResponse, mockAuthnCtxt); - } - - @Test - public void initiateAuthenticationRequestGeneralTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - - AuthenticatorConfig authenticatorConfig = new AuthenticatorConfig(); - Map paramMap = new HashMap<>(); - paramMap.put("throttleLimit", "3"); - paramMap.put("throttleTimePeriod", "180"); - paramMap.put("showAuthFailureReason", "true"); - - authenticatorConfig.setParameterMap(paramMap); - - mockStatic(FileBasedConfigurationBuilder.class); - mockFileBasedConfigurationBuilder = mock(FileBasedConfigurationBuilder.class); - when(FileBasedConfigurationBuilder.getInstance()).thenReturn(mockFileBasedConfigurationBuilder); - when(mockFileBasedConfigurationBuilder.getAuthenticatorBean(anyString())).thenReturn(authenticatorConfig); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - isUserTenantDomainMismatch = (Boolean) invocation.getArguments()[1]; - return null; - } - }).when(mockAuthnCtxt).setProperty(anyString(), anyBoolean()); - - when(ConfigurationFacade.getInstance().getAuthenticationEndpointURL()).thenReturn(dummyLoginPage); - when(mockAuthnCtxt.getContextIdIncludedQueryParams()).thenReturn(dummyQueryParam); - when(mockAuthnCtxt.isRetrying()).thenReturn(true); - - mockStatic(IdentityUtil.class); - when(IdentityUtil.getClientIpAddress(mockRequest)).thenReturn("127.0.0.1"); - - - when(obThrottleService.isThrottled(Mockito.anyString(), Mockito.anyString())).thenReturn(true); - Mockito.doNothing().when(obThrottleService) - .updateThrottleData(Mockito.anyString(), Mockito.anyString(), Mockito.anyInt(), Mockito.anyInt()); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - redirect = (String) invocation.getArguments()[0]; - return null; - } - }).when(mockResponse).sendRedirect(anyString()); - - mockIdentityErrorMsgContext = mock(IdentityErrorMsgContext.class); - when(mockIdentityErrorMsgContext.getErrorCode()).thenReturn("dummyErrorCode"); - when(IdentityUtil.getIdentityErrorMsg()).thenReturn(mockIdentityErrorMsgContext); - - mockStatic(IdentityExtensionsDataHolder.class); - when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolder); - when(identityExtensionsDataHolder.getOBThrottleService()).thenReturn(obThrottleService); - when(mockRequest.getParameter("username")).thenReturn(dummyUserName); - - obIdentifierAuthenticator.initiateAuthenticationRequest(mockRequest, mockResponse, mockAuthnCtxt); - } - - @Test - public void initiateAuthenticationRequestIsThrottledFalseTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - - AuthenticatorConfig authenticatorConfig = new AuthenticatorConfig(); - Map paramMap = new HashMap<>(); - paramMap.put("throttleLimit", "3"); - paramMap.put("throttleTimePeriod", "180"); - paramMap.put("showAuthFailureReason", "true"); - - authenticatorConfig.setParameterMap(paramMap); - - mockStatic(FileBasedConfigurationBuilder.class); - mockFileBasedConfigurationBuilder = mock(FileBasedConfigurationBuilder.class); - when(FileBasedConfigurationBuilder.getInstance()).thenReturn(mockFileBasedConfigurationBuilder); - when(mockFileBasedConfigurationBuilder.getAuthenticatorBean(anyString())).thenReturn(authenticatorConfig); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - isUserTenantDomainMismatch = (Boolean) invocation.getArguments()[1]; - return null; - } - }).when(mockAuthnCtxt).setProperty(anyString(), anyBoolean()); - - when(ConfigurationFacade.getInstance().getAuthenticationEndpointURL()).thenReturn(dummyLoginPage); - when(mockAuthnCtxt.getContextIdIncludedQueryParams()).thenReturn(dummyQueryParam); - when(mockAuthnCtxt.isRetrying()).thenReturn(true); - - mockStatic(IdentityUtil.class); - when(IdentityUtil.getClientIpAddress(mockRequest)).thenReturn("127.0.0.1"); - when(mockAuthnCtxt.getProperty("InvalidEmailUsername")).thenReturn(true); - - - when(obThrottleService.isThrottled(Mockito.anyString(), Mockito.anyString())).thenReturn(false); - Mockito.doNothing().when(obThrottleService) - .updateThrottleData(Mockito.anyString(), Mockito.anyString(), Mockito.anyInt(), Mockito.anyInt()); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - redirect = (String) invocation.getArguments()[0]; - return null; - } - }).when(mockResponse).sendRedirect(anyString()); - - mockIdentityErrorMsgContext = mock(IdentityErrorMsgContext.class); - when(mockIdentityErrorMsgContext.getErrorCode()).thenReturn("dummyErrorCode"); - when(IdentityUtil.getIdentityErrorMsg()).thenReturn(mockIdentityErrorMsgContext); - - mockStatic(IdentityExtensionsDataHolder.class); - when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolder); - when(identityExtensionsDataHolder.getOBThrottleService()).thenReturn(obThrottleService); - when(mockRequest.getParameter("username")).thenReturn(dummyUserName); - - obIdentifierAuthenticator.initiateAuthenticationRequest(mockRequest, mockResponse, mockAuthnCtxt); - } - - @Test - public void initiateAuthenticationRequestAuthFailureFalseTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - - AuthenticatorConfig authenticatorConfig = new AuthenticatorConfig(); - Map paramMap = new HashMap<>(); - paramMap.put("throttleLimit", "3"); - paramMap.put("throttleTimePeriod", "180"); - paramMap.put("showAuthFailureReason", "false"); - - authenticatorConfig.setParameterMap(paramMap); - - mockStatic(FileBasedConfigurationBuilder.class); - mockFileBasedConfigurationBuilder = mock(FileBasedConfigurationBuilder.class); - when(FileBasedConfigurationBuilder.getInstance()).thenReturn(mockFileBasedConfigurationBuilder); - when(mockFileBasedConfigurationBuilder.getAuthenticatorBean(anyString())).thenReturn(authenticatorConfig); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - isUserTenantDomainMismatch = (Boolean) invocation.getArguments()[1]; - return null; - } - }).when(mockAuthnCtxt).setProperty(anyString(), anyBoolean()); - - when(ConfigurationFacade.getInstance().getAuthenticationEndpointURL()).thenReturn(dummyLoginPage); - when(mockAuthnCtxt.getContextIdIncludedQueryParams()).thenReturn(dummyQueryParam); - when(mockAuthnCtxt.isRetrying()).thenReturn(true); - - mockStatic(IdentityUtil.class); - when(IdentityUtil.getClientIpAddress(mockRequest)).thenReturn("127.0.0.1"); - - - when(obThrottleService.isThrottled(Mockito.anyString(), Mockito.anyString())).thenReturn(true); - Mockito.doNothing().when(obThrottleService) - .updateThrottleData(Mockito.anyString(), Mockito.anyString(), Mockito.anyInt(), Mockito.anyInt()); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - redirect = (String) invocation.getArguments()[0]; - return null; - } - }).when(mockResponse).sendRedirect(anyString()); - - mockIdentityErrorMsgContext = mock(IdentityErrorMsgContext.class); - when(mockIdentityErrorMsgContext.getErrorCode()).thenReturn("dummyErrorCode"); - when(IdentityUtil.getIdentityErrorMsg()).thenReturn(mockIdentityErrorMsgContext); - - mockStatic(IdentityExtensionsDataHolder.class); - when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolder); - when(identityExtensionsDataHolder.getOBThrottleService()).thenReturn(obThrottleService); - when(mockRequest.getParameter("username")).thenReturn(dummyUserName); - - obIdentifierAuthenticator.initiateAuthenticationRequest(mockRequest, mockResponse, mockAuthnCtxt); - } - - @Test - public void initiateAuthenticationRequestNullErrorContextTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - - AuthenticatorConfig authenticatorConfig = new AuthenticatorConfig(); - Map paramMap = new HashMap<>(); - paramMap.put("throttleLimit", "3"); - paramMap.put("throttleTimePeriod", "180"); - paramMap.put("showAuthFailureReason", "false"); - - authenticatorConfig.setParameterMap(paramMap); - - mockStatic(FileBasedConfigurationBuilder.class); - mockFileBasedConfigurationBuilder = mock(FileBasedConfigurationBuilder.class); - when(FileBasedConfigurationBuilder.getInstance()).thenReturn(mockFileBasedConfigurationBuilder); - when(mockFileBasedConfigurationBuilder.getAuthenticatorBean(anyString())).thenReturn(authenticatorConfig); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - isUserTenantDomainMismatch = (Boolean) invocation.getArguments()[1]; - return null; - } - }).when(mockAuthnCtxt).setProperty(anyString(), anyBoolean()); - - when(ConfigurationFacade.getInstance().getAuthenticationEndpointURL()).thenReturn(dummyLoginPage); - when(mockAuthnCtxt.getContextIdIncludedQueryParams()).thenReturn(dummyQueryParam); - when(mockAuthnCtxt.isRetrying()).thenReturn(true); - - mockStatic(IdentityUtil.class); - when(IdentityUtil.getClientIpAddress(mockRequest)).thenReturn("127.0.0.1"); - - - when(obThrottleService.isThrottled(Mockito.anyString(), Mockito.anyString())).thenReturn(true); - Mockito.doNothing().when(obThrottleService) - .updateThrottleData(Mockito.anyString(), Mockito.anyString(), Mockito.anyInt(), Mockito.anyInt()); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - redirect = (String) invocation.getArguments()[0]; - return null; - } - }).when(mockResponse).sendRedirect(anyString()); - - mockStatic(IdentityExtensionsDataHolder.class); - when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolder); - when(identityExtensionsDataHolder.getOBThrottleService()).thenReturn(obThrottleService); - when(mockRequest.getParameter("username")).thenReturn(dummyUserName); - - obIdentifierAuthenticator.initiateAuthenticationRequest(mockRequest, mockResponse, mockAuthnCtxt); - } - - @Test(expectedExceptions = AuthenticationFailedException.class) - public void initiateAuthenticationRequestThrottlerExceptionTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - - AuthenticatorConfig authenticatorConfig = new AuthenticatorConfig(); - Map paramMap = new HashMap<>(); - paramMap.put("throttleLimit", "3"); - paramMap.put("throttleTimePeriod", "180"); - paramMap.put("showAuthFailureReason", "false"); - - authenticatorConfig.setParameterMap(paramMap); - - mockStatic(FileBasedConfigurationBuilder.class); - mockFileBasedConfigurationBuilder = mock(FileBasedConfigurationBuilder.class); - when(FileBasedConfigurationBuilder.getInstance()).thenReturn(mockFileBasedConfigurationBuilder); - when(mockFileBasedConfigurationBuilder.getAuthenticatorBean(anyString())).thenReturn(authenticatorConfig); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - isUserTenantDomainMismatch = (Boolean) invocation.getArguments()[1]; - return null; - } - }).when(mockAuthnCtxt).setProperty(anyString(), anyBoolean()); - - when(ConfigurationFacade.getInstance().getAuthenticationEndpointURL()).thenReturn(dummyLoginPage); - when(mockAuthnCtxt.getContextIdIncludedQueryParams()).thenReturn(dummyQueryParam); - when(mockAuthnCtxt.isRetrying()).thenReturn(true); - - mockStatic(IdentityUtil.class); - when(IdentityUtil.getClientIpAddress(mockRequest)).thenReturn("127.0.0.1"); - - - when(obThrottleService.isThrottled(Mockito.anyString(), Mockito.anyString())).thenReturn(true); - Mockito.doThrow(OBThrottlerException.class).when(obThrottleService) - .updateThrottleData(Mockito.anyString(), Mockito.anyString(), Mockito.anyInt(), Mockito.anyInt()); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - redirect = (String) invocation.getArguments()[0]; - return null; - } - }).when(mockResponse).sendRedirect(anyString()); - - mockStatic(IdentityExtensionsDataHolder.class); - when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolder); - when(identityExtensionsDataHolder.getOBThrottleService()).thenReturn(obThrottleService); - when(mockRequest.getParameter("username")).thenReturn(dummyUserName); - - obIdentifierAuthenticator.initiateAuthenticationRequest(mockRequest, mockResponse, mockAuthnCtxt); - } - - @Test - public void initiateAuthenticationRequestAccountNotConfirmedErrorCodeTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - - AuthenticatorConfig authenticatorConfig = new AuthenticatorConfig(); - Map paramMap = new HashMap<>(); - paramMap.put("throttleLimit", "3"); - paramMap.put("throttleTimePeriod", "180"); - paramMap.put("showAuthFailureReason", "false"); - - authenticatorConfig.setParameterMap(paramMap); - - mockStatic(FileBasedConfigurationBuilder.class); - mockFileBasedConfigurationBuilder = mock(FileBasedConfigurationBuilder.class); - when(FileBasedConfigurationBuilder.getInstance()).thenReturn(mockFileBasedConfigurationBuilder); - when(mockFileBasedConfigurationBuilder.getAuthenticatorBean(anyString())).thenReturn(authenticatorConfig); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - isUserTenantDomainMismatch = (Boolean) invocation.getArguments()[1]; - return null; - } - }).when(mockAuthnCtxt).setProperty(anyString(), anyBoolean()); - - when(ConfigurationFacade.getInstance().getAuthenticationEndpointURL()).thenReturn(dummyLoginPage); - when(mockAuthnCtxt.getContextIdIncludedQueryParams()).thenReturn(dummyQueryParam); - when(mockAuthnCtxt.isRetrying()).thenReturn(true); - - mockStatic(IdentityUtil.class); - when(IdentityUtil.getClientIpAddress(mockRequest)).thenReturn("127.0.0.1"); - - Map mockedThreadLocalMap = new HashMap<>(); - mockedThreadLocalMap.put("user-domain-recaptcha", dummyVal); - IdentityUtil.threadLocalProperties.set(mockedThreadLocalMap); - when(IdentityUtil.addDomainToName(Mockito.anyString(), Mockito.anyString())).thenReturn(dummyUserName); - - mockIdentityErrorMsgContext = mock(IdentityErrorMsgContext.class); - when(mockIdentityErrorMsgContext.getErrorCode()).thenReturn("17005"); - when(IdentityUtil.getIdentityErrorMsg()).thenReturn(mockIdentityErrorMsgContext); - - when(obThrottleService.isThrottled(Mockito.anyString(), Mockito.anyString())).thenReturn(true); - Mockito.doNothing().when(obThrottleService) - .updateThrottleData(Mockito.anyString(), Mockito.anyString(), Mockito.anyInt(), Mockito.anyInt()); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - redirect = (String) invocation.getArguments()[0]; - return null; - } - }).when(mockResponse).sendRedirect(anyString()); - - mockStatic(IdentityExtensionsDataHolder.class); - when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolder); - when(identityExtensionsDataHolder.getOBThrottleService()).thenReturn(obThrottleService); - when(mockRequest.getParameter("username")).thenReturn(dummyUserName); - - obIdentifierAuthenticator.initiateAuthenticationRequest(mockRequest, mockResponse, mockAuthnCtxt); - } - - @Test - public void initiateAuthenticationRequestInvalidClientErrorCodeTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - - AuthenticatorConfig authenticatorConfig = new AuthenticatorConfig(); - Map paramMap = new HashMap<>(); - paramMap.put("throttleLimit", "3"); - paramMap.put("throttleTimePeriod", "180"); - paramMap.put("showAuthFailureReason", "true"); - - authenticatorConfig.setParameterMap(paramMap); - - mockStatic(FileBasedConfigurationBuilder.class); - mockFileBasedConfigurationBuilder = mock(FileBasedConfigurationBuilder.class); - when(FileBasedConfigurationBuilder.getInstance()).thenReturn(mockFileBasedConfigurationBuilder); - when(mockFileBasedConfigurationBuilder.getAuthenticatorBean(anyString())).thenReturn(authenticatorConfig); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - isUserTenantDomainMismatch = (Boolean) invocation.getArguments()[1]; - return null; - } - }).when(mockAuthnCtxt).setProperty(anyString(), anyBoolean()); - - when(ConfigurationFacade.getInstance().getAuthenticationEndpointURL()).thenReturn(dummyLoginPage); - when(mockAuthnCtxt.getContextIdIncludedQueryParams()).thenReturn(dummyQueryParam); - when(mockAuthnCtxt.isRetrying()).thenReturn(true); - - mockStatic(IdentityUtil.class); - when(IdentityUtil.getClientIpAddress(mockRequest)).thenReturn("127.0.0.1"); - - mockIdentityErrorMsgContext = mock(IdentityErrorMsgContext.class); - when(mockIdentityErrorMsgContext.getErrorCode()).thenReturn("17002"); - when(IdentityUtil.getIdentityErrorMsg()).thenReturn(mockIdentityErrorMsgContext); - - when(obThrottleService.isThrottled(Mockito.anyString(), Mockito.anyString())).thenReturn(true); - Mockito.doNothing().when(obThrottleService) - .updateThrottleData(Mockito.anyString(), Mockito.anyString(), Mockito.anyInt(), Mockito.anyInt()); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - redirect = (String) invocation.getArguments()[0]; - return null; - } - }).when(mockResponse).sendRedirect(anyString()); - - mockStatic(IdentityExtensionsDataHolder.class); - when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolder); - when(identityExtensionsDataHolder.getOBThrottleService()).thenReturn(obThrottleService); - when(mockRequest.getParameter("username")).thenReturn(dummyUserName); - - obIdentifierAuthenticator.initiateAuthenticationRequest(mockRequest, mockResponse, mockAuthnCtxt); - } - - @Test - public void initiateAuthenticationRequestLockedUserWithNullReasonTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - - AuthenticatorConfig authenticatorConfig = new AuthenticatorConfig(); - Map paramMap = new HashMap<>(); - paramMap.put("throttleLimit", "3"); - paramMap.put("throttleTimePeriod", "180"); - paramMap.put("showAuthFailureReason", "true"); - - authenticatorConfig.setParameterMap(paramMap); - - mockStatic(FileBasedConfigurationBuilder.class); - mockFileBasedConfigurationBuilder = mock(FileBasedConfigurationBuilder.class); - when(FileBasedConfigurationBuilder.getInstance()).thenReturn(mockFileBasedConfigurationBuilder); - when(mockFileBasedConfigurationBuilder.getAuthenticatorBean(anyString())).thenReturn(authenticatorConfig); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - isUserTenantDomainMismatch = (Boolean) invocation.getArguments()[1]; - return null; - } - }).when(mockAuthnCtxt).setProperty(anyString(), anyBoolean()); - - when(ConfigurationFacade.getInstance().getAuthenticationEndpointURL()).thenReturn(dummyLoginPage); - when(mockAuthnCtxt.getContextIdIncludedQueryParams()).thenReturn(dummyQueryParam); - when(mockAuthnCtxt.isRetrying()).thenReturn(true); - - mockStatic(IdentityUtil.class); - when(IdentityUtil.getClientIpAddress(mockRequest)).thenReturn("127.0.0.1"); - - mockIdentityErrorMsgContext = mock(IdentityErrorMsgContext.class); - when(mockIdentityErrorMsgContext.getErrorCode()).thenReturn("17003"); - when(IdentityUtil.getIdentityErrorMsg()).thenReturn(mockIdentityErrorMsgContext); - - when(obThrottleService.isThrottled(Mockito.anyString(), Mockito.anyString())).thenReturn(true); - Mockito.doNothing().when(obThrottleService) - .updateThrottleData(Mockito.anyString(), Mockito.anyString(), Mockito.anyInt(), Mockito.anyInt()); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - redirect = (String) invocation.getArguments()[0]; - return null; - } - }).when(mockResponse).sendRedirect(anyString()); - - mockStatic(IdentityExtensionsDataHolder.class); - when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolder); - when(identityExtensionsDataHolder.getOBThrottleService()).thenReturn(obThrottleService); - when(mockRequest.getParameter("username")).thenReturn(dummyUserName); - - obIdentifierAuthenticator.initiateAuthenticationRequest(mockRequest, mockResponse, mockAuthnCtxt); - } - - @Test - public void initiateAuthenticationRequestLockedUserWithReasonTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - - AuthenticatorConfig authenticatorConfig = new AuthenticatorConfig(); - Map paramMap = new HashMap<>(); - paramMap.put("throttleLimit", "3"); - paramMap.put("throttleTimePeriod", "180"); - paramMap.put("showAuthFailureReason", "true"); - - authenticatorConfig.setParameterMap(paramMap); - - mockStatic(FileBasedConfigurationBuilder.class); - mockFileBasedConfigurationBuilder = mock(FileBasedConfigurationBuilder.class); - when(FileBasedConfigurationBuilder.getInstance()).thenReturn(mockFileBasedConfigurationBuilder); - when(mockFileBasedConfigurationBuilder.getAuthenticatorBean(anyString())).thenReturn(authenticatorConfig); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - isUserTenantDomainMismatch = (Boolean) invocation.getArguments()[1]; - return null; - } - }).when(mockAuthnCtxt).setProperty(anyString(), anyBoolean()); - - when(ConfigurationFacade.getInstance().getAuthenticationEndpointURL()).thenReturn(dummyLoginPage); - when(mockAuthnCtxt.getContextIdIncludedQueryParams()).thenReturn(dummyQueryParam); - when(mockAuthnCtxt.isRetrying()).thenReturn(true); - - mockStatic(IdentityUtil.class); - when(IdentityUtil.getClientIpAddress(mockRequest)).thenReturn("127.0.0.1"); - - mockIdentityErrorMsgContext = mock(IdentityErrorMsgContext.class); - when(mockIdentityErrorMsgContext.getErrorCode()).thenReturn("17003:reason"); - when(IdentityUtil.getIdentityErrorMsg()).thenReturn(mockIdentityErrorMsgContext); - - when(obThrottleService.isThrottled(Mockito.anyString(), Mockito.anyString())).thenReturn(true); - Mockito.doNothing().when(obThrottleService) - .updateThrottleData(Mockito.anyString(), Mockito.anyString(), Mockito.anyInt(), Mockito.anyInt()); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - redirect = (String) invocation.getArguments()[0]; - return null; - } - }).when(mockResponse).sendRedirect(anyString()); - - mockStatic(IdentityExtensionsDataHolder.class); - when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolder); - when(identityExtensionsDataHolder.getOBThrottleService()).thenReturn(obThrottleService); - when(mockRequest.getParameter("username")).thenReturn(dummyUserName); - - obIdentifierAuthenticator.initiateAuthenticationRequest(mockRequest, mockResponse, mockAuthnCtxt); - } - - @Test - public void initiateAuthenticationRequestLockedUserWithRetryAttemptsAndNullReasonTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - - AuthenticatorConfig authenticatorConfig = new AuthenticatorConfig(); - Map paramMap = new HashMap<>(); - paramMap.put("throttleLimit", "3"); - paramMap.put("throttleTimePeriod", "180"); - paramMap.put("showAuthFailureReason", "true"); - - authenticatorConfig.setParameterMap(paramMap); - - mockStatic(FileBasedConfigurationBuilder.class); - mockFileBasedConfigurationBuilder = mock(FileBasedConfigurationBuilder.class); - when(FileBasedConfigurationBuilder.getInstance()).thenReturn(mockFileBasedConfigurationBuilder); - when(mockFileBasedConfigurationBuilder.getAuthenticatorBean(anyString())).thenReturn(authenticatorConfig); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - isUserTenantDomainMismatch = (Boolean) invocation.getArguments()[1]; - return null; - } - }).when(mockAuthnCtxt).setProperty(anyString(), anyBoolean()); - - when(ConfigurationFacade.getInstance().getAuthenticationEndpointURL()).thenReturn(dummyLoginPage); - when(mockAuthnCtxt.getContextIdIncludedQueryParams()).thenReturn(dummyQueryParam); - when(mockAuthnCtxt.isRetrying()).thenReturn(true); - - mockStatic(IdentityUtil.class); - when(IdentityUtil.getClientIpAddress(mockRequest)).thenReturn("127.0.0.1"); - - mockIdentityErrorMsgContext = mock(IdentityErrorMsgContext.class); - when(mockIdentityErrorMsgContext.getErrorCode()).thenReturn("17003"); - when(IdentityUtil.getIdentityErrorMsg()).thenReturn(mockIdentityErrorMsgContext); - - when(mockIdentityErrorMsgContext.getMaximumLoginAttempts()).thenReturn(4); - when(mockIdentityErrorMsgContext.getFailedLoginAttempts()).thenReturn(2); - - when(obThrottleService.isThrottled(Mockito.anyString(), Mockito.anyString())).thenReturn(true); - Mockito.doNothing().when(obThrottleService) - .updateThrottleData(Mockito.anyString(), Mockito.anyString(), Mockito.anyInt(), Mockito.anyInt()); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - redirect = (String) invocation.getArguments()[0]; - return null; - } - }).when(mockResponse).sendRedirect(anyString()); - - mockStatic(IdentityExtensionsDataHolder.class); - when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolder); - when(identityExtensionsDataHolder.getOBThrottleService()).thenReturn(obThrottleService); - when(mockRequest.getParameter("username")).thenReturn(dummyUserName); - - obIdentifierAuthenticator.initiateAuthenticationRequest(mockRequest, mockResponse, mockAuthnCtxt); - } - - @Test - public void initiateAuthenticationRequestLockedUserWithRetryAttemptsAndReasonTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - - AuthenticatorConfig authenticatorConfig = new AuthenticatorConfig(); - Map paramMap = new HashMap<>(); - paramMap.put("throttleLimit", "3"); - paramMap.put("throttleTimePeriod", "180"); - paramMap.put("showAuthFailureReason", "true"); - - authenticatorConfig.setParameterMap(paramMap); - - mockStatic(FileBasedConfigurationBuilder.class); - mockFileBasedConfigurationBuilder = mock(FileBasedConfigurationBuilder.class); - when(FileBasedConfigurationBuilder.getInstance()).thenReturn(mockFileBasedConfigurationBuilder); - when(mockFileBasedConfigurationBuilder.getAuthenticatorBean(anyString())).thenReturn(authenticatorConfig); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - isUserTenantDomainMismatch = (Boolean) invocation.getArguments()[1]; - return null; - } - }).when(mockAuthnCtxt).setProperty(anyString(), anyBoolean()); - - when(ConfigurationFacade.getInstance().getAuthenticationEndpointURL()).thenReturn(dummyLoginPage); - when(mockAuthnCtxt.getContextIdIncludedQueryParams()).thenReturn(dummyQueryParam); - when(mockAuthnCtxt.isRetrying()).thenReturn(true); - - mockStatic(IdentityUtil.class); - when(IdentityUtil.getClientIpAddress(mockRequest)).thenReturn("127.0.0.1"); - - mockIdentityErrorMsgContext = mock(IdentityErrorMsgContext.class); - when(mockIdentityErrorMsgContext.getErrorCode()).thenReturn("17003:reason"); - when(IdentityUtil.getIdentityErrorMsg()).thenReturn(mockIdentityErrorMsgContext); - - when(mockIdentityErrorMsgContext.getMaximumLoginAttempts()).thenReturn(4); - when(mockIdentityErrorMsgContext.getFailedLoginAttempts()).thenReturn(2); - - when(obThrottleService.isThrottled(Mockito.anyString(), Mockito.anyString())).thenReturn(true); - Mockito.doNothing().when(obThrottleService) - .updateThrottleData(Mockito.anyString(), Mockito.anyString(), Mockito.anyInt(), Mockito.anyInt()); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - redirect = (String) invocation.getArguments()[0]; - return null; - } - }).when(mockResponse).sendRedirect(anyString()); - - mockStatic(IdentityExtensionsDataHolder.class); - when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolder); - when(identityExtensionsDataHolder.getOBThrottleService()).thenReturn(obThrottleService); - when(mockRequest.getParameter("username")).thenReturn(dummyUserName); - - obIdentifierAuthenticator.initiateAuthenticationRequest(mockRequest, mockResponse, mockAuthnCtxt); - } - - @Test - public void processAuthenticationResponseGeneralTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - mockSequenceConfig = mock(SequenceConfig.class); - mockApplicationConfig = mock(ApplicationConfig.class); - mockRealmService = mock(RealmService.class); - mockUserRealm = mock(UserRealm.class); - mockUserStoreManager = mock(UserStoreManager.class); - authenticatedUser = mock(AuthenticatedUser.class); - - AuthenticatorConfig authenticatorConfig = new AuthenticatorConfig(); - Map paramMap = new HashMap<>(); - paramMap.put("ValidateUsername", "true"); - - authenticatorConfig.setParameterMap(paramMap); - - mockStatic(FileBasedConfigurationBuilder.class); - mockFileBasedConfigurationBuilder = mock(FileBasedConfigurationBuilder.class); - when(FileBasedConfigurationBuilder.getInstance()).thenReturn(mockFileBasedConfigurationBuilder); - when(mockFileBasedConfigurationBuilder.getAuthenticatorBean(anyString())).thenReturn(authenticatorConfig); - - mockStatic(IdentityUtil.class); - when(IdentityUtil.getClientIpAddress(mockRequest)).thenReturn("127.0.0.1"); - - when(obThrottleService.isThrottled(Mockito.anyString(), Mockito.anyString())).thenReturn(false); - - mockStatic(IdentityExtensionsDataHolder.class); - when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolder); - when(identityExtensionsDataHolder.getOBThrottleService()).thenReturn(obThrottleService); - when(mockRequest.getParameter("username")).thenReturn(dummyUserName); - - when(mockAuthnCtxt.getSequenceConfig()).thenReturn(mockSequenceConfig); - when(mockSequenceConfig.getApplicationConfig()).thenReturn(mockApplicationConfig); - when(mockApplicationConfig.isSaaSApp()).thenReturn(true); - - mockStatic(IdentityTenantUtil.class); - when(IdentityTenantUtil.getTenantId(Mockito.anyString())).thenReturn(dummyTenantId); - - when(identityExtensionsDataHolder.getRealmService()).thenReturn(mockRealmService); - when(mockRealmService.getTenantUserRealm(Mockito.anyInt())).thenReturn(mockUserRealm); - when(mockUserRealm.getUserStoreManager()).thenReturn(mockUserStoreManager); - - mockStatic(MultitenantUtils.class); - when(MultitenantUtils.getTenantAwareUsername(Mockito.anyString())).thenReturn(dummyUserName); - when(mockUserStoreManager.isExistingUser(Mockito.anyString())).thenReturn(true); - - mockStatic(AuthenticatedUser.class); - when(AuthenticatedUser - .createLocalAuthenticatedUserFromSubjectIdentifier(Mockito.anyString())).thenReturn(authenticatedUser); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - authenticatedUser = (AuthenticatedUser) invocation.getArguments()[0]; - return null; - } - }).when(mockAuthnCtxt).setSubject(authenticatedUser); - - obIdentifierAuthenticator.processAuthenticationResponse(mockRequest, mockResponse, mockAuthnCtxt); - } - - @Test(expectedExceptions = InvalidCredentialsException.class, priority = 1) - public void processAuthenticationResponseNotSaasAppAndInvalidTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - mockSequenceConfig = mock(SequenceConfig.class); - mockApplicationConfig = mock(ApplicationConfig.class); - - mockStatic(IdentityUtil.class); - when(IdentityUtil.isEmailUsernameEnabled()).thenReturn(true); - - when(mockAuthnCtxt.getSequenceConfig()).thenReturn(mockSequenceConfig); - when(mockSequenceConfig.getApplicationConfig()).thenReturn(mockApplicationConfig); - when(mockApplicationConfig.isSaaSApp()).thenReturn(false); - - mockStatic(MultitenantUtils.class); - when(MultitenantUtils.getTenantDomain(anyString())).thenReturn("carbon.super"); - when(MultitenantUtils.getTenantAwareUsername(Mockito.anyString())).thenReturn(dummyUserName); - - obIdentifierAuthenticator.processAuthenticationResponse(mockRequest, mockResponse, mockAuthnCtxt); - } - - @Test(expectedExceptions = AuthenticationFailedException.class, priority = 1) - public void processAuthenticationResponseEmaiEnabledUsernameWithThrottleExceptionTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - mockSequenceConfig = mock(SequenceConfig.class); - mockApplicationConfig = mock(ApplicationConfig.class); - - mockStatic(IdentityUtil.class); - when(IdentityUtil.isEmailUsernameEnabled()).thenReturn(true); - when(IdentityUtil.getClientIpAddress(mockRequest)).thenReturn("127.0.0.1"); - when(mockRequest.getParameter("username")).thenReturn("admin@wso2.com"); - - mockStatic(IdentityExtensionsDataHolder.class); - when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolder); - when(identityExtensionsDataHolder.getOBThrottleService()).thenReturn(obThrottleService); - - Mockito.doThrow(OBThrottlerException.class) - .when(obThrottleService).isThrottled(Mockito.anyString(), Mockito.anyString()); - - when(mockAuthnCtxt.getSequenceConfig()).thenReturn(mockSequenceConfig); - when(mockSequenceConfig.getApplicationConfig()).thenReturn(mockApplicationConfig); - when(mockApplicationConfig.isSaaSApp()).thenReturn(false); - - mockStatic(MultitenantUtils.class); - when(MultitenantUtils.getTenantDomain(anyString())).thenReturn("carbon.super"); - when(MultitenantUtils.getTenantAwareUsername(Mockito.anyString())).thenReturn("admin@wso2.com"); - - obIdentifierAuthenticator.processAuthenticationResponse(mockRequest, mockResponse, mockAuthnCtxt); - } - - @Test(expectedExceptions = AuthenticationFailedException.class, priority = 1) - public void processAuthenticationResponseNonEmailUserNameWithThrottleExceptionTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - mockResponse = mock(HttpServletResponse.class); - mockAuthnCtxt = mock(AuthenticationContext.class); - mockSequenceConfig = mock(SequenceConfig.class); - mockApplicationConfig = mock(ApplicationConfig.class); - - mockStatic(IdentityUtil.class); - when(IdentityUtil.isEmailUsernameEnabled()).thenReturn(false); - when(IdentityUtil.getClientIpAddress(mockRequest)).thenReturn("127.0.0.1"); - when(mockRequest.getParameter("username")).thenReturn(dummyUserName); - - mockStatic(IdentityExtensionsDataHolder.class); - when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolder); - when(identityExtensionsDataHolder.getOBThrottleService()).thenReturn(obThrottleService); - - Mockito.doThrow(OBThrottlerException.class) - .when(obThrottleService).isThrottled(Mockito.anyString(), Mockito.anyString()); - - when(mockAuthnCtxt.getSequenceConfig()).thenReturn(mockSequenceConfig); - when(mockSequenceConfig.getApplicationConfig()).thenReturn(mockApplicationConfig); - when(mockApplicationConfig.isSaaSApp()).thenReturn(false); - - mockStatic(MultitenantUtils.class); - when(MultitenantUtils.getTenantDomain(anyString())).thenReturn("carbon.super"); - when(MultitenantUtils.getTenantAwareUsername(Mockito.anyString())).thenReturn(dummyUserName); - - obIdentifierAuthenticator.processAuthenticationResponse(mockRequest, mockResponse, mockAuthnCtxt); - } - - @Test - public void retryAuthenticationEnabledTestCase() throws Exception { - - assertEquals(obIdentifierAuthenticator.retryAuthenticationEnabled(), true); - } - - @Test - public void getContextIdentifierTestCase() throws Exception { - - mockRequest = mock(HttpServletRequest.class); - when(mockRequest.getParameter(Mockito.anyString())).thenReturn("sessionData"); - - assertNotNull(obIdentifierAuthenticator.getContextIdentifier(mockRequest)); - } - - @Test - public void getFriendlyNameTestCase() throws Exception { - - assertEquals(obIdentifierAuthenticator.getFriendlyName(), "ob-identifier-first"); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void getSessionDataExceptionTestCase() throws Exception { - - mockRealmService = mock(RealmService.class); - mockUserCoreRealm = mock(org.wso2.carbon.user.core.UserRealm.class); - mockUserStoreManager = mock(UserStoreManager.class); - - AuthenticatorConfig authenticatorConfig = new AuthenticatorConfig(); - Map paramMap = new HashMap<>(); - paramMap.put("authRequestURL", "someURL"); - - authenticatorConfig.setParameterMap(paramMap); - - mockStatic(FileBasedConfigurationBuilder.class); - mockFileBasedConfigurationBuilder = mock(FileBasedConfigurationBuilder.class); - when(FileBasedConfigurationBuilder.getInstance()).thenReturn(mockFileBasedConfigurationBuilder); - when(mockFileBasedConfigurationBuilder.getAuthenticatorBean(anyString())).thenReturn(authenticatorConfig); - - mockStatic(IdentityExtensionsDataHolder.class); - RealmConfiguration mockRealmConfiguration = mock(RealmConfiguration.class); - when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolder); - when(identityExtensionsDataHolder.getRealmService()).thenReturn(mockRealmService); - when(mockRealmService.getBootstrapRealm()).thenReturn(mockUserCoreRealm); - when(mockUserCoreRealm.getUserStoreManager()).thenReturn(mockUserStoreManager); - when(mockUserStoreManager.getRealmConfiguration()).thenReturn(mockRealmConfiguration); - - when(mockRealmConfiguration.getAdminUserName()).thenReturn("adminUserName"); - when(mockRealmConfiguration.getAdminPassword()).thenReturn("adminPassword"); - - mockStatic(HTTPClientUtils.class); - CloseableHttpClient closeableHttpClient = mock(CloseableHttpClient.class); - CloseableHttpResponse closeableHttpResponse = mock(CloseableHttpResponse.class); - InputStream inputStream = mock(InputStream.class); - HttpEntity httpEntity = mock(HttpEntity.class); - BufferedReader bufferedReader = mock(BufferedReader.class); - StatusLine statusLine = mock(StatusLine.class); - final HttpGet[] httpGet = {mock(HttpGet.class)}; - when(HTTPClientUtils.getHttpsClient()).thenReturn(closeableHttpClient); - - doAnswer(new Answer() { - @Override - public Object answer(InvocationOnMock invocation) throws Throwable { - - httpGet[0] = (HttpGet) invocation.getArguments()[0]; - return closeableHttpResponse; - } - }).when(closeableHttpClient).execute(Mockito.anyObject()); - - when(closeableHttpResponse.getEntity()).thenReturn(httpEntity); - when(httpEntity.getContent()).thenReturn(inputStream); - when(bufferedReader.readLine()).thenReturn("sessionDataValues"); - when(closeableHttpResponse.getStatusLine()).thenReturn(statusLine); - when(statusLine.getStatusCode()).thenReturn(200); - - obIdentifierAuthenticator.getSessionData(dummySessionDataKey); - } - - @Test(expectedExceptions = OpenBankingException.class) - public void getSessionDataUserStoreExceptionTestCase() throws Exception { - - mockRealmService = mock(RealmService.class); - mockUserCoreRealm = mock(org.wso2.carbon.user.core.UserRealm.class); - mockUserStoreManager = mock(UserStoreManager.class); - - AuthenticatorConfig authenticatorConfig = new AuthenticatorConfig(); - Map paramMap = new HashMap<>(); - paramMap.put("authRequestURL", "someURL"); - - authenticatorConfig.setParameterMap(paramMap); - - mockStatic(FileBasedConfigurationBuilder.class); - mockFileBasedConfigurationBuilder = mock(FileBasedConfigurationBuilder.class); - when(FileBasedConfigurationBuilder.getInstance()).thenReturn(mockFileBasedConfigurationBuilder); - when(mockFileBasedConfigurationBuilder.getAuthenticatorBean(anyString())).thenReturn(authenticatorConfig); - - mockStatic(IdentityExtensionsDataHolder.class); - RealmConfiguration mockRealmConfiguration = mock(RealmConfiguration.class); - when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolder); - when(identityExtensionsDataHolder.getRealmService()).thenReturn(mockRealmService); - when(mockRealmService.getBootstrapRealm()).thenThrow(UserStoreException.class); - - obIdentifierAuthenticator.getSessionData(dummySessionDataKey); - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/authenticator/util/OBIdentifierAuthenticatorTestData.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/authenticator/util/OBIdentifierAuthenticatorTestData.java deleted file mode 100644 index a0bceec6..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/authenticator/util/OBIdentifierAuthenticatorTestData.java +++ /dev/null @@ -1,48 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.authenticator.util; - -import org.wso2.carbon.identity.application.authentication.framework.config.model.AuthenticatorConfig; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedIdPData; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedUser; -import org.wso2.carbon.identity.application.authenticator.basicauth.BasicAuthenticator; - -/** - * Test data for Open Banking identifier authentication. - */ -public class OBIdentifierAuthenticatorTestData { - - public static AuthenticatorConfig getAuthenticatorConfig() { - - AuthenticatorConfig authenticatorConfig = new AuthenticatorConfig(); - - authenticatorConfig.setApplicationAuthenticator(new BasicAuthenticator()); - return authenticatorConfig; - } - - public static AuthenticatedIdPData getSampleAuthenticatedIdPData() { - - AuthenticatedIdPData authenticatedIdPData = new AuthenticatedIdPData(); - - authenticatedIdPData.setUser(new AuthenticatedUser()); - authenticatedIdPData.setIdpName("Sample"); - authenticatedIdPData.addAuthenticator(OBIdentifierAuthenticatorTestData.getAuthenticatorConfig()); - return authenticatedIdPData; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/builders/DefaultOBRequestUriRequestObjectBuilderTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/builders/DefaultOBRequestUriRequestObjectBuilderTest.java deleted file mode 100644 index 4bec2e06..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/builders/DefaultOBRequestUriRequestObjectBuilderTest.java +++ /dev/null @@ -1,149 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.builders; - -import com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.util.test.jwt.builder.TestJwtBuilder; -import org.mockito.Mockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.oauth.cache.SessionDataCache; -import org.wso2.carbon.identity.oauth.cache.SessionDataCacheEntry; -import org.wso2.carbon.identity.oauth.config.OAuthServerConfiguration; -import org.wso2.carbon.identity.oauth.dao.OAuthAppDO; -import org.wso2.carbon.identity.oauth2.RequestObjectException; -import org.wso2.carbon.identity.oauth2.model.OAuth2Parameters; -import org.wso2.carbon.identity.oauth2.util.OAuth2Util; -import org.wso2.carbon.identity.openidconnect.model.RequestObject; - -import java.io.File; -import java.io.FileInputStream; -import java.io.InputStream; -import java.security.KeyStore; -import java.security.interfaces.RSAPrivateKey; - -import static org.powermock.api.mockito.PowerMockito.mock; -import static org.powermock.api.mockito.PowerMockito.mockStatic; -import static org.powermock.api.mockito.PowerMockito.when; - -/** - * Test for default Open Banking request, uri request builder. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({SessionDataCacheEntry.class, SessionDataCache.class, OAuth2Util.class, OAuthAppDO.class, - OAuthServerConfiguration.class}) -public class DefaultOBRequestUriRequestObjectBuilderTest extends PowerMockTestCase { - - private DefaultOBRequestUriRequestObjectBuilder defaultOBRequestUriRequestObjectBuilder; - - @Test() - public void buildRequestObjectSuccessScenario() throws Exception { - - String requestUri = "urn:ietf:params:oauth:request_uri:XKnDFSbXJWjuf0AY6gOT1EIuvdP8BQLo"; - String requestObjectString = TestJwtBuilder.getValidSignedJWT(); - OAuth2Parameters oAuth2Parameters = new OAuth2Parameters(); - oAuth2Parameters.setEssentialClaims(requestObjectString + ":" + "3600666666"); - - defaultOBRequestUriRequestObjectBuilder = new DefaultOBRequestUriRequestObjectBuilder(); - - SessionDataCache sessionDataCacheMock = mock(SessionDataCache.class); - SessionDataCacheEntry sessionDataCacheEntry = new SessionDataCacheEntry(); - mockStatic(SessionDataCacheEntry.class); - mockStatic(SessionDataCache.class); - when(SessionDataCache.getInstance()).thenReturn(sessionDataCacheMock); - when(sessionDataCacheMock.getValueFromCache(Mockito.anyObject())).thenReturn(sessionDataCacheEntry); - - sessionDataCacheEntry.setoAuth2Parameters(oAuth2Parameters); - - RequestObject result = defaultOBRequestUriRequestObjectBuilder - .buildRequestObject(requestUri, oAuth2Parameters); - - Assert.assertNotNull(result); - } - - @Test(expectedExceptions = RequestObjectException.class) - public void buildRequestObjectExpiredScenario() throws Exception { - - String requestUri = "urn:ietf:params:oauth:request_uri:XKnDFSbXJWjuf0AY6gOT1EIuvdP8BQLo"; - String requestObjectString = TestJwtBuilder.getValidSignedJWT(); - OAuth2Parameters oAuth2Parameters = new OAuth2Parameters(); - oAuth2Parameters.setEssentialClaims(requestObjectString + ":" + "60"); - - defaultOBRequestUriRequestObjectBuilder = new DefaultOBRequestUriRequestObjectBuilder(); - - SessionDataCache sessionDataCacheMock = mock(SessionDataCache.class); - SessionDataCacheEntry sessionDataCacheEntry = new SessionDataCacheEntry(); - mockStatic(SessionDataCacheEntry.class); - mockStatic(SessionDataCache.class); - when(SessionDataCache.getInstance()).thenReturn(sessionDataCacheMock); - when(sessionDataCacheMock.getValueFromCache(Mockito.anyObject())).thenReturn(sessionDataCacheEntry); - - sessionDataCacheEntry.setoAuth2Parameters(oAuth2Parameters); - - defaultOBRequestUriRequestObjectBuilder.buildRequestObject(requestUri, oAuth2Parameters); - } - - @Test(expectedExceptions = RequestObjectException.class) - public void testDecryptEncryptedReqObjFailure() throws Exception { - - defaultOBRequestUriRequestObjectBuilder = new DefaultOBRequestUriRequestObjectBuilder(); - String requestUri = "urn:ietf:params:oauth:request_uri:XKnDFSbXJWjuf0AY6gOT1EIuvdP8BQLo"; - String requestObjectString = TestJwtBuilder.getValidEncryptedJWT(); - OAuth2Parameters oAuth2Parameters = new OAuth2Parameters(); - oAuth2Parameters.setTenantDomain("dummyTenantDomain"); - oAuth2Parameters.setEssentialClaims(requestObjectString + ":" + "3600666666"); - - SessionDataCache sessionDataCacheMock = mock(SessionDataCache.class); - SessionDataCacheEntry sessionDataCacheEntry = new SessionDataCacheEntry(); - mockStatic(SessionDataCacheEntry.class); - mockStatic(SessionDataCache.class); - when(SessionDataCache.getInstance()).thenReturn(sessionDataCacheMock); - when(sessionDataCacheMock.getValueFromCache(Mockito.anyObject())).thenReturn(sessionDataCacheEntry); - - sessionDataCacheEntry.setoAuth2Parameters(oAuth2Parameters); - - OAuthServerConfiguration oAuthServerConfigurationMock = mock(OAuthServerConfiguration.class); - mockStatic(OAuthServerConfiguration.class); - when(OAuthServerConfiguration.getInstance()).thenReturn(oAuthServerConfigurationMock); - - mockStatic(OAuth2Util.class); - when(OAuth2Util.getTenantId(Mockito.anyString())).thenReturn(5); - - String path = "src/test/resources"; - File file = new File(path); - String absolutePathForTestResources = file.getAbsolutePath(); - - InputStream keystoreFile = new FileInputStream(absolutePathForTestResources + - "/wso2carbon.jks"); - KeyStore keystore = KeyStore.getInstance(KeyStore.getDefaultType()); - keystore.load(keystoreFile, "wso2carbon".toCharArray()); - - String alias = "wso2carbon"; - - // Get the private key. Password for the key store is 'wso2carbon'. - RSAPrivateKey privateKey = (RSAPrivateKey) keystore.getKey(alias, "wso2carbon".toCharArray()); - - when(OAuth2Util.getPrivateKey(Mockito.anyString(), Mockito.anyInt())).thenReturn(privateKey); - - defaultOBRequestUriRequestObjectBuilder - .buildRequestObject(requestUri, oAuth2Parameters); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/claims/OBDefaultOIDCClaimsCallbackHandlerTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/claims/OBDefaultOIDCClaimsCallbackHandlerTest.java deleted file mode 100644 index 8b0e3466..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/claims/OBDefaultOIDCClaimsCallbackHandlerTest.java +++ /dev/null @@ -1,146 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.claims; - -import com.nimbusds.jwt.JWTClaimsSet; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.token.util.TestConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.collections.map.SingletonMap; -import org.apache.commons.lang.StringUtils; -import org.mockito.Mockito; -import org.mockito.Spy; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedUser; -import org.wso2.carbon.identity.application.authentication.framework.util.FrameworkUtils; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AccessTokenReqDTO; -import org.wso2.carbon.identity.oauth2.model.HttpRequestHeader; -import org.wso2.carbon.identity.oauth2.token.OAuthTokenReqMessageContext; - -import java.io.File; -import java.net.URISyntaxException; -import java.net.URL; -import java.util.HashMap; -import java.util.Map; - -import static org.junit.Assert.assertEquals; -import static org.powermock.api.mockito.PowerMockito.mockStatic; -import static org.powermock.api.mockito.PowerMockito.when; -import static org.wso2.carbon.identity.core.util.IdentityCoreConstants.MULTI_ATTRIBUTE_SEPARATOR_DEFAULT; - -/** - * Class which tests OBDefaultOIDCClaimsCallbackHandlerTest. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({FrameworkUtils.class, IdentityCommonUtil.class, JWTClaimsSet.class}) -public class OBDefaultOIDCClaimsCallbackHandlerTest { - - @Spy - private OBDefaultOIDCClaimsCallbackHandler obDefaultOIDCClaimsCallbackHandler; - - @BeforeClass - public void beforeClass() { - - Map configMap = new HashMap<>(); - configMap.put(IdentityCommonConstants.CONSENT_ID_CLAIM_NAME, "consent_id"); - configMap.put(IdentityCommonConstants.REMOVE_TENANT_DOMAIN_FROM_SUBJECT, "true"); - configMap.put(IdentityCommonConstants.REMOVE_USER_STORE_DOMAIN_FROM_SUBJECT, "true"); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - mockStatic(FrameworkUtils.class); - mockStatic(IdentityCommonUtil.class); - when(FrameworkUtils.getMultiAttributeSeparator()).thenReturn(MULTI_ATTRIBUTE_SEPARATOR_DEFAULT); - obDefaultOIDCClaimsCallbackHandler = Mockito.spy(OBDefaultOIDCClaimsCallbackHandler.class); - } - - public static String getFilePath(String fileName) { - - if (StringUtils.isNotBlank(fileName)) { - URL url = OBDefaultOIDCClaimsCallbackHandlerTest.class.getClassLoader().getResource(fileName); - if (url != null) { - try { - File file = new File(url.toURI()); - return file.getAbsolutePath(); - } catch (URISyntaxException e) { - throw new IllegalArgumentException("Could not resolve a file with given path: " + - url.toExternalForm()); - } - } - } - throw new IllegalArgumentException("DB Script file name cannot be empty."); - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - - @Test(description = "Test the best case scenario in handling custom claims") - public void testHandleCustomClaims() throws OpenBankingException, IdentityOAuth2Exception { - - JWTClaimsSet.Builder jwtClaimsSetBuilder = new JWTClaimsSet.Builder(); - OAuth2AccessTokenReqDTO oauth2AccessTokenReqDTO = new OAuth2AccessTokenReqDTO(); - - HttpRequestHeader[] httpRequestHeaders = new HttpRequestHeader[1]; - httpRequestHeaders[0] = new HttpRequestHeader(IdentityCommonConstants.CERTIFICATE_HEADER, - TestConstants.CERTIFICATE_CONTENT); - oauth2AccessTokenReqDTO.setHttpRequestHeaders(httpRequestHeaders); - - oauth2AccessTokenReqDTO.setGrantType("client_credentials"); - oauth2AccessTokenReqDTO.setClientId("123"); - OAuthTokenReqMessageContext oAuthTokenReqMessageContext = - new OAuthTokenReqMessageContext(oauth2AccessTokenReqDTO); - - String[] scopes = new String[1]; - scopes[0] = "consent_id" + "123"; - oAuthTokenReqMessageContext.setScope(scopes); - AuthenticatedUser authenticatedUser = new AuthenticatedUser(); - authenticatedUser.setUserName("aaa@gold.com"); - authenticatedUser.setTenantDomain("carbon.super"); - authenticatedUser.setUserStoreDomain("PRIMARY"); - authenticatedUser.setFederatedIdPName("LOCAL"); - authenticatedUser.setFederatedUser(false); - authenticatedUser.setAuthenticatedSubjectIdentifier("aaa@gold.com@carbon.super"); - oAuthTokenReqMessageContext.setAuthorizedUser(authenticatedUser); - JWTClaimsSet jwtClaimsSetInitial = PowerMockito.mock(JWTClaimsSet.class); - PowerMockito.when(jwtClaimsSetInitial.getClaims()).thenReturn(new SingletonMap("scope", "test")); - Mockito.doReturn(jwtClaimsSetInitial).when(obDefaultOIDCClaimsCallbackHandler) - .getJwtClaimsFromSuperClass(jwtClaimsSetBuilder, oAuthTokenReqMessageContext); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("123")).thenReturn(true); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - JWTClaimsSet jwtClaimsSet = obDefaultOIDCClaimsCallbackHandler.handleCustomClaims(jwtClaimsSetBuilder, - oAuthTokenReqMessageContext); - - - assertEquals("123", jwtClaimsSet.getClaim("consent_id")); - assertEquals("{x5t#S256=807-E8KgUMV6dRHTQi1_QYo5eyPvjmjbxCtunbFixV0}", jwtClaimsSet.getClaim( - "cnf").toString()); - assertEquals("aaa@gold.com", jwtClaimsSet.getClaim("sub")); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/claims/RoleClaimProviderImplTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/claims/RoleClaimProviderImplTest.java deleted file mode 100644 index bb964578..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/claims/RoleClaimProviderImplTest.java +++ /dev/null @@ -1,152 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.claims; - -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedUser; -import org.wso2.carbon.identity.base.IdentityRuntimeException; -import org.wso2.carbon.identity.core.util.IdentityTenantUtil; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.token.OAuthTokenReqMessageContext; -import org.wso2.carbon.user.api.UserRealm; -import org.wso2.carbon.user.api.UserStoreException; -import org.wso2.carbon.user.api.UserStoreManager; -import org.wso2.carbon.user.core.service.RealmService; - -import java.util.Map; - -/** - * Test for role claim provider implementation. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({IdentityTenantUtil.class, IdentityExtensionsDataHolder.class}) -public class RoleClaimProviderImplTest extends PowerMockTestCase { - - private static final String USER_MARK = "mark@gold.com"; - private static final String USER_TOM = "tom@gold.com"; - private static final String USER_ANN = "ann@gold.com"; - private static final String CCO_ROLE = "Internal/CustomerCareOfficerRole"; - private static final String CC_OFFICER = "customerCareOfficer"; - private static final String[] SCOPES = new String[]{"openid", "consents:read_all"}; - - private RoleClaimProviderImpl uut; - - @BeforeClass - public void init() { - this.uut = new RoleClaimProviderImpl(); - } - - public void setup(String userName) throws UserStoreException { - PowerMockito.mockStatic(IdentityTenantUtil.class); - if (USER_ANN.equals(userName)) { - PowerMockito.when(IdentityTenantUtil.getTenantIdOfUser(USER_ANN)) - .thenThrow(new IdentityRuntimeException("")); - } else { - PowerMockito.when(IdentityTenantUtil.getTenantIdOfUser(Mockito.anyString())).thenReturn(1234); - } - - UserStoreManager userStoreManagerMock = Mockito.mock(UserStoreManager.class); - if (USER_TOM.equals(userName)) { - Mockito.when(userStoreManagerMock.getRoleListOfUser(USER_TOM)).thenThrow(new UserStoreException()); - } else { - Mockito.when(userStoreManagerMock.getRoleListOfUser(Mockito.anyString())) - .thenReturn(new String[]{CCO_ROLE}); - } - UserRealm userRealmMock = Mockito.mock(UserRealm.class); - Mockito.when(userRealmMock.getUserStoreManager()).thenReturn(userStoreManagerMock); - - RealmService realmServiceMock = Mockito.mock(RealmService.class); - Mockito.when(realmServiceMock.getTenantUserRealm(Mockito.anyInt())).thenReturn(userRealmMock); - - IdentityExtensionsDataHolder identityExtensionsDataHolderMock = Mockito - .mock(IdentityExtensionsDataHolder.class); - Mockito.when(identityExtensionsDataHolderMock.getRealmService()).thenReturn(realmServiceMock); - - PowerMockito.mockStatic(IdentityExtensionsDataHolder.class); - PowerMockito.when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolderMock); - } - - @Test(description = "when customer care officer is sending request, return customerCareOfficer role") - public void testGetAdditionalClaimsWithCustomerCareOfficerRole() - throws IdentityOAuth2Exception, UserStoreException { - setup(USER_MARK); - // mock - OAuthTokenReqMessageContext oAuthTokenReqMessageContextMock = Mockito.mock(OAuthTokenReqMessageContext.class); - AuthenticatedUser authorizedUserMock = Mockito.mock(AuthenticatedUser.class); - - // when - Mockito.when(authorizedUserMock.getUserName()).thenReturn(USER_MARK); - - Mockito.when(oAuthTokenReqMessageContextMock.getScope()).thenReturn(SCOPES); - Mockito.when(oAuthTokenReqMessageContextMock.getAuthorizedUser()).thenReturn(authorizedUserMock); - - Map claims = uut.getAdditionalClaims(oAuthTokenReqMessageContextMock, null); - - // assert - Assert.assertEquals(claims.get("user_role"), CC_OFFICER); - } - - @Test(description = "when IdentityRuntimeException occurs, do not return user_role value") - public void testGetAdditionalClaimsThrowIdentityRuntimeException() - throws IdentityOAuth2Exception, UserStoreException { - setup(USER_ANN); - // mock - OAuthTokenReqMessageContext oAuthTokenReqMessageContextMock = Mockito.mock(OAuthTokenReqMessageContext.class); - AuthenticatedUser authorizedUserMock = Mockito.mock(AuthenticatedUser.class); - - // when - Mockito.when(authorizedUserMock.getUserName()).thenReturn(USER_ANN); - - Mockito.when(oAuthTokenReqMessageContextMock.getScope()).thenReturn(SCOPES); - Mockito.when(oAuthTokenReqMessageContextMock.getAuthorizedUser()).thenReturn(authorizedUserMock); - - Map claims = uut.getAdditionalClaims(oAuthTokenReqMessageContextMock, null); - - // assert - Assert.assertFalse(claims.containsKey("user_role")); - } - - @Test(description = "when UserStoreException occurs, do not return user_role value") - public void testGetAdditionalClaimsThrowUserStoreException() - throws IdentityOAuth2Exception, UserStoreException { - setup(USER_TOM); - // mock - OAuthTokenReqMessageContext oAuthTokenReqMessageContextMock = Mockito.mock(OAuthTokenReqMessageContext.class); - AuthenticatedUser authorizedUserMock = Mockito.mock(AuthenticatedUser.class); - - // when - Mockito.when(authorizedUserMock.getUserName()).thenReturn(USER_TOM); - - Mockito.when(oAuthTokenReqMessageContextMock.getScope()).thenReturn(SCOPES); - Mockito.when(oAuthTokenReqMessageContextMock.getAuthorizedUser()).thenReturn(authorizedUserMock); - - Map claims = uut.getAdditionalClaims(oAuthTokenReqMessageContextMock, null); - - // assert - Assert.assertFalse(claims.containsKey("user_role")); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/clientauth/OBMutualTLSClientAuthenticatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/clientauth/OBMutualTLSClientAuthenticatorTest.java deleted file mode 100644 index 01c2e95f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/clientauth/OBMutualTLSClientAuthenticatorTest.java +++ /dev/null @@ -1,190 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.clientauth; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.token.util.TestConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.lang.StringUtils; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.springframework.mock.web.MockHttpServletRequest; -import org.springframework.mock.web.MockHttpServletResponse; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.core.util.IdentityUtil; -import org.wso2.carbon.identity.oauth2.bean.OAuthClientAuthnContext; -import org.wso2.carbon.identity.oauth2.client.authentication.OAuthClientAuthnException; -import org.wso2.carbon.identity.oauth2.token.handler.clientauth.mutualtls.utils.MutualTLSUtil; - -import java.net.URL; -import java.security.cert.CertificateException; -import java.security.cert.X509Certificate; -import java.util.Collections; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import static org.testng.Assert.assertEquals; -import static org.testng.Assert.assertFalse; -import static org.testng.Assert.assertTrue; - -/** - * Test for Open Banking mutual TLS client authenticator. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({IdentityCommonUtil.class, MutualTLSUtil.class}) -public class OBMutualTLSClientAuthenticatorTest extends PowerMockTestCase { - - MockHttpServletResponse response; - MockHttpServletRequest request; - OAuthClientAuthnContext clientAuthnContext = new OAuthClientAuthnContext(); - - @BeforeClass - public void beforeClass() { - clientAuthnContext.setClientId("test"); - } - - @BeforeMethod - public void beforeMethod() { - - request = new MockHttpServletRequest(); - response = new MockHttpServletResponse(); - - } - - @Test(description = "Test whether can authenticate is engaged for mtls request") - public void canAuthenticateTest() throws OpenBankingException { - PowerMockito.mockStatic(IdentityCommonUtil.class); - Map bodyParams = new HashMap<>(); - clientAuthnContext.setClientId(""); - bodyParams.put("client_id", Collections.singletonList("test")); - - OBMutualTLSClientAuthenticator authenticator = Mockito.spy(OBMutualTLSClientAuthenticator.class); - - request.setParameter(IdentityCommonConstants.CLIENT_ID, "test"); - request.addHeader(TestConstants.CERTIFICATE_HEADER, TestConstants.CERTIFICATE_CONTENT); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - - boolean response = authenticator.canAuthenticate(request, bodyParams, clientAuthnContext); - assertTrue(response); - } - - @Test(description = "Test whether can authenticate is not engaged when request does not have a client ID") - public void canAuthenticateNoClientIDTest() throws OpenBankingException { - PowerMockito.mockStatic(IdentityCommonUtil.class); - OBMutualTLSClientAuthenticator authenticator = Mockito.spy(OBMutualTLSClientAuthenticator.class); - - request.addHeader(TestConstants.CERTIFICATE_HEADER, TestConstants.CERTIFICATE_CONTENT); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - boolean response = authenticator.canAuthenticate(request, null, clientAuthnContext); - assertFalse(response); - } - - @Test(description = "Test whether can authenticate is not engaged when request has invalid certificate") - public void canAuthenticateInvalidCertTest() throws OpenBankingException { - PowerMockito.mockStatic(IdentityCommonUtil.class); - OBMutualTLSClientAuthenticator authenticator = Mockito.spy(OBMutualTLSClientAuthenticator.class); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - request.setParameter(IdentityCommonConstants.CLIENT_ID, "test"); - request.addHeader(TestConstants.CERTIFICATE_HEADER, "test"); - try { - authenticator.canAuthenticate(request, null, clientAuthnContext); - } catch (Exception e) { - assertEquals(e.getMessage(), "Transport certificate passed through the request not valid"); - } - } - - @Test(description = "Test whether can authenticate is not engaged when request does not have a cert header") - public void canAuthenticateNoCertHeaderTest() throws OpenBankingException { - PowerMockito.mockStatic(IdentityCommonUtil.class); - OBMutualTLSClientAuthenticator authenticator = Mockito.spy(OBMutualTLSClientAuthenticator.class); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - request.setParameter(IdentityCommonConstants.CLIENT_ID, "test"); - boolean response = authenticator.canAuthenticate(request, null, clientAuthnContext); - assertFalse(response); - } - - @Test(description = "Test whether obtaining JWKS endpoint of the SP is succesful") - public void getJWKSEndpointOfSPTest() throws OAuthClientAuthnException { - PowerMockito.mockStatic(MutualTLSUtil.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - PowerMockito.when(IdentityCommonUtil.getJWKURITransportCert()).thenReturn("dummy"); - OBMutualTLSClientAuthenticator authenticator = Mockito.spy(OBMutualTLSClientAuthenticator.class); - String expectedUrl = "https://dummy.com"; - PowerMockito.when(MutualTLSUtil.getPropertyValue(Mockito.anyObject(), Mockito.anyObject())) - .thenReturn(expectedUrl); - URL url = authenticator.getJWKSEndpointOfSP(Mockito.anyObject(), Mockito.anyObject()); - assertEquals(url.getHost(), "dummy.com"); - } - - @Test(description = "Test whether obtaining JWKS endpoint of the SP is failing when empty JWKS URI is given") - public void getJWKSEndpointOfSPEmptyTest() { - PowerMockito.mockStatic(MutualTLSUtil.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - PowerMockito.when(IdentityCommonUtil.getJWKURITransportCert()).thenReturn(""); - OBMutualTLSClientAuthenticator authenticator = Mockito.spy(OBMutualTLSClientAuthenticator.class); - String expectedUrl = "https://dummy.com"; - PowerMockito.when(MutualTLSUtil.getPropertyValue(Mockito.anyObject(), Mockito.anyObject())) - .thenReturn(expectedUrl); - try { - authenticator.getJWKSEndpointOfSP(Mockito.anyObject(), Mockito.anyObject()); - } catch (OAuthClientAuthnException e) { - assertEquals(e.getErrorCode(), "server_error"); - } - } - - @Test(description = "Test whether obtaining JWKS endpoint of the SP is failing when malformed URI is given") - public void getJWKSEndpointOfSPMalformedURITest() { - PowerMockito.mockStatic(MutualTLSUtil.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - PowerMockito.when(IdentityCommonUtil.getJWKURITransportCert()).thenReturn(""); - OBMutualTLSClientAuthenticator authenticator = Mockito.spy(OBMutualTLSClientAuthenticator.class); - String expectedUrl = "dummy"; - PowerMockito.when(MutualTLSUtil.getPropertyValue(Mockito.anyObject(), Mockito.anyObject())) - .thenReturn(expectedUrl); - try { - authenticator.getJWKSEndpointOfSP(Mockito.anyObject(), Mockito.anyObject()); - } catch (OAuthClientAuthnException e) { - assertEquals(e.getErrorCode(), "server_error"); - } - } - - private X509Certificate getCertificate(String certificateContent) { - - if (StringUtils.isNotBlank(certificateContent)) { - // Build the Certificate object from cert content. - try { - return (X509Certificate) IdentityUtil.convertPEMEncodedContentToCertificate(certificateContent); - } catch (CertificateException e) { - //do nothing - } - } - return null; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/DCRExtendedValidatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/DCRExtendedValidatorTest.java deleted file mode 100644 index 90efe69c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/DCRExtendedValidatorTest.java +++ /dev/null @@ -1,103 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr; - -import com.google.gson.Gson; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.identity.dcr.exception.DCRValidationException; -import com.wso2.openbanking.accelerator.identity.dcr.model.RegistrationRequest; -import com.wso2.openbanking.accelerator.identity.dcr.model.SoftwareStatementBody; -import com.wso2.openbanking.accelerator.identity.dcr.util.ExtendedSoftwareStatementBody; -import com.wso2.openbanking.accelerator.identity.dcr.util.ExtendedValidatorImpl; -import com.wso2.openbanking.accelerator.identity.dcr.util.RegistrationTestConstants; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -import java.text.ParseException; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -/** - * Test for DCR extended validator. - */ -public class DCRExtendedValidatorTest { - - private RegistrationRequest registrationRequest; - private ExtendedValidatorImpl extendedValidator = new ExtendedValidatorImpl(); - - private static final Log log = LogFactory.getLog(DCRValidationTest.class); - - @BeforeClass - public void beforeClass() { - - Map confMap = new HashMap<>(); - Map> dcrRegistrationConfMap = new HashMap<>(); - Gson gson = new Gson(); - registrationRequest = gson.fromJson(RegistrationTestConstants.extendedRegistrationRequestJson, - RegistrationRequest.class); - String decodedSSA = null; - try { - decodedSSA = JWTUtils - .decodeRequestJWT(registrationRequest.getSoftwareStatement(), "body").toJSONString(); - } catch (ParseException e) { - log.error("Error while parsing the SSA", e); - } - extendedValidator.setSoftwareStatementPayload(registrationRequest, decodedSSA); - - List validAlgorithms = new ArrayList<>(); - validAlgorithms.add("PS256"); - validAlgorithms.add("ES256"); - confMap.put(OpenBankingConstants.SIGNATURE_ALGORITHMS, validAlgorithms); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(confMap); - IdentityExtensionsDataHolder.getInstance().setDcrRegistrationConfigMap(dcrRegistrationConfMap); - } - - @Test - public void testExtendedValidatorFailure() { - - try { - extendedValidator.validatePost(registrationRequest); - } catch (DCRValidationException e) { - Assert.assertTrue(e.getErrorDescription().contains("Redirect URIs can not be null")); - - } - } - - @Test - public void testExtendedSSAAttributes() { - - SoftwareStatementBody softwareStatementBody = registrationRequest.getSoftwareStatementBody(); - Assert.assertEquals(((ExtendedSoftwareStatementBody) softwareStatementBody).getLogURI(), - "https://wso2.com/wso2.jpg"); - } - - @Test - public void testExtendedRegistrationResponse() { - String additionalAttributes = "\"additional_attribute_1\":\"111111\",\"additional_attribute_2\":\"222222\""; - String registrationResponse = extendedValidator.getRegistrationResponse(new HashMap<>()); - Assert.assertTrue(registrationResponse.contains(additionalAttributes)); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/DCRValidationTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/DCRValidationTest.java deleted file mode 100644 index af5dccf8..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/DCRValidationTest.java +++ /dev/null @@ -1,301 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr; - -import com.google.gson.Gson; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.identity.dcr.exception.DCRValidationException; -import com.wso2.openbanking.accelerator.identity.dcr.model.RegistrationRequest; -import com.wso2.openbanking.accelerator.identity.dcr.util.RegistrationTestConstants; -import com.wso2.openbanking.accelerator.identity.dcr.utils.ValidatorUtils; -import com.wso2.openbanking.accelerator.identity.dcr.validation.DCRCommonConstants; -import com.wso2.openbanking.accelerator.identity.dcr.validation.DefaultRegistrationValidatorImpl; -import com.wso2.openbanking.accelerator.identity.dcr.validation.RegistrationValidator; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -import java.text.ParseException; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -/** - * Test for DCR validation. - */ -public class DCRValidationTest { - - private static final Log log = LogFactory.getLog(DCRValidationTest.class); - - private RegistrationValidator registrationValidator; - private RegistrationRequest registrationRequest; - private static final String NULL = "null"; - - @BeforeClass - public void beforeClass() { - - Map confMap = new HashMap<>(); - Map> dcrRegistrationConfMap = new HashMap<>(); - Map> dcrRegistrationConfMap2 = new HashMap<>(); - List registrationParams = Arrays.asList("Issuer:true:null", - "TokenEndPointAuthentication:true:private_key_jwt", "ResponseTypes:true:code id_token", - "GrantTypes:true:authorization_code,refresh_token", "ApplicationType:false:web", - "IdTokenSignedResponseAlg:true:null", "SoftwareStatement:true:null", "Scope:false:accounts,payments"); - confMap.put(DCRCommonConstants.DCR_VALIDATOR, "com.wso2.openbanking.accelerator.identity.dcr" + - ".validation.DefaultRegistrationValidatorImpl"); - List validAlgorithms = new ArrayList<>(); - validAlgorithms.add("PS256"); - validAlgorithms.add("ES256"); - confMap.put(OpenBankingConstants.SIGNATURE_ALGORITHMS, validAlgorithms); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(confMap); - String dcrValidator = confMap.get(DCRCommonConstants.DCR_VALIDATOR).toString(); - registrationValidator = getDCRValidator(dcrValidator); - registrationRequest = getRegistrationRequestObject(RegistrationTestConstants.registrationRequestJson); - //set registration parameter values for testing - for (String param : registrationParams) { - setParamConfig(param, dcrRegistrationConfMap); - } - IdentityExtensionsDataHolder.getInstance().setDcrRegistrationConfigMap(dcrRegistrationConfMap); - } - - @Test - public void testInvalidAlgorithm() { - - registrationRequest.setIdTokenSignedResponseAlg("RS256"); - - String decodedSSA = null; - try { - decodedSSA = JWTUtils - .decodeRequestJWT(registrationRequest.getSoftwareStatement(), "body").toJSONString(); - } catch (ParseException e) { - log.error("Error while parsing the SSA", e); - } - registrationValidator.setSoftwareStatementPayload(registrationRequest, decodedSSA); - try { - ValidatorUtils.getValidationViolations(registrationRequest); - } catch (DCRValidationException e) { - Assert.assertTrue(e.getErrorDescription().contains("Invalid signing algorithm sent")); - } - } - - @Test(dependsOnMethods = "testInvalidAlgorithm") - public void testInvalidIssuer() { - - registrationRequest.setIdTokenSignedResponseAlg("PS256"); - registrationRequest.setIssuer("222"); - - try { - ValidatorUtils.getValidationViolations(registrationRequest); - } catch (DCRValidationException e) { - Assert.assertTrue(e.getErrorDescription().contains("Invalid issuer")); - } - - } - - @Test(dependsOnMethods = "testInvalidIssuer") - public void testIssuerExists() { - - registrationRequest.setIssuer(null); - - try { - ValidatorUtils.getValidationViolations(registrationRequest); - } catch (DCRValidationException e) { - Assert.assertTrue(e.getErrorDescription().contains("Required parameter issuer cannot be null")); - } - - } - - @Test(dependsOnMethods = "testIssuerExists") - public void testTokenEndPointAuthMethodExists() { - - registrationRequest.setIssuer("9b5usDpbNtmxDcTzs7GzKp"); - registrationRequest.setTokenEndPointAuthentication(""); - - try { - ValidatorUtils.getValidationViolations(registrationRequest); - } catch (DCRValidationException e) { - Assert.assertTrue(e.getErrorDescription() - .contains("Required parameter tokenEndPointAuthentication cannot be empty")); - } - } - - @Test(dependsOnMethods = "testTokenEndPointAuthMethodExists") - public void testResponseTypesExists() { - - registrationRequest.setTokenEndPointAuthentication("private_key_jwt"); - registrationRequest.setResponseTypes(new ArrayList<>()); - - try { - ValidatorUtils.getValidationViolations(registrationRequest); - } catch (DCRValidationException e) { - Assert.assertTrue(e.getErrorDescription() - .contains("Required parameter responseTypes cannot be empty")); - } - } - - @Test(dependsOnMethods = "testResponseTypesExists") - public void testGrantTypesExists() { - - List responseTypeList = new ArrayList(); - responseTypeList.add("code id_token"); - registrationRequest.setResponseTypes(responseTypeList); - - registrationRequest.setGrantTypes(null); - - try { - ValidatorUtils.getValidationViolations(registrationRequest); - } catch (DCRValidationException e) { - Assert.assertTrue(e.getErrorDescription().contains("Required parameter grantTypes cannot be null")); - } - } - - @Test(dependsOnMethods = "testGrantTypesExists") - public void testIdTokenSignedResponseAlgExists() { - - List grantTypeList = new ArrayList(); - grantTypeList.add("authorization_code"); - grantTypeList.add("refresh_token"); - registrationRequest.setGrantTypes(grantTypeList); - - registrationRequest.setIdTokenSignedResponseAlg(null); - - try { - ValidatorUtils.getValidationViolations(registrationRequest); - } catch (OpenBankingException e) { - Assert.assertTrue(e.getMessage().contains("Required parameter idTokenSignedResponseAlg cannot be null")); - } - } - - @Test(dependsOnMethods = "testIdTokenSignedResponseAlgExists") - public void testValidationViolations() { - - try { - ValidatorUtils.getValidationViolations(registrationRequest); - } catch (OpenBankingException e) { - Assert.assertTrue(e.getMessage().contains("Required parameter idTokenSignedResponseAlg cannot be null")); - } - } - - @Test(dependsOnMethods = "testValidationViolations") - public void testDefaultValidator() { - - try { - registrationValidator.validatePost(registrationRequest); - - registrationValidator.validateUpdate(registrationRequest); - - registrationValidator.validateGet("1234"); - - registrationValidator.validateDelete("1234"); - } catch (OpenBankingException e) { - Assert.assertTrue(e.getMessage().contains("Required parameter idTokenSignedResponseAlg cannot be null")); - } - } - - @Test(dependsOnMethods = "testDefaultValidator") - public void testSoftwareStatementExists() { - - registrationRequest.setIdTokenSignedResponseAlg("PS256"); - registrationRequest.setSoftwareStatement(null); - - try { - ValidatorUtils.getValidationViolations(registrationRequest); - } catch (OpenBankingException e) { - Assert.assertTrue(e.getMessage().contains("Required parameter softwareStatement cannot be null")); - } - } - - @Test(dependsOnMethods = "testSoftwareStatementExists") - public void testSSAParsingException() { - - registrationRequest.setSoftwareStatement("effff"); - try { - ValidatorUtils.getValidationViolations(registrationRequest); - } catch (OpenBankingException e) { - Assert.assertTrue(e.getMessage().contains("Invalid issuer")); - } - } - - @Test (dependsOnMethods = "testSSAParsingException") - public void testResponseTypesAllowedValues() { - - List responseTypeList = new ArrayList(); - responseTypeList.add(""); - registrationRequest.setResponseTypes(responseTypeList); - try { - ValidatorUtils.getValidationViolations(registrationRequest); - } catch (DCRValidationException e) { - Assert.assertTrue(e.getErrorDescription().contains("Invalid responseTypes provided")); - } - } - - @Test (dependsOnMethods = "testResponseTypesAllowedValues") - public void testApplicationTypeAllowedValues() { - - List responseTypeList = new ArrayList(); - responseTypeList.add("code id_token"); - registrationRequest.setResponseTypes(responseTypeList); - registrationRequest.setApplicationType(""); - try { - ValidatorUtils.getValidationViolations(registrationRequest); - } catch (DCRValidationException e) { - Assert.assertTrue(e.getErrorDescription().contains("Invalid applicationType provided")); - } - } - - private static RegistrationRequest getRegistrationRequestObject(String request) { - - Gson gson = new Gson(); - return gson.fromJson(request, RegistrationRequest.class); - } - - public static RegistrationValidator getDCRValidator(String dcrValidator) { - - if (StringUtils.isEmpty(dcrValidator)) { - return new DefaultRegistrationValidatorImpl(); - } - try { - return (RegistrationValidator) Class.forName(dcrValidator).newInstance(); - } catch (InstantiationException | IllegalAccessException e) { - log.error("Error instantiating " + dcrValidator, e); - return new DefaultRegistrationValidatorImpl(); - } catch (ClassNotFoundException e) { - log.error("Cannot find class: " + dcrValidator, e); - return new DefaultRegistrationValidatorImpl(); - } - } - - private void setParamConfig(String configParam, Map> dcrRegistrationConfMap) { - Map parameterValues = new HashMap<>(); - parameterValues.put(DCRCommonConstants.DCR_REGISTRATION_PARAM_REQUIRED, configParam.split(":")[1]); - if (!NULL.equalsIgnoreCase(configParam.split(":")[2])) { - List allowedValues = new ArrayList<>(); - allowedValues.addAll(Arrays.asList(configParam.split(":")[2].split(","))); - parameterValues.put(DCRCommonConstants.DCR_REGISTRATION_PARAM_ALLOWED_VALUES, allowedValues); - } - dcrRegistrationConfMap.put(configParam.split(":")[0], parameterValues); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/DCRValidationUtilTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/DCRValidationUtilTest.java deleted file mode 100644 index f71510c0..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/DCRValidationUtilTest.java +++ /dev/null @@ -1,69 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr; - -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.identity.dcr.util.RegistrationTestConstants; -import com.wso2.openbanking.accelerator.identity.dcr.utils.ValidatorUtils; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import net.minidev.json.JSONObject; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.testng.Assert; -import org.testng.annotations.Test; - -import java.text.ParseException; -import java.util.HashMap; - -/** - * Test for DCR validation util. - */ -public class DCRValidationUtilTest { - - private static final Log log = LogFactory.getLog(DCRValidationTest.class); - - @Test - public void testJWTDecodeHead() { - - try { - JSONObject decodedObject = JWTUtils.decodeRequestJWT(RegistrationTestConstants.SSA, "head"); - Assert.assertEquals(decodedObject.getAsString("alg"), "PS256"); - } catch (ParseException e) { - log.error("Error while parsing the jwt", e); - } - - } - - @Test - public void testJWTDecodeBody() { - - try { - JSONObject decodedObject = JWTUtils.decodeRequestJWT(RegistrationTestConstants.SSA, "body"); - Assert.assertEquals(decodedObject.getAsString("software_environment"), "sandbox"); - } catch (ParseException e) { - log.error("Error while parsing the jwt", e); - } - } - - @Test - public void testGetRegistrationClientURI() { - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(new HashMap<>()); - String registrationClientURI = ValidatorUtils.getRegistrationClientURI(); - Assert.assertEquals(registrationClientURI, "https://localhost:8243/open-banking/0.1/register/"); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/ExtendedRegistrationRequest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/ExtendedRegistrationRequest.java deleted file mode 100644 index db22db7d..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/ExtendedRegistrationRequest.java +++ /dev/null @@ -1,85 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.util; - -import com.wso2.openbanking.accelerator.identity.common.annotations.validationgroups.MandatoryChecks; -import com.wso2.openbanking.accelerator.identity.dcr.model.RegistrationRequest; -import com.wso2.openbanking.accelerator.identity.dcr.model.SoftwareStatementBody; -import com.wso2.openbanking.accelerator.identity.dcr.validation.DCRCommonConstants; - -import java.util.List; - -import javax.validation.constraints.NotNull; - -/** - * Extended registration request. - */ -public class ExtendedRegistrationRequest extends RegistrationRequest { - - private RegistrationRequest registrationRequest; - - ExtendedRegistrationRequest(RegistrationRequest registrationRequest) { - - this.registrationRequest = registrationRequest; - } - - @Override - @NotNull(message = "Redirect URIs can not be null:" + DCRCommonConstants.INVALID_META_DATA, - groups = MandatoryChecks.class) - public List getCallbackUris() { - - return registrationRequest.getCallbackUris(); - } - - @Override - public String getIssuer() { - - return registrationRequest.getIssuer(); - } - - @Override - public String getTokenEndPointAuthentication() { - - return registrationRequest.getTokenEndPointAuthentication(); - } - - @Override - public List getGrantTypes() { - - return registrationRequest.getGrantTypes(); - } - - @Override - public String getSoftwareStatement() { - - return registrationRequest.getSoftwareStatement(); - } - - @Override - public String getIdTokenSignedResponseAlg() { - - return registrationRequest.getIdTokenSignedResponseAlg(); - } - - @Override - public SoftwareStatementBody getSoftwareStatementBody() { - - return registrationRequest.getSoftwareStatementBody(); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/ExtendedRegistrationResponse.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/ExtendedRegistrationResponse.java deleted file mode 100644 index 58f65184..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/ExtendedRegistrationResponse.java +++ /dev/null @@ -1,49 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.util; - -import com.google.gson.annotations.SerializedName; -import com.wso2.openbanking.accelerator.identity.dcr.model.RegistrationResponse; -/** - * Extended class for RegistrationResponse. - */ -public class ExtendedRegistrationResponse extends RegistrationResponse { - - @SerializedName("additional_attribute_1") - protected String additionalAttribute1 = null; - - @SerializedName("additional_attribute_2") - protected String additionalAttribute2 = null; - - public String getAdditionalAttribute1() { - return additionalAttribute1; - } - - public void setAdditionalAttribute1(String additionalAttribute1) { - this.additionalAttribute1 = additionalAttribute1; - } - - public String getAdditionalAttribute2() { - return additionalAttribute2; - } - - public void setAdditionalAttribute2(String additionalAttribute2) { - this.additionalAttribute2 = additionalAttribute2; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/ExtendedSoftwareStatementBody.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/ExtendedSoftwareStatementBody.java deleted file mode 100644 index 7055dd16..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/ExtendedSoftwareStatementBody.java +++ /dev/null @@ -1,42 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.util; - -import com.google.gson.annotations.SerializedName; -import com.wso2.openbanking.accelerator.identity.dcr.model.SoftwareStatementBody; - -/** - * Extended software statement body. - */ -public class ExtendedSoftwareStatementBody extends SoftwareStatementBody { - - - public String getLogURI() { - - return logURI; - } - - public void setLogURI(String logURI) { - - this.logURI = logURI; - } - - @SerializedName("software_logo_uri") - private String logURI; - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/ExtendedValidatorImpl.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/ExtendedValidatorImpl.java deleted file mode 100644 index a2515876..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/ExtendedValidatorImpl.java +++ /dev/null @@ -1,79 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.util; - -import com.google.gson.Gson; -import com.google.gson.GsonBuilder; -import com.google.gson.JsonElement; -import com.wso2.openbanking.accelerator.identity.dcr.exception.DCRValidationException; -import com.wso2.openbanking.accelerator.identity.dcr.model.RegistrationRequest; -import com.wso2.openbanking.accelerator.identity.dcr.model.SoftwareStatementBody; -import com.wso2.openbanking.accelerator.identity.dcr.utils.ValidatorUtils; -import com.wso2.openbanking.accelerator.identity.dcr.validation.DefaultRegistrationValidatorImpl; - -import java.util.Map; - -/** - * Extended validator implementation. - */ -public class ExtendedValidatorImpl extends DefaultRegistrationValidatorImpl { - - @Override - public void validatePost(RegistrationRequest registrationRequest) throws DCRValidationException { - - ExtendedRegistrationRequest request = new ExtendedRegistrationRequest(registrationRequest); - ValidatorUtils.getValidationViolations(request); - } - - @Override - public void validateGet(String clientId) throws DCRValidationException { - - } - - @Override - public void validateDelete(String clientId) throws DCRValidationException { - - } - - @Override - public void validateUpdate(RegistrationRequest registrationRequest) throws DCRValidationException { - - } - - @Override - public String getRegistrationResponse(Map clientMetaData) { - - clientMetaData.put("additional_attribute_1", "111111"); - clientMetaData.put("additional_attribute_2", "222222"); - - Gson gson = new Gson(); - JsonElement jsonElement = gson.toJsonTree(clientMetaData); - ExtendedRegistrationResponse registrationResponse = - gson.fromJson(jsonElement, ExtendedRegistrationResponse.class); - return gson.toJson(registrationResponse); - } - - @Override - public void setSoftwareStatementPayload(RegistrationRequest registrationRequest, String decodedSSA) { - - SoftwareStatementBody softwareStatementPayload = new GsonBuilder().create() - .fromJson(decodedSSA, ExtendedSoftwareStatementBody.class); - registrationRequest.setSoftwareStatementBody(softwareStatementPayload); - - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/RegistrationTestConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/RegistrationTestConstants.java deleted file mode 100644 index 51dc5e76..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/util/RegistrationTestConstants.java +++ /dev/null @@ -1,99 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.util; - -/** - * Registration test constants. - */ -public class RegistrationTestConstants { - - public static final String SSA = "eyJhbGciOiJQUzI1NiIsImtpZCI6IkR3TUtkV01tajdQV2ludm9xZlF5WFZ6eVo2USIs" + - "InR5cCI6IkpXVCJ9.eyJpc3MiOiJPcGVuQmFua2luZyBMdGQiLCJpYXQiOjE1OTIzNjQ1NjgsImp0aSI6IjNkMWIzNTk1ZWZh" + - "YzRlMzYiLCJzb2Z0d2FyZV9lbnZpcm9ubWVudCI6InNhbmRib3giLCJzb2Z0d2FyZV9tb2RlIjoiVGVzdCIsInNvZnR3YXJlX2" + - "lkIjoiOWI1dXNEcGJOdG14RGNUenM3R3pLcCIsInNvZnR3YXJlX2NsaWVudF9pZCI6IjliNXVzRHBiTnRteERjVHpzN0d6S3AiLC" + - "Jzb2Z0d2FyZV9jbGllbnRfbmFtZSI6IlRlc3QgQVBQIE5ldyIsInNvZnR3YXJlX2NsaWVudF9kZXNjcmlwdGlvbiI6IlRoaXMgVFBQ" + - "IElzIGNyZWF0ZWQgZm9yIHRlc3RpbmcgcHVycG9zZXMuICIsInNvZnR3YXJlX3ZlcnNpb24iOjEuNSwic29mdHdhcmVfY2xpZW50X3V" + - "yaSI6Imh0dHBzOi8vd3NvMi5jb20iLCJzb2Z0d2FyZV9yZWRpcmVjdF91cmlzIjpbImh0dHBzOi8vd3NvMi5jb20iXSwic29mdHdhcmV" + - "fcm9sZXMiOlsiQUlTUCIsIlBJU1AiXSwib3JnYW5pc2F0aW9uX2NvbXBldGVudF9hdXRob3JpdHlfY2xhaW1zIjp7ImF1dGhvcml0eV" + - "9pZCI6Ik9CR0JSIiwicmVnaXN0cmF0aW9uX2lkIjoiVW5rbm93bjAwMTU4MDAwMDFIUVFyWkFBWCIsInN0YXR1cyI6IkFjdGl2ZSIsI" + - "mF1dGhvcmlzYXRpb25zIjpbeyJtZW1iZXJfc3RhdGUiOiJHQiIsInJvbGVzIjpbIkFJU1AiLCJQSVNQIl19LHsibWVtYmVyX3N0YXRl" + - "IjoiSUUiLCJyb2xlcyI6WyJBSVNQIiwiUElTUCJdfSx7Im1lbWJlcl9zdGF0ZSI6Ik5MIiwicm9sZXMiOlsiQUlTUCIsIlBJU1AiXX1" + - "dfSwic29mdHdhcmVfbG9nb191cmkiOiJodHRwczovL3dzbzIuY29tL3dzbzIuanBnIiwib3JnX3N0YXR1cyI6IkFjdGl2ZSIsIm9yZ1" + - "9pZCI6IjAwMTU4MDAwMDFIUVFyWkFBWCIsIm9yZ19uYW1lIjoiV1NPMiAoVUspIExJTUlURUQiLCJvcmdfY29udGFjdHMiOlt7Im5hb" + - "WUiOiJUZWNobmljYWwiLCJlbWFpbCI6InNhY2hpbmlzQHdzbzIuY29tIiwicGhvbmUiOiIrOTQ3NzQyNzQzNzQiLCJ0eXBlIjoiVGVj" + - "aG5pY2FsIn0seyJuYW1lIjoiQnVzaW5lc3MiLCJlbWFpbCI6InNhY2hpbmlzQHdzbzIuY29tIiwicGhvbmUiOiIrOTQ3NzQyNzQzNzQ" + - "iLCJ0eXBlIjoiQnVzaW5lc3MifV0sIm9yZ19qd2tzX2VuZHBvaW50IjoiaHR0cHM6Ly9rZXlzdG9yZS5vcGVuYmFua2luZ3Rlc3Qub3" + - "JnLnVrLzAwMTU4MDAwMDFIUVFyWkFBWC8wMDE1ODAwMDAxSFFRclpBQVguandrcyIsIm9yZ19qd2tzX3Jldm9rZWRfZW5kcG9pbnQiO" + - "iJodHRwczovL2tleXN0b3JlLm9wZW5iYW5raW5ndGVzdC5vcmcudWsvMDAxNTgwMDAwMUhRUXJaQUFYL3Jldm9rZWQvMDAxNTgwMDAw" + - "MUhRUXJaQUFYLmp3a3MiLCJzb2Z0d2FyZV9qd2tzX2VuZHBvaW50IjoiaHR0cHM6Ly9rZXlzdG9yZS5vcGVuYmFua2luZ3Rlc3Qub3J" + - "nLnVrLzAwMTU4MDAwMDFIUVFyWkFBWC85YjV1c0RwYk50bXhEY1R6czdHektwLmp3a3MiLCJzb2Z0d2FyZV9qd2tzX3Jldm9rZWRfZW5" + - "kcG9pbnQiOiJodHRwczovL2tleXN0b3JlLm9wZW5iYW5raW5ndGVzdC5vcmcudWsvMDAxNTgwMDAwMUhRUXJaQUFYL3Jldm9rZWQvOW" + - "I1dXNEcGJOdG14RGNUenM3R3pLcC5qd2tzIiwic29mdHdhcmVfcG9saWN5X3VyaSI6Imh0dHBzOi8vd3NvMi5jb20iLCJzb2Z0d2FyZ" + - "V90b3NfdXJpIjoiaHR0cHM6Ly93c28yLmNvbSIsInNvZnR3YXJlX29uX2JlaGFsZl9vZl9vcmciOiJXU08yIE9wZW4gQmFua2luZyJ9" + - ".mkbNeMGPNPEGqZbm06__7rWG9RWeEZ8MKgdLZGPkF0HMXX6MoPrw3e5ymZ_kxtVe5cRM2IVFThN1VBSuafThMH0PYwwRY2_3NApUWa" + - "f6BExL34Sbq_plmz8Ciq2zXYiYWPq2ReS1aPSJ-67nRF8Dnap5QLhqmowIDcGz1byTe2mukFc6CmBmwTBeDC_px56i4_n5xHXtrVBIf" + - "jFYcv2VewJ7K050JMmdIvdODafGei61JQIDrRUT_w0yU4-8WG9IDBI7G4H_GCPWmckJFApZyCnIWeBaEmfe6l2_GQs9VkQq1U1xJXtd" + - "WAfrzEjbMMnZSvqdoQAISq0y6mQofA0n5g"; - - public static String registrationRequestJson = "{\n" + - " \"iss\": \"9b5usDpbNtmxDcTzs7GzKp\",\n" + - " \"iat\": 1593752054,\n" + - " \"exp\": 1743573565,\n" + - " \"jti\": \"92713892-5514-11e9-8647-d663bd873d93\",\n" + - " \"aud\": \"https://localbank.com\",\n" + - " \"scope\": \"accounts payments\",\n" + - " \"token_endpoint_auth_method\": \"private_key_jwt\",\n" + - " \"grant_types\": [\n" + - " \"authorization_code\",\n" + - " \"refresh_token\"\n" + - " ],\n" + - " \"response_types\": [\n" + - " \"code id_token\"\n" + - " ],\n" + - " \"id_token_signed_response_alg\": \"PS256\",\n" + - " \"request_object_signing_alg\": \"PS256\",\n" + - " \"software_id\": \"9b5usDpbNtmxDcTzs7GzKp\",\n" + - " \"application_type\": \"web\",\n" + - " \"redirect_uris\": [\n" + - " \"https://wso2.com\"\n" + - " ],\n" + - " \"software_statement\" : " + RegistrationTestConstants.SSA + - "}"; - - public static String extendedRegistrationRequestJson = "{\n" + - " \"iss\": \"9b5usDpbNtmxDcTzs7GzKp\",\n" + - " \"iat\": 1593752054,\n" + - " \"exp\": 1743573565,\n" + - " \"jti\": \"92713892-5514-11e9-8647-d663bd873d93\",\n" + - " \"aud\": \"https://localbank.com\",\n" + - " \"scope\": \"accounts payments\",\n" + - " \"token_endpoint_auth_method\": \"private_key_jwt\",\n" + - " \"grant_types\": [\n" + - " \"authorization_code\",\n" + - " \"refresh_token\"\n" + - " ],\n" + - " \"response_types\": [\n" + - " \"code id_token\"\n" + - " ],\n" + - " \"id_token_signed_response_alg\": \"PS256\",\n" + - " \"request_object_signing_alg\": \"PS256\",\n" + - " \"software_id\": \"9b5usDpbNtmxDcTzs7GzKp\",\n" + - " \"application_type\": \"web\",\n" + - " \"software_statement\" : " + RegistrationTestConstants.SSA + - "}"; -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/validation/AlgorithmValidatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/validation/AlgorithmValidatorTest.java deleted file mode 100644 index 3e474e73..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/validation/AlgorithmValidatorTest.java +++ /dev/null @@ -1,108 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation; - -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.identity.dcr.validation.annotation.ValidateAlgorithm; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import org.apache.commons.beanutils.BeanUtils; -import org.mockito.Mock; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -import java.util.Arrays; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import javax.validation.ConstraintValidatorContext; - -import static org.mockito.Mockito.mock; -import static org.mockito.Mockito.when; - -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({BeanUtils.class, IdentityExtensionsDataHolder.class}) -public class AlgorithmValidatorTest extends PowerMockTestCase { - - private AlgorithmValidator validator; - - @Mock - private ValidateAlgorithm validateAlgorithm; - - @BeforeMethod - public void setUp() { - validator = new AlgorithmValidator(); - - when(validateAlgorithm.idTokenAlg()).thenReturn("idTokenAlg"); - when(validateAlgorithm.reqObjAlg()).thenReturn("reqObjAlg"); - when(validateAlgorithm.tokenAuthAlg()).thenReturn("tokenAuthAlg"); - - validator.initialize(validateAlgorithm); - } - - @Test - public void testIsValid_ReturnsTrue_WhenAlgorithmsAreAllowed() throws Exception { - Object requestObject = mock(Object.class); - ConstraintValidatorContext context = mock(ConstraintValidatorContext.class); - - PowerMockito.mockStatic(BeanUtils.class); - PowerMockito.when(BeanUtils.getProperty(requestObject, "idTokenAlg")).thenReturn("RS256"); - PowerMockito.when(BeanUtils.getProperty(requestObject, "reqObjAlg")).thenReturn("RS256"); - PowerMockito.when(BeanUtils.getProperty(requestObject, "tokenAuthAlg")).thenReturn("RS256"); - - List allowedAlgorithms = Arrays.asList("RS256", "HS256"); - - PowerMockito.mockStatic(IdentityExtensionsDataHolder.class); - IdentityExtensionsDataHolder dataHolder = PowerMockito.mock(IdentityExtensionsDataHolder.class); - Map configMap = new HashMap<>(); - configMap.put(OpenBankingConstants.SIGNATURE_ALGORITHMS, allowedAlgorithms); - when(dataHolder.getConfigurationMap()).thenReturn(configMap); - PowerMockito.when(IdentityExtensionsDataHolder.getInstance()).thenReturn(dataHolder); - - boolean result = validator.isValid(requestObject, context); - Assert.assertTrue(result); - } - - @Test - public void testIsValid_ReturnsFalse_WhenAlgorithmsAreNotAllowed() throws Exception { - Object requestObject = mock(Object.class); - ConstraintValidatorContext context = mock(ConstraintValidatorContext.class); - - PowerMockito.mockStatic(BeanUtils.class); - PowerMockito.when(BeanUtils.getProperty(requestObject, "idTokenAlg")).thenReturn("RS512"); - PowerMockito.when(BeanUtils.getProperty(requestObject, "reqObjAlg")).thenReturn("RS512"); - PowerMockito.when(BeanUtils.getProperty(requestObject, "tokenAuthAlg")).thenReturn("RS512"); - - List allowedAlgorithms = Arrays.asList("RS256", "HS256"); - - PowerMockito.mockStatic(IdentityExtensionsDataHolder.class); - IdentityExtensionsDataHolder dataHolder = PowerMockito.mock(IdentityExtensionsDataHolder.class); - Map configMap = new HashMap<>(); - configMap.put(OpenBankingConstants.SIGNATURE_ALGORITHMS, allowedAlgorithms); - when(dataHolder.getConfigurationMap()).thenReturn(configMap); - PowerMockito.when(IdentityExtensionsDataHolder.getInstance()).thenReturn(dataHolder); - - boolean result = validator.isValid(requestObject, context); - Assert.assertFalse(result); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/validation/IssuerValidatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/validation/IssuerValidatorTest.java deleted file mode 100644 index a5f30382..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/validation/IssuerValidatorTest.java +++ /dev/null @@ -1,81 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation; - -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.identity.dcr.validation.annotation.ValidateIssuer; -import org.apache.commons.beanutils.BeanUtils; -import org.mockito.Mock; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -import javax.validation.ConstraintValidatorContext; - -import static org.mockito.Mockito.mock; -import static org.mockito.Mockito.when; - -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({JWTUtils.class, BeanUtils.class}) -public class IssuerValidatorTest extends PowerMockTestCase { - - private IssuerValidator validator; - - @Mock - private ValidateIssuer validateIssuer; - - @BeforeMethod - public void setUp() { - validator = new IssuerValidator(); - - when(validateIssuer.issuerProperty()).thenReturn("issuer"); - when(validateIssuer.ssa()).thenReturn("ssa"); - - validator.initialize(validateIssuer); - } - - @Test - public void testIsValid_ReturnsTrue_WhenIssuerOrSoftwareStatementIsNull() throws Exception { - Object registrationRequest = mock(Object.class); - ConstraintValidatorContext context = mock(ConstraintValidatorContext.class); - - PowerMockito.mockStatic(BeanUtils.class); - PowerMockito.when(BeanUtils.getProperty(registrationRequest, "issuer")).thenReturn(null); - PowerMockito.when(BeanUtils.getProperty(registrationRequest, "ssa")).thenReturn(null); - - boolean result = validator.isValid(registrationRequest, context); - Assert.assertTrue(result); - } - - @Test - public void testIsValid_ReturnsFalse_OnException() throws Exception { - Object registrationRequest = mock(Object.class); - ConstraintValidatorContext context = mock(ConstraintValidatorContext.class); - - PowerMockito.mockStatic(BeanUtils.class); - PowerMockito.when(BeanUtils.getProperty(registrationRequest, "issuer")).thenThrow(new NoSuchMethodException()); - PowerMockito.when(BeanUtils.getProperty(registrationRequest, "ssa")).thenReturn("dummy-ssa"); - - boolean result = validator.isValid(registrationRequest, context); - Assert.assertFalse(result); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/validation/RequiredParamsValidatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/validation/RequiredParamsValidatorTest.java deleted file mode 100644 index 1c286cbf..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dcr/validation/RequiredParamsValidatorTest.java +++ /dev/null @@ -1,160 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.validation; - -import com.wso2.openbanking.accelerator.identity.dcr.model.RegistrationRequest; -import com.wso2.openbanking.accelerator.identity.dcr.validation.annotation.ValidateRequiredParams; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import org.hibernate.validator.internal.engine.constraintvalidation.ConstraintValidatorContextImpl; -import org.hibernate.validator.internal.engine.path.PathImpl; -import org.mockito.Mock; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -import java.util.Arrays; -import java.util.HashMap; -import java.util.Map; - -import javax.validation.ConstraintValidatorContext; - -import static org.mockito.Matchers.anyString; -import static org.mockito.Mockito.doReturn; -import static org.mockito.Mockito.mock; - -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({IdentityExtensionsDataHolder.class}) -public class RequiredParamsValidatorTest extends PowerMockTestCase { - - private RequiredParamsValidator validator; - - @Mock - private ValidateRequiredParams validateRequiredParams; - - private IdentityExtensionsDataHolder identityExtensionsDataHolderMock; - - @BeforeMethod - public void setUp() { - validator = new RequiredParamsValidator(); - validator.initialize(validateRequiredParams); - - PowerMockito.mockStatic(IdentityExtensionsDataHolder.class); - identityExtensionsDataHolderMock = PowerMockito.mock(IdentityExtensionsDataHolder.class); - PowerMockito.when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolderMock); - - // Mock the DCR registration config map with some test data - Map> configMap = new HashMap<>(); - Map paramConfig = new HashMap<>(); - paramConfig.put(DCRCommonConstants.DCR_REGISTRATION_PARAM_REQUIRED, "true"); - configMap.put("tokenEndPointAuthentication", paramConfig); - - Map scopeAllowedValuesConfig = new HashMap<>(); - scopeAllowedValuesConfig.put(DCRCommonConstants.DCR_REGISTRATION_PARAM_ALLOWED_VALUES, Arrays.asList( - "scope1", "scope2")); - configMap.put("scope", scopeAllowedValuesConfig); - - - Map appTypeAllowedValuesConfig = new HashMap<>(); - appTypeAllowedValuesConfig.put(DCRCommonConstants.DCR_REGISTRATION_PARAM_ALLOWED_VALUES, Arrays.asList( - "web", "mobile")); - configMap.put("applicationType", appTypeAllowedValuesConfig); - - PowerMockito.when(identityExtensionsDataHolderMock.getDcrRegistrationConfigMap()).thenReturn(configMap); - } - - @Test - public void testIsValid_ReturnsTrue_WhenAllRequestObjectIsEmpty() { - PowerMockito.when(identityExtensionsDataHolderMock.getDcrRegistrationConfigMap()).thenReturn(new HashMap<>()); - ConstraintValidatorContext context = mock(ConstraintValidatorContext.class); - doReturn(getConstraintViolationBuilder()).when(context).buildConstraintViolationWithTemplate(anyString()); - RegistrationRequest request = new RegistrationRequest(); - boolean result = validator.isValid(request, context); - Assert.assertTrue(result); - } - - @Test - public void testIsValid_ReturnsTrue_WhenRequiredParametersArePresent() { - ConstraintValidatorContext context = mock(ConstraintValidatorContext.class); - doReturn(getConstraintViolationBuilder()).when(context).buildConstraintViolationWithTemplate(anyString()); - RegistrationRequest request = getSampleRegistrationRequestWithRequiredParams(); - boolean result = validator.isValid(request, context); - Assert.assertTrue(result); - } - - @Test - public void testIsValid_ReturnsFalse_WhenRequiredParameterIsBlank() { - ConstraintValidatorContext context = mock(ConstraintValidatorContext.class); - doReturn(getConstraintViolationBuilder()).when(context).buildConstraintViolationWithTemplate(anyString()); - RegistrationRequest request = getSampleRegistrationRequestWithBlankRequiredParams(); - boolean result = validator.isValid(request, context); - Assert.assertFalse(result); - } - - @Test - public void testIsValid_ReturnsFalse_WhenScopeNotAllowed() { - ConstraintValidatorContext context = mock(ConstraintValidatorContext.class); - doReturn(getConstraintViolationBuilder()).when(context).buildConstraintViolationWithTemplate(anyString()); - RegistrationRequest request = getSampleRegistrationRequestWithScope(); - boolean result = validator.isValid(request, context); - Assert.assertFalse(result); - } - - private ConstraintValidatorContext.ConstraintViolationBuilder getConstraintViolationBuilder() { - - PathImpl propertyPath = PathImpl.createPathFromString("example.path"); - ConstraintValidatorContextImpl context = new ConstraintValidatorContextImpl( - null, - null, - propertyPath, - null, - null - ); - return context.buildConstraintViolationWithTemplate("message"); - } - - private RegistrationRequest getSampleRegistrationRequestWithRequiredParams() { - - RegistrationRequest registrationRequest = new RegistrationRequest(); - registrationRequest.setApplicationType("web"); - registrationRequest.setTokenEndPointAuthentication("auth_method"); - registrationRequest.setScope("scope1 scope2"); - return registrationRequest; - } - - private RegistrationRequest getSampleRegistrationRequestWithBlankRequiredParams() { - - RegistrationRequest registrationRequest = new RegistrationRequest(); - registrationRequest.setApplicationType("web"); - registrationRequest.setTokenEndPointAuthentication(""); - registrationRequest.setScope("scope1 scope2"); - return registrationRequest; - } - - private RegistrationRequest getSampleRegistrationRequestWithScope() { - - RegistrationRequest registrationRequest = new RegistrationRequest(); - registrationRequest.setTokenEndPointAuthentication("auth_method"); - registrationRequest.setScope("scope1 scope3"); - return registrationRequest; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dispute/resolution/DisputeResolutionFilterTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dispute/resolution/DisputeResolutionFilterTest.java deleted file mode 100644 index fec44953..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/dispute/resolution/DisputeResolutionFilterTest.java +++ /dev/null @@ -1,116 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dispute.resolution; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.util.ServiceProviderUtils; -import com.wso2.openbanking.accelerator.data.publisher.common.util.OBDataPublisherUtil; -import org.mockito.Mock; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.springframework.mock.web.MockHttpServletRequest; -import org.springframework.mock.web.MockHttpServletResponse; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; -import org.wso2.carbon.apimgt.api.APIManagementException; - -import java.util.Collections; -import java.util.Enumeration; -import java.util.HashMap; -import java.util.Map; - -import javax.servlet.FilterChain; - -import static org.mockito.Mockito.times; -import static org.mockito.Mockito.verify; -import static org.powermock.api.mockito.PowerMockito.when; - -/** - * Test for dispute resolution filter. - */ -@PrepareForTest({OpenBankingConfigParser.class, OBDataPublisherUtil.class, APIManagementException.class, - ServiceProviderUtils.class}) -@PowerMockIgnore("jdk.internal.reflect.*") -public class DisputeResolutionFilterTest extends PowerMockTestCase { - MockHttpServletRequest request; - MockHttpServletResponse response; - FilterChain filterChain; - - @Mock - OpenBankingConfigParser openBankingConfigParser; - - Map sampleRequestParams = new HashMap<>(); - - Map sampleHeaderMap = new HashMap<>(); - - @BeforeMethod - public void beforeMethod() { - - request = new MockHttpServletRequest(); - response = new MockHttpServletResponse(); - filterChain = Mockito.spy(FilterChain.class); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - openBankingConfigParser = PowerMockito.mock(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()) - .thenReturn(openBankingConfigParser); - } - - public DisputeResolutionFilterTest() { - sampleRequestParams.put("jsonRequest", ""); - sampleHeaderMap.put("Accept", "application/json"); - sampleHeaderMap.put("Postman-Token", "5c04e832-1a05-4d9a-974b-9e01d4c978f0"); - sampleHeaderMap.put("Host", "api.example.com"); - sampleHeaderMap.put("Accept-Encoding", "gzip, deflate, br"); - sampleHeaderMap.put("Connection", "keep-alive"); - sampleHeaderMap.put("Content-Type", "application/json"); - sampleHeaderMap.put("X-WSO2-Mutual-Auth-Cert", "MIIDczCCAlugAwIBAgIINeDHEkE4dGowDQYJKoZIhvcNAQELBQ" + - "AwJDEiMCAGA1UEAxMZcG9ydGFsLXByb2R1Y3Rpb24tc2lnbmVy"); - } - - @Test - public void capturingRequestResponseDataTest() throws Exception { - - when(openBankingConfigParser.isDisputeResolutionEnabled()).thenReturn(true); - when(openBankingConfigParser.isNonErrorDisputeDataPublishingEnabled()).thenReturn(true); - - DisputeResolutionFilter filter = Mockito.spy(DisputeResolutionFilter.class); - - request.setMethod("GET"); - request.setRequestURI("/register"); - request.setCharacterEncoding("UTF-8"); - request.setParameters(sampleRequestParams); - response.setStatus(200); - response.setCharacterEncoding("UTF-8"); - - Enumeration enumeration = Collections.enumeration(sampleHeaderMap.keySet()); - - PowerMockito.mockStatic(OBDataPublisherUtil.class); - PowerMockito.doNothing().when(OBDataPublisherUtil.class, "publishData", Mockito.anyString(), - Mockito.anyString(), Mockito.anyObject()); - - filter.doFilter(request, response, filterChain); - verify(filter, times(1)); - - } -} - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/idtoken/OBIDTokenBuilderTests.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/idtoken/OBIDTokenBuilderTests.java deleted file mode 100644 index 9347bf22..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/idtoken/OBIDTokenBuilderTests.java +++ /dev/null @@ -1,259 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.idtoken; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.mockito.Mockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedUser; -import org.wso2.carbon.identity.oauth.config.OAuthServerConfiguration; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.authz.OAuthAuthzReqMessageContext; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AccessTokenRespDTO; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AuthorizeReqDTO; -import org.wso2.carbon.identity.oauth2.dto.OAuth2AuthorizeRespDTO; -import org.wso2.carbon.identity.oauth2.token.OAuthTokenReqMessageContext; - -import java.util.HashMap; -import java.util.Map; - -import static org.powermock.api.mockito.PowerMockito.mock; -import static org.powermock.api.mockito.PowerMockito.mockStatic; -import static org.powermock.api.mockito.PowerMockito.when; - -/** - * Test for Open banking token builder. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({OAuthServerConfiguration.class, IdentityCommonUtil.class}) -public class OBIDTokenBuilderTests extends PowerMockTestCase { - - private OBIDTokenBuilder obidTokenBuilder; - private OAuthServerConfiguration oAuthServerConfigurationMock; - private OBIDTokenBuilderForSectorId obidTokenBuilderForSecId; - OBIDTokenBuilderForCallBackUri obidTokenBuilderForCallBackUri; - private AuthenticatedUser authenticatedUser = new AuthenticatedUser(); - private static final String USER = "admin@wso2.com"; - private static final String USER2 = "aaa@gold.com"; - private static final String CLIENT_ID = "DummyClientId"; - private static final String TENANT_DOMAIN = "DummyTenantDomain"; - private static final String SCOPES = "accounts basic:read openid"; - - @BeforeClass - public void setup() throws Exception { - - oAuthServerConfigurationMock = mock(OAuthServerConfiguration.class); - mockStatic(OAuthServerConfiguration.class); - when(OAuthServerConfiguration.getInstance()).thenReturn(oAuthServerConfigurationMock); - when(oAuthServerConfigurationMock.getIdTokenSignatureAlgorithm()).thenReturn("SHA256withRSA"); - - authenticatedUser.setUserName("aaa@gold.com"); - authenticatedUser.setTenantDomain("carbon.super"); - authenticatedUser.setUserStoreDomain("PRIMARY"); - authenticatedUser.setFederatedIdPName("LOCAL"); - authenticatedUser.setFederatedUser(false); - authenticatedUser.setAuthenticatedSubjectIdentifier("aaa@gold.com@carbon.super"); - - Map configMap = new HashMap<>(); - configMap.put(IdentityCommonConstants.ENABLE_SUBJECT_AS_PPID, true); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - obidTokenBuilder = new OBIDTokenBuilder(); - obidTokenBuilderForSecId = new OBIDTokenBuilderForSectorId(); - obidTokenBuilderForCallBackUri = new OBIDTokenBuilderForCallBackUri(); - } - - @Test - public void getSubjectClaimAuthFlowFromSectorIdentifierSuccess() throws Exception { - - OAuthAuthzReqMessageContext oAuthAuthzReqMessageContextMock = mock(OAuthAuthzReqMessageContext.class); - OAuth2AuthorizeReqDTO oAuth2AuthorizeReqDTOMock = mock(OAuth2AuthorizeReqDTO.class); - Mockito.doReturn(oAuth2AuthorizeReqDTOMock).when(oAuthAuthzReqMessageContextMock).getAuthorizationReqDTO(); - Mockito.doReturn("https://www.google.com/redirects/redirect1").when(oAuth2AuthorizeReqDTOMock) - .getCallbackUrl(); - - AuthenticatedUser authenticatedUserMock = mock(AuthenticatedUser.class); - Mockito.doReturn(authenticatedUserMock).when(oAuth2AuthorizeReqDTOMock).getUser(); - Mockito.doReturn(USER).when(authenticatedUserMock) - .getUsernameAsSubjectIdentifier(Mockito.anyBoolean(), Mockito.anyBoolean()); - - OAuth2AuthorizeRespDTO oAuth2AuthorizeRespDTOMock = mock(OAuth2AuthorizeRespDTO.class); - - mockStatic(IdentityCommonUtil.class); - when(IdentityCommonUtil.getRegulatoryFromSPMetaData(Mockito.anyString())).thenReturn(true); - - String subject = obidTokenBuilderForSecId.getSubjectClaim(oAuthAuthzReqMessageContextMock, - oAuth2AuthorizeRespDTOMock, CLIENT_ID, TENANT_DOMAIN, authenticatedUserMock); - Assert.assertNotNull(subject); - } - - @Test - public void getSubjectClaimTokenFlowFromSectorIdentifierSuccess() throws Exception { - - OAuthTokenReqMessageContext oAuthTokenReqMessageContextMock = mock(OAuthTokenReqMessageContext.class); - OAuth2AccessTokenRespDTO oAuth2AccessTokenRespDTOMock = mock(OAuth2AccessTokenRespDTO.class); - Mockito.doReturn("https://www.google.com/redirects/redirect1").when(oAuth2AccessTokenRespDTOMock) - .getCallbackURI(); - - Mockito.doReturn(SCOPES).when(oAuth2AccessTokenRespDTOMock) - .getAuthorizedScopes(); - - AuthenticatedUser authenticatedUserMock = mock(AuthenticatedUser.class); - Mockito.doReturn(authenticatedUserMock).when(oAuthTokenReqMessageContextMock).getAuthorizedUser(); - Mockito.doReturn(USER).when(authenticatedUserMock) - .getUsernameAsSubjectIdentifier(Mockito.anyBoolean(), Mockito.anyBoolean()); - - mockStatic(IdentityCommonUtil.class); - when(IdentityCommonUtil.getRegulatoryFromSPMetaData(Mockito.anyString())).thenReturn(true); - - String subject = obidTokenBuilderForSecId.getSubjectClaim(oAuthTokenReqMessageContextMock, - oAuth2AccessTokenRespDTOMock, CLIENT_ID, TENANT_DOMAIN, authenticatedUserMock); - Assert.assertNotNull(subject); - } - - @Test - public void getSubjectClaimAuthFlowFromCallBackUriSuccess() throws Exception { - - OAuthAuthzReqMessageContext oAuthAuthzReqMessageContextMock = mock(OAuthAuthzReqMessageContext.class); - OAuth2AuthorizeReqDTO oAuth2AuthorizeReqDTOMock = mock(OAuth2AuthorizeReqDTO.class); - Mockito.doReturn(oAuth2AuthorizeReqDTOMock).when(oAuthAuthzReqMessageContextMock).getAuthorizationReqDTO(); - Mockito.doReturn("https://www.google.com/redirects/redirect1").when(oAuth2AuthorizeReqDTOMock) - .getCallbackUrl(); - - AuthenticatedUser authenticatedUserMock = mock(AuthenticatedUser.class); - Mockito.doReturn(authenticatedUserMock).when(oAuth2AuthorizeReqDTOMock).getUser(); - Mockito.doReturn(USER).when(authenticatedUserMock) - .getUsernameAsSubjectIdentifier(Mockito.anyBoolean(), Mockito.anyBoolean()); - - OAuth2AuthorizeRespDTO oAuth2AuthorizeRespDTOMock = mock(OAuth2AuthorizeRespDTO.class); - - mockStatic(IdentityCommonUtil.class); - when(IdentityCommonUtil.getRegulatoryFromSPMetaData(Mockito.anyString())).thenReturn(true); - - String subject = obidTokenBuilderForCallBackUri.getSubjectClaim(oAuthAuthzReqMessageContextMock, - oAuth2AuthorizeRespDTOMock, CLIENT_ID, TENANT_DOMAIN, authenticatedUserMock); - Assert.assertNotNull(subject); - } - - @Test - public void getSubjectClaimTokenFlowFromCallBackUriSuccess() throws Exception { - - OAuthTokenReqMessageContext oAuthTokenReqMessageContextMock = mock(OAuthTokenReqMessageContext.class); - OAuth2AccessTokenRespDTO oAuth2AccessTokenRespDTOMock = mock(OAuth2AccessTokenRespDTO.class); - Mockito.doReturn("regexp=(https://www.google.com/redirects/redirect1|" + - "https://www.google.com/redirects/redirect2)").when(oAuth2AccessTokenRespDTOMock) - .getCallbackURI(); - - Mockito.doReturn(SCOPES).when(oAuth2AccessTokenRespDTOMock) - .getAuthorizedScopes(); - - AuthenticatedUser authenticatedUserMock = mock(AuthenticatedUser.class); - Mockito.doReturn(authenticatedUserMock).when(oAuthTokenReqMessageContextMock).getAuthorizedUser(); - Mockito.doReturn(USER).when(authenticatedUserMock) - .getUsernameAsSubjectIdentifier(Mockito.anyBoolean(), Mockito.anyBoolean()); - - mockStatic(IdentityCommonUtil.class); - when(IdentityCommonUtil.getRegulatoryFromSPMetaData(Mockito.anyString())).thenReturn(true); - - String subject = obidTokenBuilderForCallBackUri.getSubjectClaim(oAuthTokenReqMessageContextMock, - oAuth2AccessTokenRespDTOMock, CLIENT_ID, TENANT_DOMAIN, authenticatedUserMock); - Assert.assertNotNull(subject); - } - - @Test - public void getNonPPIDSubjectClaimWithoutTenantAndUserDomainSuccess() throws Exception { - - Map configMap = new HashMap<>(); - configMap.put(IdentityCommonConstants.REMOVE_USER_STORE_DOMAIN_FROM_SUBJECT, true); - configMap.put(IdentityCommonConstants.REMOVE_TENANT_DOMAIN_FROM_SUBJECT, true); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - OBIDTokenBuilderForCallBackUri obidTokenBuilderForCallBackUri = new OBIDTokenBuilderForCallBackUri(); - - OAuthAuthzReqMessageContext oAuthAuthzReqMessageContextMock = mock(OAuthAuthzReqMessageContext.class); - OAuth2AuthorizeReqDTO oAuth2AuthorizeReqDTOMock = mock(OAuth2AuthorizeReqDTO.class); - Mockito.doReturn(oAuth2AuthorizeReqDTOMock).when(oAuthAuthzReqMessageContextMock).getAuthorizationReqDTO(); - Mockito.doReturn("https://www.google.com/redirects/redirect1").when(oAuth2AuthorizeReqDTOMock) - .getCallbackUrl(); - - oAuth2AuthorizeReqDTOMock.setUser(authenticatedUser); - Mockito.doReturn(authenticatedUser).when(oAuth2AuthorizeReqDTOMock).getUser(); - - OAuth2AuthorizeRespDTO oAuth2AuthorizeRespDTOMock = mock(OAuth2AuthorizeRespDTO.class); - - mockStatic(IdentityCommonUtil.class); - when(IdentityCommonUtil.getRegulatoryFromSPMetaData(Mockito.anyString())).thenReturn(true); - - String subject = obidTokenBuilderForCallBackUri.getSubjectClaim(oAuthAuthzReqMessageContextMock, - oAuth2AuthorizeRespDTOMock, CLIENT_ID, TENANT_DOMAIN, authenticatedUser); - Assert.assertNotNull(subject); - Assert.assertEquals(subject, USER2); - - OAuthTokenReqMessageContext oAuthTokenReqMessageContextMock = mock(OAuthTokenReqMessageContext.class); - OAuth2AccessTokenRespDTO oAuth2AccessTokenRespDTOMock = mock(OAuth2AccessTokenRespDTO.class); - Mockito.doReturn("https://www.google.com/redirects/redirect1").when(oAuth2AccessTokenRespDTOMock) - .getCallbackURI(); - - Mockito.doReturn(SCOPES).when(oAuth2AccessTokenRespDTOMock) - .getAuthorizedScopes(); - - oAuthTokenReqMessageContextMock.setAuthorizedUser(authenticatedUser); - Mockito.doReturn(authenticatedUser).when(oAuthTokenReqMessageContextMock).getAuthorizedUser(); - - mockStatic(IdentityCommonUtil.class); - when(IdentityCommonUtil.getRegulatoryFromSPMetaData(Mockito.anyString())).thenReturn(true); - - String subjectTokenFlow = obidTokenBuilderForCallBackUri.getSubjectClaim(oAuthTokenReqMessageContextMock, - oAuth2AccessTokenRespDTOMock, CLIENT_ID, TENANT_DOMAIN, authenticatedUser); - Assert.assertNotNull(subjectTokenFlow); - Assert.assertEquals(subjectTokenFlow, USER2); - } - -} - -class OBIDTokenBuilderForSectorId extends OBIDTokenBuilder { - - public OBIDTokenBuilderForSectorId() throws IdentityOAuth2Exception { - } - - @Override - protected String getSectorIdentifierUri(String clientId) throws OpenBankingException { - - return "https://wso2.com/"; - } -} - -class OBIDTokenBuilderForCallBackUri extends OBIDTokenBuilder { - - public OBIDTokenBuilderForCallBackUri() throws IdentityOAuth2Exception { - } - - @Override - protected String getSectorIdentifierUri(String clientId) throws OpenBankingException { - - return null; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/listener/TokenRevocationListenerTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/listener/TokenRevocationListenerTest.java deleted file mode 100644 index f744fbbc..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/listener/TokenRevocationListenerTest.java +++ /dev/null @@ -1,64 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.listener; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import org.junit.Assert; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -import java.util.HashMap; -import java.util.Map; -/** - * Test class for TokenRevocationListener. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({OpenBankingConfigParser.class}) -public class TokenRevocationListenerTest extends PowerMockTestCase { - - OpenBankingConfigParser openBankingConfigParserMock; - private Map configMap = new HashMap<>(); - - @BeforeClass - public void init() { - mockOpenBankingConfigParser(); - configMap.put("Identity.ConsentIDClaimName", "consent_id"); - } - - @Test - public void testGetConsentIdFromScopes() { - String[] scopes = {"dummy-scope1", "dummy-scope2", "consent_idConsentId", "dummy-scope3"}; - TokenRevocationListener tokenRevocationListener = new TokenRevocationListener(); - String consentId = tokenRevocationListener.getConsentIdFromScopes(scopes); - Assert.assertEquals("ConsentId", consentId); - } - - private void mockOpenBankingConfigParser() { - openBankingConfigParserMock = Mockito.mock(OpenBankingConfigParser.class); - Mockito.when(openBankingConfigParserMock.getConfiguration()) - .thenReturn(configMap); - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/PushAuthRequestValidatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/PushAuthRequestValidatorTest.java deleted file mode 100644 index b21e6c1f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/PushAuthRequestValidatorTest.java +++ /dev/null @@ -1,421 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.ServiceProviderUtils; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.constants.PushAuthRequestConstants; -import com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.exception.PushAuthRequestValidatorException; -import com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.model.PushAuthErrorResponse; -import com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.util.test.jwt.builder.TestJwtBuilder; -import net.minidev.json.JSONObject; -import org.junit.Assert; -import org.mockito.Mockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.application.common.IdentityApplicationManagementException; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.application.common.model.ServiceProviderProperty; -import org.wso2.carbon.identity.application.mgt.ApplicationManagementService; -import org.wso2.carbon.identity.core.util.IdentityUtil; -import org.wso2.carbon.identity.oauth.config.OAuthServerConfiguration; -import org.wso2.carbon.identity.oauth.dao.OAuthAppDO; -import org.wso2.carbon.identity.oauth2.dto.OAuth2ClientValidationResponseDTO; -import org.wso2.carbon.identity.oauth2.util.OAuth2Util; - -import java.io.File; -import java.io.FileInputStream; -import java.io.InputStream; -import java.security.KeyStore; -import java.security.interfaces.RSAPrivateKey; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import javax.servlet.http.HttpServletRequest; - -import static org.powermock.api.mockito.PowerMockito.mock; -import static org.powermock.api.mockito.PowerMockito.mockStatic; -import static org.powermock.api.mockito.PowerMockito.when; - -/** - * Test for push authorization request validator. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({IdentityExtensionsDataHolder.class, OAuth2Util.class, OAuthServerConfiguration.class, - IdentityUtil.class, ServiceProviderUtils.class}) -public class PushAuthRequestValidatorTest extends PowerMockTestCase { - - private Map> parameterMap; - private Map configMap; - private PushAuthRequestValidator pushAuthRequestValidator; - private HttpServletRequest httpServletRequestMock; - private ServiceProviderProperty[] spProperties; - private ServiceProvider serviceProviderMock; - - @BeforeClass - public void setup() throws Exception { - - parameterMap = new HashMap<>(); - configMap = new HashMap<>(); - parameterMap.put("request", Arrays.asList(TestJwtBuilder.getValidSignedJWT())); - configMap.put("SignatureValidation.AllowedAlgorithms.Algorithm", - new ArrayList<>(Arrays.asList("PS256 ES256".split(" ")))); - httpServletRequestMock = mock(HttpServletRequest.class); - pushAuthRequestValidator = new PushAuthRequestValidator(); - - serviceProviderMock = new ServiceProvider(); - - ServiceProviderProperty serviceProviderProperty = new ServiceProviderProperty(); - serviceProviderProperty.setName("scope"); - serviceProviderProperty.setValue("accounts payments"); - spProperties = new ServiceProviderProperty[1]; - spProperties[0] = serviceProviderProperty; - serviceProviderMock.setSpProperties(spProperties); - } - - @BeforeMethod - public void initMethods() throws OpenBankingException, IdentityApplicationManagementException { - - IdentityExtensionsDataHolder identityExtensionsDataHolderMock = mock(IdentityExtensionsDataHolder.class); - ApplicationManagementService applicationManagementServiceMock = mock(ApplicationManagementService.class); - - mockStatic(IdentityExtensionsDataHolder.class); - mockStatic(ServiceProviderUtils.class); - when(IdentityExtensionsDataHolder.getInstance()).thenReturn(identityExtensionsDataHolderMock); - when(identityExtensionsDataHolderMock.getConfigurationMap()).thenReturn(configMap); - when(identityExtensionsDataHolderMock.getApplicationManagementService()) - .thenReturn(applicationManagementServiceMock); - when(ServiceProviderUtils.getSpTenantDomain(Mockito.anyString())).thenReturn("dummyTenantDomain"); - when(applicationManagementServiceMock.getServiceProviderByClientId(Mockito.anyString(), - Mockito.anyString(), Mockito.anyString())).thenReturn(serviceProviderMock); - - OAuthServerConfiguration oAuthServerConfigurationMock = mock(OAuthServerConfiguration.class); - mockStatic(OAuthServerConfiguration.class); - when(OAuthServerConfiguration.getInstance()).thenReturn(oAuthServerConfigurationMock); - } - - @Test(expectedExceptions = PushAuthRequestValidatorException.class) - public void validateRepeatedParametersInRequest() throws Exception { - - parameterMap.put("client_assertion", Arrays.asList("firstParam,secondRepeatedParam".split(","))); - pushAuthRequestValidator.validateParams(httpServletRequestMock, parameterMap); - } - - @Test(expectedExceptions = PushAuthRequestValidatorException.class, - dependsOnMethods = "validateRepeatedParametersInRequest") - public void validateRequestUriParamInRequest() throws Exception { - - // remove previous invalid parameters - parameterMap.remove("client_assertion"); - // add new parameters to be tested - parameterMap.put("request_uri", Arrays.asList("dummyValue")); - pushAuthRequestValidator.validateParams(httpServletRequestMock, parameterMap); - } - - @Test(expectedExceptions = PushAuthRequestValidatorException.class, - dependsOnMethods = "validateRequestUriParamInRequest") - public void validateFormBodyParamsInRequest() throws Exception { - - // remove previous invalid parameters - parameterMap.remove("request_uri"); - // add new parameters to be tested - parameterMap.put("scope", Arrays.asList("dummyScope")); - pushAuthRequestValidator.validateParams(httpServletRequestMock, parameterMap); - } - - @Test(expectedExceptions = PushAuthRequestValidatorException.class, - dependsOnMethods = "validateRequestUriParamInRequest") - public void validateRequestObject() throws Exception { - - // remove previous invalid parameters - parameterMap.remove("scope"); - // add new parameters to be tested - parameterMap.put("request", Arrays.asList("invalidReqObj")); - pushAuthRequestValidator.validateParams(httpServletRequestMock, parameterMap); - } - - @Test(expectedExceptions = PushAuthRequestValidatorException.class, - dependsOnMethods = "validateRequestObject") - public void validateClientIdInRequest() throws Exception { - - // add parameters to be tested - parameterMap.put("request", - Arrays.asList(TestJwtBuilder.getValidSignedJWT())); - PushAuthRequestValidatorInvalidClientMock pushAuthRequestValidatorInvalidClientMock = - new PushAuthRequestValidatorInvalidClientMock(); - pushAuthRequestValidatorInvalidClientMock.validateParams(httpServletRequestMock, parameterMap); - } - - @Test(expectedExceptions = PushAuthRequestValidatorException.class, - dependsOnMethods = "validateClientIdInRequest") - public void validateSignatureAlgInRequestObject() throws Exception { - - // add parameters to be tested - parameterMap.put("request", - Arrays.asList(TestJwtBuilder.getInvalidJWTWithUnsupportedAlgorithm())); - PushAuthRequestValidatorMockClass pushAuthRequestValidatorMockClass = new PushAuthRequestValidatorMockClass(); - - pushAuthRequestValidatorMockClass.validateParams(httpServletRequestMock, parameterMap); - } - - @Test(expectedExceptions = PushAuthRequestValidatorException.class, - dependsOnMethods = "validateSignatureAlgInRequestObject") - public void validateNonceInRequestObject() throws Exception { - - parameterMap.put("request", - Arrays.asList(TestJwtBuilder.getInvalidJWTWithUnsupportedNonce())); - PushAuthRequestValidatorMockClass pushAuthRequestValidatorMockClass = new PushAuthRequestValidatorMockClass(); - - pushAuthRequestValidatorMockClass.validateParams(httpServletRequestMock, parameterMap); - } - - @Test(expectedExceptions = PushAuthRequestValidatorException.class, - dependsOnMethods = "validateNonceInRequestObject") - public void validateScopeParameter() throws Exception { - - parameterMap.put("request", - Arrays.asList(TestJwtBuilder.getValidSignedJWT())); - PushAuthRequestValidatorMockClass pushAuthRequestValidatorMockClass = new PushAuthRequestValidatorMockClass(); - - pushAuthRequestValidatorMockClass.validateParams(httpServletRequestMock, parameterMap); - } - - @Test(expectedExceptions = PushAuthRequestValidatorException.class, - dependsOnMethods = "validateNonceInRequestObject") - public void validateUnsupportedClaimsInSignedJWT() throws Exception { - - parameterMap.put("request", - Arrays.asList(TestJwtBuilder.getInvalidJWTWithUnsupportedClaims())); - - ServiceProviderProperty[] serviceProviderProperties = serviceProviderMock.getSpProperties(); - serviceProviderProperties[0].setValue("bank:accounts.basic:read bank:accounts.detail:read " + - "bank:transactions:read bank:payees:read bank:regular_payments:read common:customer.basic:read " + - "common:customer.detail:read cdr:registration openid"); - - PushAuthRequestValidatorMockClass pushAuthRequestValidatorMockClass = new PushAuthRequestValidatorMockClass(); - - pushAuthRequestValidatorMockClass.validateParams(httpServletRequestMock, parameterMap); - } - - @Test(dependsOnMethods = "validateUnsupportedClaimsInSignedJWT") - public void successfulParameterValidationFlowForSignedJWT() throws Exception { - - parameterMap.put("request", - Arrays.asList(TestJwtBuilder.getValidSignedJWT())); - - ServiceProviderProperty[] serviceProviderProperties = serviceProviderMock.getSpProperties(); - serviceProviderProperties[0].setValue("bank:accounts.basic:read bank:accounts.detail:read " + - "bank:transactions:read bank:payees:read bank:regular_payments:read common:customer.basic:read " + - "common:customer.detail:read cdr:registration openid"); - - PushAuthRequestValidatorMockClass pushAuthRequestValidatorMockClass = new PushAuthRequestValidatorMockClass(); - - Map result = pushAuthRequestValidatorMockClass - .validateParams(httpServletRequestMock, parameterMap); - - Assert.assertNotNull(result); - } - - @Test(priority = 1) - public void testErrorResponseCreation() { - - PushAuthErrorResponse result = pushAuthRequestValidator.createErrorResponse(400, - PushAuthRequestConstants.INVALID_REQUEST, "Bad Request"); - - Assert.assertEquals("Bad Request", result.getPayload().get("error_description").toString()); - } - - @Test(expectedExceptions = PushAuthRequestValidatorException.class, - dependsOnMethods = "validateScopeParameter") - public void validateMissingExpClaimInSignedJWT() throws Exception { - parameterMap.put("request", - Arrays.asList(TestJwtBuilder.getInvalidJWTWithoutExpClaim())); - - ServiceProviderProperty[] serviceProviderProperties = serviceProviderMock.getSpProperties(); - serviceProviderProperties[0].setValue("bank:accounts.basic:read bank:accounts.detail:read " + - "bank:transactions:read bank:payees:read bank:regular_payments:read common:customer.basic:read " + - "common:customer.detail:read cdr:registration openid"); - - PushAuthRequestValidatorMockClass pushAuthRequestValidatorMockClass = new PushAuthRequestValidatorMockClass(); - - pushAuthRequestValidatorMockClass.validateParams(httpServletRequestMock, parameterMap); - } - - @Test(expectedExceptions = PushAuthRequestValidatorException.class, - dependsOnMethods = "validateMissingExpClaimInSignedJWT") - public void validateExpClaimOver60MinInSignedJWT() throws Exception { - parameterMap.put("request", - Arrays.asList(TestJwtBuilder.getInvalidJWTWithExpClaimOver60Min())); - - ServiceProviderProperty[] serviceProviderProperties = serviceProviderMock.getSpProperties(); - serviceProviderProperties[0].setValue("bank:accounts.basic:read bank:accounts.detail:read " + - "bank:transactions:read bank:payees:read bank:regular_payments:read common:customer.basic:read " + - "common:customer.detail:read cdr:registration openid"); - - PushAuthRequestValidatorMockClass pushAuthRequestValidatorMockClass = new PushAuthRequestValidatorMockClass(); - - pushAuthRequestValidatorMockClass.validateParams(httpServletRequestMock, parameterMap); - } - - @Test(expectedExceptions = PushAuthRequestValidatorException.class, - dependsOnMethods = "validateExpClaimOver60MinInSignedJWT") - public void validateMissingNbfClaimInSignedJWT() throws Exception { - parameterMap.put("request", - Arrays.asList(TestJwtBuilder.getInvalidJWTWithoutNbfClaim())); - - ServiceProviderProperty[] serviceProviderProperties = serviceProviderMock.getSpProperties(); - serviceProviderProperties[0].setValue("bank:accounts.basic:read bank:accounts.detail:read " + - "bank:transactions:read bank:payees:read bank:regular_payments:read common:customer.basic:read " + - "common:customer.detail:read cdr:registration openid"); - - PushAuthRequestValidatorMockClass pushAuthRequestValidatorMockClass = new PushAuthRequestValidatorMockClass(); - - pushAuthRequestValidatorMockClass.validateParams(httpServletRequestMock, parameterMap); - } - - @Test(expectedExceptions = PushAuthRequestValidatorException.class, - dependsOnMethods = "validateMissingNbfClaimInSignedJWT") - public void validateNbfClaimOver60MinInSignedJWT() throws Exception { - parameterMap.put("request", - Arrays.asList(TestJwtBuilder.getInvalidJWTWithNbfClaimOver60Min())); - - ServiceProviderProperty[] serviceProviderProperties = serviceProviderMock.getSpProperties(); - serviceProviderProperties[0].setValue("bank:accounts.basic:read bank:accounts.detail:read " + - "bank:transactions:read bank:payees:read bank:regular_payments:read common:customer.basic:read " + - "common:customer.detail:read cdr:registration openid"); - - PushAuthRequestValidatorMockClass pushAuthRequestValidatorMockClass = new PushAuthRequestValidatorMockClass(); - - pushAuthRequestValidatorMockClass.validateParams(httpServletRequestMock, parameterMap); - } - - @Test(expectedExceptions = PushAuthRequestValidatorException.class, - dependsOnMethods = "validateMissingNbfClaimInSignedJWT") - public void validateMissingCodeChallengeInSignedJWT() throws Exception { - - // add parameters to be tested - parameterMap.put("request", - Arrays.asList(TestJwtBuilder.getInvalidJWTWithoutCodeChallenge())); - PushAuthRequestValidatorMockClass pushAuthRequestValidatorMockClass = new PushAuthRequestValidatorMockClass(); - - pushAuthRequestValidatorMockClass.validateParams(httpServletRequestMock, parameterMap); - } - - @Test(expectedExceptions = PushAuthRequestValidatorException.class, - dependsOnMethods = "validateMissingCodeChallengeInSignedJWT") - public void validateMissingCodeChallengeMethodInSignedJWT() throws Exception { - - // add parameters to be tested - parameterMap.put("request", - Arrays.asList(TestJwtBuilder.getInvalidJWTWithoutCodeChallengeMethod())); - PushAuthRequestValidatorMockClass pushAuthRequestValidatorMockClass = new PushAuthRequestValidatorMockClass(); - - pushAuthRequestValidatorMockClass.validateParams(httpServletRequestMock, parameterMap); - } - - @Test(expectedExceptions = PushAuthRequestValidatorException.class, - dependsOnMethods = "validateMissingCodeChallengeMethodInSignedJWT") - public void validateMissingResponseTypeInSignedJWT() throws Exception { - - // add parameters to be tested - parameterMap.put("request", - Arrays.asList(TestJwtBuilder.getInvalidJWTWithoutResponseType())); - PushAuthRequestValidatorMockClass pushAuthRequestValidatorMockClass = new PushAuthRequestValidatorMockClass(); - - pushAuthRequestValidatorMockClass.validateParams(httpServletRequestMock, parameterMap); - } - - @Test(priority = 2, expectedExceptions = PushAuthRequestValidatorException.class) - public void testDecryptEncryptedReqObjFailure() throws Exception { - - parameterMap.put("request", - Arrays.asList(TestJwtBuilder.getValidEncryptedJWT())); - - OAuthServerConfiguration oAuthServerConfigurationMock = mock(OAuthServerConfiguration.class); - mockStatic(OAuthServerConfiguration.class); - when(OAuthServerConfiguration.getInstance()).thenReturn(oAuthServerConfigurationMock); - - mockStatic(OAuth2Util.class); - OAuthAppDO oAuthAppDOMock = mock(OAuthAppDO.class); - when(OAuth2Util.getAppInformationByClientId(Mockito.anyString())).thenReturn(oAuthAppDOMock); - when(OAuth2Util.getTenantDomainOfOauthApp(oAuthAppDOMock)).thenReturn("dummyTenantDomain"); - - String path = "src/test/resources"; - File file = new File(path); - String absolutePathForTestResources = file.getAbsolutePath(); - String absolutePathForKeyStore = absolutePathForTestResources + "/wso2carbon.jks"; - String[] pathParts = absolutePathForKeyStore.split("/"); - String platformAbsolutePathForKeyStore = String.join(File.separator, pathParts); - - InputStream keystoreFile = new FileInputStream(platformAbsolutePathForKeyStore); - KeyStore keystore = KeyStore.getInstance(KeyStore.getDefaultType()); - keystore.load(keystoreFile, "wso2carbon".toCharArray()); - - String alias = "wso2carbon"; - - // Get the private key. Password for the key store is 'wso2carbon'. - RSAPrivateKey privateKey = (RSAPrivateKey) keystore.getKey(alias, "wso2carbon".toCharArray()); - - when(OAuth2Util.getPrivateKey(Mockito.anyString(), Mockito.anyInt())).thenReturn(privateKey); - - pushAuthRequestValidator.validateParams(httpServletRequestMock, parameterMap); - } -} - -class PushAuthRequestValidatorMockClass extends PushAuthRequestValidator { - - @Override - protected OAuth2ClientValidationResponseDTO getClientValidationInfo(JSONObject requestBodyJson) { - - OAuth2ClientValidationResponseDTO oAuth2ClientValidationResponseDTO = new OAuth2ClientValidationResponseDTO(); - oAuth2ClientValidationResponseDTO.setValidClient(true); - return oAuth2ClientValidationResponseDTO; - } - - @Override - protected void validateSignature(String requestObjectString, JSONObject requestBodyJson) - throws PushAuthRequestValidatorException { - - } - - @Override - protected void validateAudience(JSONObject requestBodyJson) - throws PushAuthRequestValidatorException { - - } -} - -class PushAuthRequestValidatorInvalidClientMock extends PushAuthRequestValidator { - - @Override - protected OAuth2ClientValidationResponseDTO getClientValidationInfo(JSONObject requestBodyJson) { - - OAuth2ClientValidationResponseDTO oAuth2ClientValidationResponseDTO = new OAuth2ClientValidationResponseDTO(); - oAuth2ClientValidationResponseDTO.setValidClient(false); - return oAuth2ClientValidationResponseDTO; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/util/test/jwt/builder/TestJwtBuilder.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/util/test/jwt/builder/TestJwtBuilder.java deleted file mode 100644 index 76c5543c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/util/test/jwt/builder/TestJwtBuilder.java +++ /dev/null @@ -1,401 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.util.test.jwt.builder; - -import com.nimbusds.jose.EncryptionMethod; -import com.nimbusds.jose.JOSEObjectType; -import com.nimbusds.jose.JWEAlgorithm; -import com.nimbusds.jose.JWEEncrypter; -import com.nimbusds.jose.JWEHeader; -import com.nimbusds.jose.JWEObject; -import com.nimbusds.jose.JWSAlgorithm; -import com.nimbusds.jose.JWSHeader; -import com.nimbusds.jose.JWSSigner; -import com.nimbusds.jose.Payload; -import com.nimbusds.jose.crypto.RSAEncrypter; -import com.nimbusds.jose.crypto.RSASSASigner; -import com.nimbusds.jose.crypto.bc.BouncyCastleProviderSingleton; -import com.nimbusds.jwt.JWTClaimsSet; -import com.nimbusds.jwt.SignedJWT; -import com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.util.test.jwt.builder.constants.TestJwtBuilderConstants; - -import java.io.File; -import java.io.FileInputStream; -import java.io.InputStream; -import java.security.KeyStore; -import java.security.cert.Certificate; -import java.security.interfaces.RSAPrivateKey; -import java.security.interfaces.RSAPublicKey; -import java.time.Instant; -import java.util.Date; -import java.util.HashMap; -import java.util.Map; - -/** - * JWT Builder class to Automate the process of building a JWT for Testing purposes. - * - */ -public class TestJwtBuilder { - private static String issuer = "wHKH6jd5YRJtG_CXSLVfcStMfOAa"; - private static String responseType = "code id_token"; - private static String codeChallengeMethod = "S256"; - private static String nonce = "n-jBXhOmOKCB"; - private static String invalidNonce = null; - private static String clientId = "wHKH6jd5YRJtG_CXSLVfcStMfOAa"; - private static String audience = "https://localhost:9443/oauth2/token"; - private static String scope = "bank:accounts.basic:read bank:transactions:read " + - "common:customer.detail:read openid"; - private static String redirectUri = "https://www.google.com/redirects/redirect1"; - private static String state = "0pN0NBTHcv"; - private static String codeChallenge = "E9Melhoa2OwvFrEMTJguCHaoeK1t8URWbuGJSstw-cM"; - private static String kid = "W_TcnQVcHAy20q8zCMcdByrootw"; - private static String alias = "wso2carbon"; - private static String keyPassword = "wso2carbon"; - private static String acrValue = "urn:cds.au:cdr:2"; - private static String invalidParameter = "invalidParameter"; - private static boolean acrEssential = true; - private static int sharingDuration = 7776000; - private static int expiryPeriod = 3600; - private static int notBeforePeriod = 3600; - private static String keyStorePath = "src/test/resources"; - - private TestJwtBuilder() { - } - - /** - * This method is used to get a valid signed JWT with signature algorithm PS256. - * - * @return String - * @throws Exception if an error occurs - */ - public static String getValidSignedJWT() throws Exception { - JWTClaimsSet claimsSet = getValidJWTClaimsSetBuilder().build(); - SignedJWT signedJWT = getSignedJWT(JWSAlgorithm.PS256, claimsSet); - return signedJWT.serialize(); - } - - /** - * This method is used to get a valid encrypted JWT with signature algorithm PS256, - * encryption algorithm RSA-OAEP-256 and encryption method A256GCM. - * - * @return String - * @throws Exception if an error occurs - */ - public static String getValidEncryptedJWT() throws Exception { - JWTClaimsSet claimsSet = getValidJWTClaimsSetBuilder().build(); - SignedJWT signedJWT = getSignedJWT(JWSAlgorithm.PS256 , claimsSet); - return getEncryptedJWT(JWEAlgorithm.RSA_OAEP_256, EncryptionMethod.A128GCM, signedJWT).serialize(); - } - - /** - * This method is used to get a invalid JWT signed with unsupported signature algorithm. - * ex: RS256 - * - * @return String - * @throws Exception if an error occurs - */ - public static String getInvalidJWTWithUnsupportedAlgorithm() throws Exception { - JWTClaimsSet claimsSet = getValidJWTClaimsSetBuilder().build(); - SignedJWT signedJWT = getSignedJWT(JWSAlgorithm.RS256, claimsSet); - return signedJWT.serialize(); - } - - /** - * This method is used to get a invalid JWT with invalid nonce value. - * ex: null - * - * @return String - * @throws Exception if an error occurs - */ - public static String getInvalidJWTWithUnsupportedNonce() throws Exception { - - JWTClaimsSet.Builder builder = getValidJWTClaimsSetBuilder(); - JWTClaimsSet invalidClaimSet = builder.claim(TestJwtBuilderConstants.NONCE, invalidNonce).build(); - SignedJWT signedJWT = getSignedJWT(JWSAlgorithm.PS256, invalidClaimSet); - return signedJWT.serialize(); - } - - /** - * This method is used to get an invalid JWT with unsupported claims value. - * - * @return String - * @throws Exception if an error occurs - */ - public static String getInvalidJWTWithUnsupportedClaims() throws Exception { - Map invalidClaimsMap = getValidClaimsMap(); - JWTClaimsSet.Builder builder = getValidJWTClaimsSetBuilder(); - JWTClaimsSet invalidClaimSet = builder - .claim(TestJwtBuilderConstants.CLAIMS, invalidClaimsMap) - .claim(TestJwtBuilderConstants.REQUEST, invalidParameter) - .build(); - SignedJWT signedJWT = getSignedJWT(JWSAlgorithm.PS256, invalidClaimSet); - return signedJWT.serialize(); - } - - /** - * This method is used to get an invalid JWT with exp claim over 60 minutes in the future. - * - * @return String - * @throws Exception if an error occurs - */ - public static String getInvalidJWTWithExpClaimOver60Min() throws Exception { - JWTClaimsSet.Builder builder = getValidJWTClaimsSetBuilder(); - JWTClaimsSet invalidClaimSet = builder - .expirationTime(Date.from(Instant.now().plusSeconds(expiryPeriod + 1500))) - .build(); - SignedJWT signedJWT = getSignedJWT(JWSAlgorithm.PS256, invalidClaimSet); - return signedJWT.serialize(); - } - - /** - * This method is used to get an invalid JWT without an exp claim. - * - * @return String - * @throws Exception if an error occurs - */ - public static String getInvalidJWTWithoutExpClaim() throws Exception { - JWTClaimsSet.Builder builder = getValidJWTClaimsSetBuilder(); - JWTClaimsSet invalidClaimSet = builder - .expirationTime(null) - .build(); - SignedJWT signedJWT = getSignedJWT(JWSAlgorithm.PS256, invalidClaimSet); - return signedJWT.serialize(); - } - - /** - * This method is used to get an invalid JWT without the code challenge. - * - * @return String - * @throws Exception if an error occurs - */ - public static String getInvalidJWTWithoutCodeChallenge() throws Exception { - - JWTClaimsSet.Builder builder = getValidJWTClaimsSetBuilder(); - JWTClaimsSet invalidClaimSet = builder - .claim(TestJwtBuilderConstants.CODE_CHALLENGE, null) - .build(); - SignedJWT signedJWT = getSignedJWT(JWSAlgorithm.PS256, invalidClaimSet); - return signedJWT.serialize(); - } - - /** - * This method is used to get an invalid JWT without the code challenge method. - * - * @return String - * @throws Exception if an error occurs - */ - public static String getInvalidJWTWithoutCodeChallengeMethod() throws Exception { - - JWTClaimsSet.Builder builder = getValidJWTClaimsSetBuilder(); - JWTClaimsSet invalidClaimSet = builder - .claim(TestJwtBuilderConstants.CODE_CHALLENGE_METHOD, null) - .build(); - SignedJWT signedJWT = getSignedJWT(JWSAlgorithm.PS256, invalidClaimSet); - return signedJWT.serialize(); - } - - /** - * This method is used to get an invalid JWT without the response type. - * - * @return String - * @throws Exception if an error occurs - */ - public static String getInvalidJWTWithoutResponseType() throws Exception { - - JWTClaimsSet.Builder builder = getValidJWTClaimsSetBuilder(); - JWTClaimsSet invalidClaimSet = builder - .claim(TestJwtBuilderConstants.RESPONSE_TYPE, null) - .build(); - SignedJWT signedJWT = getSignedJWT(JWSAlgorithm.PS256, invalidClaimSet); - return signedJWT.serialize(); - } - - /** - * This method is used to get an invalid JWT without nbf claim. - * - * @return String - * @throws Exception if an error occurs - */ - public static String getInvalidJWTWithoutNbfClaim() throws Exception { - JWTClaimsSet.Builder builder = getValidJWTClaimsSetBuilder(); - JWTClaimsSet invalidClaimSet = builder - .notBeforeTime(null) - .build(); - SignedJWT signedJWT = getSignedJWT(JWSAlgorithm.PS256, invalidClaimSet); - return signedJWT.serialize(); - } - - /** - * This method is used to get an invalid JWT with nbf claim over 60 minutes in the past. - * - * @return String - * @throws Exception if an error occurs - */ - public static String getInvalidJWTWithNbfClaimOver60Min() throws Exception { - JWTClaimsSet.Builder builder = getValidJWTClaimsSetBuilder(); - JWTClaimsSet invalidClaimSet = builder - .notBeforeTime(Date.from(Instant.now().minusSeconds(notBeforePeriod + 1500))) - .build(); - SignedJWT signedJWT = getSignedJWT(JWSAlgorithm.PS256, invalidClaimSet); - return signedJWT.serialize(); - } - - /** - * This method is used to get a valid JWTClaimsSet.Builder Object. - * - * @return JWTClaimsSet.Builder - */ - private static JWTClaimsSet.Builder getValidJWTClaimsSetBuilder() { - Map claimsMap = getValidClaimsMap(); - - return new JWTClaimsSet.Builder() - .issuer(issuer) - .audience(audience) - .expirationTime(Date.from(Instant.now().plusSeconds(expiryPeriod))) - .notBeforeTime(Date.from(Instant.now())) - .claim(TestJwtBuilderConstants.RESPONSE_TYPE, responseType) - .claim(TestJwtBuilderConstants.CODE_CHALLENGE_METHOD, codeChallengeMethod) - .claim(TestJwtBuilderConstants.NONCE, nonce) - .claim(TestJwtBuilderConstants.CLIENT_ID, clientId) - .claim(TestJwtBuilderConstants.REDIRECT_URI, redirectUri) - .claim(TestJwtBuilderConstants.SCOPE, scope) - .claim(TestJwtBuilderConstants.STATE, state) - .claim(TestJwtBuilderConstants.CLAIMS, claimsMap) - .claim(TestJwtBuilderConstants.CODE_CHALLENGE, codeChallenge); - } - - - /** - * This method is used to get a valid claims value as a map object. - * - * @return Map - */ - private static Map getValidClaimsMap() { - // Create a new HashMap object to represent the claims - Map claimsMap = new HashMap<>(); - - // Add the sharing_duration key-value pair to the claims map - claimsMap.put(TestJwtBuilderConstants.SHARING_DURATION, sharingDuration); - - // Create a new HashMap object to represent the id_token object - Map idToken = new HashMap<>(); - - // Create a new HashMap object to represent the acr object - Map acr = new HashMap<>(); - acr.put(TestJwtBuilderConstants.ACR_VALUE, acrValue); - acr.put(TestJwtBuilderConstants.ACR_ESSENTIAL, acrEssential); - - // Add the acr map to the id_token map - idToken.put(TestJwtBuilderConstants.ACR, acr); - - // Add the id_token map to the claims map - claimsMap.put(TestJwtBuilderConstants.ID_TOKEN, idToken); - return claimsMap; - } - - /** - * This method is used to get a valid JWT signed with supported signature algorithm. - * ex: PS256 - * - * @param jwsAlgorithm JWSAlgorithm Value - * @param claimsSet JWTClaimsSet Object - * @return SignedJWT - * @throws Exception if an error occurs - */ - private static SignedJWT getSignedJWT(JWSAlgorithm jwsAlgorithm , JWTClaimsSet claimsSet) throws Exception { - RSAPrivateKey privateKey = getPrivateKeyFromKeyStore(); - JWSHeader header = new JWSHeader.Builder(jwsAlgorithm) - .keyID(kid) - .type(JOSEObjectType.JWT) - .build(); - SignedJWT signedJWT = new SignedJWT(header, claimsSet); - JWSSigner signer = new RSASSASigner(privateKey); - signer.getJCAContext().setProvider(BouncyCastleProviderSingleton.getInstance()); - signedJWT.sign(signer); - return signedJWT; - } - - /** - * This method is used to get a valid JWT encrypted with supported encryption algorithm and encryption method. - * ex: RSA-OAEP-256 - * - * @param jweAlgorithm JWEAlgorithm Value - * @param encryptionMethod EncryptionMethod Value - * @param signedJWT SignedJWT Object - * @return JWEObject - * @throws Exception if an error occurs - */ - private static JWEObject getEncryptedJWT(JWEAlgorithm jweAlgorithm , - EncryptionMethod encryptionMethod, SignedJWT signedJWT) throws Exception { - RSAPublicKey publicKey = getPublicKeyFromKeyStore(); - JWEHeader jweHeader = new JWEHeader.Builder(jweAlgorithm, encryptionMethod) - .build(); - JWEObject jweObject = new JWEObject(jweHeader, new Payload(signedJWT)); - JWEEncrypter encrypter = new RSAEncrypter(publicKey); - encrypter.getJCAContext().setProvider(BouncyCastleProviderSingleton.getInstance()); - jweObject.encrypt(encrypter); - return jweObject; - } - - /** - * This method is used to get the private key from the keystore. - * - * @return RSAPrivateKey - * @throws Exception if an error occurs - */ - private static RSAPrivateKey getPrivateKeyFromKeyStore() throws Exception { - KeyStore keyStore = getKeyStore(); - KeyStore.PasswordProtection keyProtection = new KeyStore.PasswordProtection(keyPassword.toCharArray()); - KeyStore.PrivateKeyEntry privateKeyEntry = (KeyStore.PrivateKeyEntry) keyStore.getEntry(alias, keyProtection); - return (RSAPrivateKey) privateKeyEntry.getPrivateKey(); - } - - /** - * This method is used to get the public key from the keystore. - * - * @return RSAPublicKey - * @throws Exception if an error occurs - */ - private static RSAPublicKey getPublicKeyFromKeyStore() throws Exception { - KeyStore keyStore = getKeyStore(); - KeyStore.PasswordProtection keyProtection = new KeyStore.PasswordProtection(keyPassword.toCharArray()); - KeyStore.PrivateKeyEntry privateKeyEntry = (KeyStore.PrivateKeyEntry) keyStore.getEntry(alias, keyProtection); - java.security.cert.Certificate[] certChain = privateKeyEntry.getCertificateChain(); - Certificate cert = certChain[0]; - return (RSAPublicKey) cert.getPublicKey(); - } - - /** - * This method is used to get the keystore. - * - * @return KeyStore - * @throws Exception if an error occurs - */ - private static KeyStore getKeyStore() throws Exception { - File file = new File(keyStorePath); - String absolutePathForTestResources = file.getAbsolutePath(); - String absolutePathForKeyStore = absolutePathForTestResources + "/wso2carbon.jks"; - String[] pathParts = absolutePathForKeyStore.split("/"); - String platformAbsolutePathForKeyStore = String.join(File.separator, pathParts); - InputStream keystoreFile = new FileInputStream(platformAbsolutePathForKeyStore); - KeyStore keyStore = KeyStore.getInstance(KeyStore.getDefaultType()); - keyStore.load(keystoreFile, keyPassword.toCharArray()); - keystoreFile.close(); - return keyStore; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/util/test/jwt/builder/constants/TestJwtBuilderConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/util/test/jwt/builder/constants/TestJwtBuilderConstants.java deleted file mode 100644 index b50c268e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/push/auth/extension/request/validator/util/test/jwt/builder/constants/TestJwtBuilderConstants.java +++ /dev/null @@ -1,40 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.util.test.jwt.builder.constants; - -/** - * Constant class for TestJwtBuilder Class. - */ -public class TestJwtBuilderConstants { - public static final String RESPONSE_TYPE = "response_type"; - public static final String NONCE = "nonce"; - public static final String SCOPE = "scope"; - public static final String CLIENT_ID = "client_id"; - public static final String REDIRECT_URI = "redirect_uri"; - public static final String STATE = "state"; - public static final String CLAIMS = "claims"; - public static final String CODE_CHALLENGE = "code_challenge"; - public static final String CODE_CHALLENGE_METHOD = "code_challenge_method"; - public static final String SHARING_DURATION = "sharing_duration"; - public static final String ACR = "acr"; - public static final String ACR_VALUE = "value"; - public static final String ACR_ESSENTIAL = "essential"; - public static final String ID_TOKEN = "id_token"; - public static final String REQUEST = "request"; - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/TokenFilterTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/TokenFilterTest.java deleted file mode 100644 index 07f54e59..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/TokenFilterTest.java +++ /dev/null @@ -1,348 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.token; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.token.util.TestConstants; -import com.wso2.openbanking.accelerator.identity.token.util.TestUtil; -import com.wso2.openbanking.accelerator.identity.token.validators.ClientAuthenticatorValidator; -import com.wso2.openbanking.accelerator.identity.token.validators.MTLSEnforcementValidator; -import com.wso2.openbanking.accelerator.identity.token.validators.OBIdentityFilterValidator; -import com.wso2.openbanking.accelerator.identity.token.validators.SignatureAlgorithmEnforcementValidator; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.http.HttpStatus; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.springframework.mock.web.MockHttpServletRequest; -import org.springframework.mock.web.MockHttpServletResponse; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -import java.io.IOException; -import java.nio.charset.StandardCharsets; -import java.util.ArrayList; -import java.util.Base64; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import javax.servlet.FilterChain; -import javax.servlet.ServletException; - -import static org.testng.Assert.assertEquals; - -/** - * Test for token filter. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({IdentityCommonUtil.class}) -public class TokenFilterTest extends PowerMockTestCase { - - MockHttpServletRequest request; - MockHttpServletResponse response; - FilterChain filterChain; - - @BeforeMethod - public void beforeMethod() { - - request = new MockHttpServletRequest(); - response = new MockHttpServletResponse(); - filterChain = Mockito.spy(FilterChain.class); - } - - @Test(description = "Test the validators are omitted in non regulatory scenario") - public void nonRegulatoryTest() throws Exception { - - Map configMap = new HashMap<>(); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - request.setParameter(IdentityCommonConstants.CLIENT_ID, "test"); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(false); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - filter.doFilter(request, response, filterChain); - assertEquals(response.getStatus(), HttpStatus.SC_OK); - } - - @Test(description = "Test client id extraction from header in non-regulatory app scenario") - public void nonRegulatoryAppWithAuthorizationHeaderTest() throws Exception { - - Map configMap = new HashMap<>(); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - String credentials = "client_id" + ':' + "client_secret"; - String authToken = Base64.getEncoder().encodeToString(credentials.getBytes(StandardCharsets.UTF_8)); - request.addHeader("Authorization", "Basic " + authToken); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("client_id")).thenReturn(false); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - filter.doFilter(request, response, filterChain); - assertEquals(response.getStatus(), HttpStatus.SC_OK); - } - - @Test(description = "Test the certificate in context/header is mandated") - public void noCertificateTest() throws IOException, OpenBankingException, ServletException { - - Map configMap = new HashMap<>(); - PowerMockito.mockStatic(IdentityCommonUtil.class); - configMap.put(IdentityCommonConstants.ENABLE_TRANSPORT_CERT_AS_HEADER, true); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - request.setParameter(IdentityCommonConstants.CLIENT_ID, "test"); - - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()) - .thenReturn(IdentityCommonConstants.CERTIFICATE_HEADER); - filter.doFilter(request, response, filterChain); - Map responseMap = TestUtil.getResponse(response.getOutputStream()); - assertEquals(response.getStatus(), HttpStatus.SC_BAD_REQUEST); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR), "invalid_request"); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION), - "Transport certificate not found in the request"); - } - - @Test(description = "Test the certificate in attribute is present if config is disabled") - public void certificateIsNotPresentInAttributeTest() throws IOException, OpenBankingException, ServletException { - - Map configMap = new HashMap<>(); - PowerMockito.mockStatic(IdentityCommonUtil.class); - configMap.put(IdentityCommonConstants.ENABLE_TRANSPORT_CERT_AS_HEADER, false); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - request.setParameter(IdentityCommonConstants.CLIENT_ID, "test"); - request.addHeader(TestConstants.CERTIFICATE_HEADER, TestConstants.CERTIFICATE_CONTENT); - - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - filter.doFilter(request, response, filterChain); - Map responseMap = TestUtil.getResponse(response.getOutputStream()); - assertEquals(response.getStatus(), HttpStatus.SC_BAD_REQUEST); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR), "invalid_request"); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION), - "Transport certificate not found in the request"); - - } - - @Test(description = "Test the certificate in the attribute is overridden in header") - public void certificateInAttributeOverriddenTest() throws IOException, OpenBankingException, ServletException { - - TokenFilter filter = Mockito.spy(TokenFilter.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - request.setParameter(IdentityCommonConstants.CLIENT_ID, "test"); - request.addHeader(TestConstants.CERTIFICATE_HEADER, "invalid"); - request.setAttribute(IdentityCommonConstants.JAVAX_SERVLET_REQUEST_CERTIFICATE, - TestUtil.getCertificate(TestConstants.CERTIFICATE_CONTENT)); - - Map configMap = new HashMap<>(); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - filter.doFilter(request, response, filterChain); - assertEquals(response.getStatus(), HttpStatus.SC_OK); - } - - @Test(description = "Test the validators are omitted if nothing is configured") - public void noValidatorsConfiguredTest() throws Exception { - - Map configMap = new HashMap<>(); - configMap.put(IdentityCommonConstants.ENABLE_TRANSPORT_CERT_AS_HEADER, true); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - request.setParameter(IdentityCommonConstants.CLIENT_ID, "test"); - request.addHeader(TestConstants.CERTIFICATE_HEADER, TestConstants.CERTIFICATE_CONTENT); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - filter.doFilter(request, response, filterChain); - assertEquals(response.getStatus(), HttpStatus.SC_OK); - } - - @Test(description = "Test the client ID is enforced") - public void clientIdEnforcementTest() throws Exception { - - TokenFilter filter = Mockito.spy(TokenFilter.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - filter.doFilter(request, response, filterChain); - - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - Map responseMap = TestUtil.getResponse(response.getOutputStream()); - assertEquals(response.getStatus(), HttpStatus.SC_BAD_REQUEST); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR), "invalid_request"); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION), - "Unable to find client id in the request"); - } - - @Test(description = "Test client auth and signature enforcement validators engaged") - public void clientAuthSignatureEnforcementValidatorTest() throws Exception { - - TokenFilter filter = Mockito.spy(TokenFilter.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - ClientAuthenticatorValidator clientAuthValidator = Mockito.spy(ClientAuthenticatorValidator.class); - SignatureAlgorithmEnforcementValidator signatureAlgorithmValidator = - Mockito.spy(SignatureAlgorithmEnforcementValidator.class); - request.setParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION_TYPE, - IdentityCommonConstants.OAUTH_JWT_BEARER_GRANT_TYPE); - request.setParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION, TestConstants.CLIENT_ASSERTION); - request.setAttribute(IdentityCommonConstants.JAVAX_SERVLET_REQUEST_CERTIFICATE, - TestUtil.getCertificate(TestConstants.CERTIFICATE_CONTENT)); - - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - Mockito.doReturn("private_key_jwt").when(clientAuthValidator) - .retrieveRegisteredAuthMethod("iYpRm64b2vmvmKDhdL6KZD9z6fca"); - Mockito.doReturn("PS256").when(signatureAlgorithmValidator) - .getRegisteredSigningAlgorithm("iYpRm64b2vmvmKDhdL6KZD9z6fca"); - Mockito.doReturn("PS256").when(signatureAlgorithmValidator) - .getRequestSigningAlgorithm(TestConstants.CLIENT_ASSERTION); - - List validators = new ArrayList<>(); - validators.add(clientAuthValidator); - validators.add(signatureAlgorithmValidator); - - Mockito.doReturn(validators).when(filter).getValidators(); - filter.doFilter(request, response, filterChain); - assertEquals(response.getStatus(), HttpStatus.SC_OK); - } - - @Test(description = "Test client auth and mtls enforcement validators engaged") - public void clientAuthMTLSEnforcementValidatorTest() throws Exception { - - TokenFilter filter = Mockito.spy(TokenFilter.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - ClientAuthenticatorValidator clientAuthValidator = Mockito.spy(ClientAuthenticatorValidator.class); - MTLSEnforcementValidator mtlsEnforcementValidator = - Mockito.spy(MTLSEnforcementValidator.class); - - request.setParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION_TYPE, - IdentityCommonConstants.OAUTH_JWT_BEARER_GRANT_TYPE); - request.setParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION, TestConstants.CLIENT_ASSERTION); - request.addHeader(TestConstants.CERTIFICATE_HEADER, TestConstants.CERTIFICATE_CONTENT); - - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - Mockito.doReturn("private_key_jwt").when(clientAuthValidator) - .retrieveRegisteredAuthMethod("iYpRm64b2vmvmKDhdL6KZD9z6fca"); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - - List validators = new ArrayList<>(); - validators.add(clientAuthValidator); - validators.add(mtlsEnforcementValidator); - - Mockito.doReturn(validators).when(filter).getValidators(); - filter.doFilter(request, response, filterChain); - assertEquals(response.getStatus(), HttpStatus.SC_OK); - } - - @Test(description = "Test mtls and signature enforcement validators engaged") - public void signatureMTLSeEnforcementValidatorTest() throws Exception { - - TokenFilter filter = Mockito.spy(TokenFilter.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - MTLSEnforcementValidator mtlsEnforcementValidator = - Mockito.spy(MTLSEnforcementValidator.class); - SignatureAlgorithmEnforcementValidator signatureAlgorithmValidator = - Mockito.spy(SignatureAlgorithmEnforcementValidator.class); - - request.setParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION_TYPE, - IdentityCommonConstants.OAUTH_JWT_BEARER_GRANT_TYPE); - request.setParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION, TestConstants.CLIENT_ASSERTION); - request.addHeader(TestConstants.CERTIFICATE_HEADER, TestConstants.CERTIFICATE_CONTENT); - - Map configMap = new HashMap<>(); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - Mockito.doReturn("PS256").when(signatureAlgorithmValidator) - .getRegisteredSigningAlgorithm("iYpRm64b2vmvmKDhdL6KZD9z6fca"); - Mockito.doReturn("PS256").when(signatureAlgorithmValidator) - .getRequestSigningAlgorithm(TestConstants.CLIENT_ASSERTION); - - List validators = new ArrayList<>(); - validators.add(mtlsEnforcementValidator); - validators.add(signatureAlgorithmValidator); - - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - - Mockito.doReturn(validators).when(filter).getValidators(); - filter.doFilter(request, response, filterChain); - assertEquals(response.getStatus(), HttpStatus.SC_OK); - } - - @Test(description = "Test client auth, signature and mtls enforcement validators engaged") - public void allValidatorTest() throws Exception { - - SignatureAlgorithmEnforcementValidator signatureValidator = - Mockito.spy(SignatureAlgorithmEnforcementValidator.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - MTLSEnforcementValidator mtlsEnforcementValidator = Mockito.spy(MTLSEnforcementValidator.class); - ClientAuthenticatorValidator clientAuthenticatorValidator = Mockito.spy(ClientAuthenticatorValidator.class); - - request.addHeader(TestConstants.CERTIFICATE_HEADER, TestConstants.CERTIFICATE_CONTENT); - request.setParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION, TestConstants.CLIENT_ASSERTION); - request.setParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION_TYPE, - IdentityCommonConstants.OAUTH_JWT_BEARER_GRANT_TYPE); - - Mockito.doReturn("private_key_jwt").when(clientAuthenticatorValidator) - .retrieveRegisteredAuthMethod("iYpRm64b2vmvmKDhdL6KZD9z6fca"); - Mockito.doReturn("PS256").when(signatureValidator) - .getRegisteredSigningAlgorithm("iYpRm64b2vmvmKDhdL6KZD9z6fca"); - Mockito.doReturn("PS256").when(signatureValidator) - .getRequestSigningAlgorithm(TestConstants.CLIENT_ASSERTION); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - - List validators = new ArrayList<>(); - validators.add(mtlsEnforcementValidator); - validators.add(signatureValidator); - validators.add(clientAuthenticatorValidator); - - Map configMap = new HashMap<>(); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - Mockito.doReturn(validators).when(filter).getValidators(); - filter.doFilter(request, response, filterChain); - assertEquals(response.getStatus(), HttpStatus.SC_OK); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/util/TestConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/util/TestConstants.java deleted file mode 100644 index 971c2158..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/util/TestConstants.java +++ /dev/null @@ -1,104 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.token.util; - -/** - * Test constants. - */ -public class TestConstants { - public static final String TARGET_STREAM = "targetStream"; - public static final String CERTIFICATE_HEADER = "x-wso2-mutual-auth-cert"; - public static final String EXPIRED_CERTIFICATE_CONTENT = "-----BEGIN CERTIFICATE-----" + - "MIIFODCCBCCgAwIBAgIEWcWGxDANBgkqhkiG9w0BAQsFADBTMQswCQYDVQQGEwJH" + - "QjEUMBIGA1UEChMLT3BlbkJhbmtpbmcxLjAsBgNVBAMTJU9wZW5CYW5raW5nIFBy" + - "ZS1Qcm9kdWN0aW9uIElzc3VpbmcgQ0EwHhcNMTkwNTE2MDg0NDQ2WhcNMjAwNjE2" + - "MDkxNDQ2WjBhMQswCQYDVQQGEwJHQjEUMBIGA1UEChMLT3BlbkJhbmtpbmcxGzAZ" + - "BgNVBAsTEjAwMTU4MDAwMDFIUVFyWkFBWDEfMB0GA1UEAxMWc0Zna2k3Mk9pcXda" + - "TkZPWmc2T2FqaTCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEBANoVwx4E" + - "iWnQs89lj8vKSy/xTbZU2AHS9tFNz7wVa+rkpFyLVPtQW8AthG4hlfrBYMne7/P9" + - "c1Fi/q+n7eomWvJJo44GV44GJhegM6yyRaIcQdpxe9x9G4twWK4cY+VU3TfE6Dbd" + - "DdmAt7ai4KFbbpB33N8RwXoeGZdwxZFNPmfaoZZbz5p9+aSMQf1UyExcdlPXah77" + - "PDZDwAnyy5kYXUPS59S78+p4twqZXyZu9hd+Su5Zod5UObRJ4F5LQzZPS1+KzBje" + - "JM0o8qoRRZTZkLNnmmQw503KXp/LCLrSbFU2ZLGy3bQpKFFc5I6tZiy67ELNzLWo" + - "DzngEbApwhX+jtsCAwEAAaOCAgQwggIAMA4GA1UdDwEB/wQEAwIHgDAgBgNVHSUB" + - "Af8EFjAUBggrBgEFBQcDAQYIKwYBBQUHAwIwgeAGA1UdIASB2DCB1TCB0gYLKwYB" + - "BAGodYEGAWQwgcIwKgYIKwYBBQUHAgEWHmh0dHA6Ly9vYi50cnVzdGlzLmNvbS9w" + - "b2xpY2llczCBkwYIKwYBBQUHAgIwgYYMgYNVc2Ugb2YgdGhpcyBDZXJ0aWZpY2F0" + - "ZSBjb25zdGl0dXRlcyBhY2NlcHRhbmNlIG9mIHRoZSBPcGVuQmFua2luZyBSb290" + - "IENBIENlcnRpZmljYXRpb24gUG9saWNpZXMgYW5kIENlcnRpZmljYXRlIFByYWN0" + - "aWNlIFN0YXRlbWVudDBtBggrBgEFBQcBAQRhMF8wJgYIKwYBBQUHMAGGGmh0dHA6" + - "Ly9vYi50cnVzdGlzLmNvbS9vY3NwMDUGCCsGAQUFBzAChilodHRwOi8vb2IudHJ1" + - "c3Rpcy5jb20vb2JfcHBfaXNzdWluZ2NhLmNydDA6BgNVHR8EMzAxMC+gLaArhilo" + - "dHRwOi8vb2IudHJ1c3Rpcy5jb20vb2JfcHBfaXNzdWluZ2NhLmNybDAfBgNVHSME" + - "GDAWgBRQc5HGIXLTd/T+ABIGgVx5eW4/UDAdBgNVHQ4EFgQU5eqvEZ6ZdQS5bq/X" + - "dzP5XY/fUXUwDQYJKoZIhvcNAQELBQADggEBAIg8bd/bIh241ewS79lXU058VjCu" + - "JC+4QtcI2XiGV3dBpg10V6Kb6E/h8Gru04uVZW1JK52ivVb5NYs6r8txRsTBIaA8" + - "Cr03LJqEftclL9NbkPZnpEkUfqCBfujNQF8XWaQgXIIA+io1UzV1TG3K9XCa/w2S" + - "sTANKfF8qK5kRsy6z9OGPUE+Oi3DUt+E9p5LCq6n5Bkp9YRGmyYRPs8JMkJmq3sf" + - "wtXOy27LE4exJRuZsF1CA78ObaRytuE3DJcnIRdhOcjWieS/MxZD7bzuuAPu5ySX" + - "i2/qxT3AlWtHtxrz0mKSC3rlgYAHCzCAHoASWKpf5tnB3TodPVZ6DYOu7oI=" + - "-----END CERTIFICATE-----"; - - public static final String CERTIFICATE_CONTENT = "-----BEGIN CERTIFICATE-----\n" + - "MIIFODCCBCCgAwIBAgIEWcbiiTANBgkqhkiG9w0BAQsFADBTMQswCQYDVQQGEwJH\n" + - "QjEUMBIGA1UEChMLT3BlbkJhbmtpbmcxLjAsBgNVBAMTJU9wZW5CYW5raW5nIFBy\n" + - "ZS1Qcm9kdWN0aW9uIElzc3VpbmcgQ0EwHhcNMjMxMTE1MDUxMDMxWhcNMjQxMjE1\n" + - "MDU0MDMxWjBhMQswCQYDVQQGEwJHQjEUMBIGA1UEChMLT3BlbkJhbmtpbmcxGzAZ\n" + - "BgNVBAsTEjAwMTU4MDAwMDFIUVFyWkFBWDEfMB0GA1UEAxMWakZRdVE0ZVFiTkNN\n" + - "U3FkQ29nMjFuRjCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEBAJslGjTm\n" + - "0tWwnnKgC7WNqUSYNxblURkJyoD5UuSmzpsM5nlUBAxYxBgztTo062LJELzUTzA/\n" + - "9kgLIMMgj+wG1OS475QCgeyoDmwf0SPuFRBl0G0AjxAvJzzs2aijzxiYRbKUa4gm\n" + - "O1KPU3Xlz89mi8lwjTZlxtGk3ABwBG4f5na5TY7uZMlgWPXDnTg7Cc1H4mrMbEFk\n" + - "UaXmb6ZhhGtp0JL04+4Lp16QWrgiHrlop+P8bd+pwmmOmLuglTIEh+v993j+7v8B\n" + - "XYqdmYQ3noiOhK9ynFPD1A7urrm71Pgkuq+Wk5HCvMiBK7zZ4Sn9FDovykDKZTFY\n" + - "MloVDXLhmfDQrmcCAwEAAaOCAgQwggIAMA4GA1UdDwEB/wQEAwIHgDAgBgNVHSUB\n" + - "Af8EFjAUBggrBgEFBQcDAQYIKwYBBQUHAwIwgeAGA1UdIASB2DCB1TCB0gYLKwYB\n" + - "BAGodYEGAWQwgcIwKgYIKwYBBQUHAgEWHmh0dHA6Ly9vYi50cnVzdGlzLmNvbS9w\n" + - "b2xpY2llczCBkwYIKwYBBQUHAgIwgYYMgYNVc2Ugb2YgdGhpcyBDZXJ0aWZpY2F0\n" + - "ZSBjb25zdGl0dXRlcyBhY2NlcHRhbmNlIG9mIHRoZSBPcGVuQmFua2luZyBSb290\n" + - "IENBIENlcnRpZmljYXRpb24gUG9saWNpZXMgYW5kIENlcnRpZmljYXRlIFByYWN0\n" + - "aWNlIFN0YXRlbWVudDBtBggrBgEFBQcBAQRhMF8wJgYIKwYBBQUHMAGGGmh0dHA6\n" + - "Ly9vYi50cnVzdGlzLmNvbS9vY3NwMDUGCCsGAQUFBzAChilodHRwOi8vb2IudHJ1\n" + - "c3Rpcy5jb20vb2JfcHBfaXNzdWluZ2NhLmNydDA6BgNVHR8EMzAxMC+gLaArhilo\n" + - "dHRwOi8vb2IudHJ1c3Rpcy5jb20vb2JfcHBfaXNzdWluZ2NhLmNybDAfBgNVHSME\n" + - "GDAWgBRQc5HGIXLTd/T+ABIGgVx5eW4/UDAdBgNVHQ4EFgQU7T6cMtCSQTT5JWW3\n" + - "O6vifRUSdpkwDQYJKoZIhvcNAQELBQADggEBAE9jrd/AE65vy3SEWdmFKPS4su7u\n" + - "EHy+KH18PETV6jMF2UFIJAOx7jl+5a3O66NkcpxFPeyvSuH+6tAAr2ZjpoQwtW9t\n" + - "Z9k2KSOdNOiJeQgjavwQC6t/BHI3yXWOIQm445BUN1cV9pagcRJjRyL3SPdHVoRf\n" + - "IbF7VI/+ULHwWdZYPXxtwUoda1mQFf6a+2lO4ziUHb3U8iD90FBURzID7WJ1ODSe\n" + - "B5zE/hG9Sxd9wlSXvl1oNmc/ha5oG/7rJpRqrx5Dcq3LEoX9iZZ3knHLkCm/abIQ\n" + - "7Nff8GQytuGhnGZxmGFYKDXdKElcl9dAlZ3bIK2I+I6jD2z2XvSfrhFyRjU=\n" + - "-----END CERTIFICATE-----"; - public static final String CLIENT_ASSERTION = "eyJraWQiOiJqeVJVY3l0MWtWQ2xjSXZsVWxjRHVrVlozdFUiLCJhbGciOiJQUzI1" + - "NiJ9.eyJzdWIiOiJpWXBSbTY0YjJ2bXZtS0RoZEw2S1pEOXo2ZmNhIiwiYXVkIjoiaHR0cHM6Ly9sb2NhbGhvc3Q6OTQ0My9vYXV0a" + - "DIvdG9rZW4iLCJpc3MiOiJpWXBSbTY0YjJ2bXZtS0RoZEw2S1pEOXo2ZmNhIiwiZXhwIjoxNjEwNjMxNDEyLCJpYXQiOjE2MTA2MDE" + - "0MTIsImp0aSI6IjE2MTA2MDE0MTI5MDAifQ.tmMTlCL-VABhFTA6QQ6UPvUydKuzynidepAa8oZGEBfVyAsiW5IF01NKYD0ynpXXJC" + - "Q6hcbWK0FEGity67p6DeI9LT-xAnaKwZY7H8rbuxWye2vhanM0jVa1vggsmwWYyOR4k55ety9lP1MkcGZpaK48qoaqsX_X7GCSGXzq" + - "BncTEPYfCpVUQtS4ctwoCl06TFbY2Lfm9E24z1rfmU9xPc7au6LpKRLMMHQ8QXuc-FhnWdgEFv_3tAai2ovVmrqEfwj6Z6Ew5bFeI9" + - "jtCR4TSol47hzDwldx5rH7m2OPUx66yEtGrM7UU62fC-4nxplZ69fjlHN4KQ62PxEaCQs0_A"; - - public static final String CLIENT_ASSERTION_NO_HEADER = - "eyJraWQiOiJqeVJVY3l0MWtWQ2xjSXZsVWxjRHVrVlozdFUiLCJhbGciOiJQUzI1" + - "NiJ.eyJzdWIiOiJpWXBSbTY0YjJ2bXZtS0RoZEw2S1pEOXo2ZmNhIiwiYXVkIjoiaHR0cHM6Ly9sb2NhbGhvc3Q6OTQ0My9vYXV0a" + - "DIvdG9rZW4iLCJpc3MiOiJpWXBSbTY0YjJ2bXZtS0RoZEw2S1pEOXo2ZmNhIiwiZXhwIjoxNjEwNjMxNDEyLCJpYXQiOjE2MTA2MDE" + - "0MTIsImp0aSI6IjE2MTA2MDE0MTI5MDAifQ.tmMTlCL-VABhFTA6QQ6UPvUydKuzynidepAa8oZGEBfVyAsiW5IF01NKYD0ynpXXJC" + - "Q6hcbWK0FEGity67p6DeI9LT-xAnaKwZY7H8rbuxWye2vhanM0jVa1vggsmwWYyOR4k55ety9lP1MkcGZpaK48qoaqsX_X7GCSGXzq" + - "BncTEPYfCpVUQtS4ctwoCl06TFbY2Lfm9E24z1rfmU9xPc7au6LpKRLMMHQ8QXuc-FhnWdgEFv_3tAai2ovVmrqEfwj6Z6Ew5bFeI9" + - "jtCR4TSol47hzDwldx5rH7m2OPUx66yEtGrM7UU62fC-4nxplZ69fjlHN4KQ62PxEaCQs0_A"; -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/util/TestUtil.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/util/TestUtil.java deleted file mode 100644 index f5d86e7c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/util/TestUtil.java +++ /dev/null @@ -1,62 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.token.util; - -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import org.apache.commons.lang.StringUtils; -import org.json.JSONObject; -import org.wso2.carbon.identity.core.util.IdentityUtil; - -import java.security.cert.CertificateException; -import java.security.cert.X509Certificate; -import java.util.HashMap; -import java.util.Map; - -import javax.servlet.ServletOutputStream; - -/** - * Test utils. - */ -public class TestUtil { - - public static Map getResponse(ServletOutputStream outputStream) { - - Map response = new HashMap<>(); - JSONObject outputStreamMap = new JSONObject(outputStream); - JSONObject targetStream = new JSONObject(outputStreamMap.get(TestConstants.TARGET_STREAM).toString()); - response.put(IdentityCommonConstants.OAUTH_ERROR, - targetStream.get(IdentityCommonConstants.OAUTH_ERROR).toString()); - response.put(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION, - targetStream.get(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION).toString()); - return response; - } - - public static X509Certificate getCertificate(String certificateContent) { - - if (StringUtils.isNotBlank(certificateContent)) { - // Build the Certificate object from cert content. - try { - return (X509Certificate) IdentityUtil.convertPEMEncodedContentToCertificate(certificateContent); - } catch (CertificateException e) { - //do nothing - } - } - return null; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/validators/ClientAuthenticatorValidatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/validators/ClientAuthenticatorValidatorTest.java deleted file mode 100644 index 09bc947d..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/validators/ClientAuthenticatorValidatorTest.java +++ /dev/null @@ -1,217 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.token.validators; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.token.DefaultTokenFilter; -import com.wso2.openbanking.accelerator.identity.token.TokenFilter; -import com.wso2.openbanking.accelerator.identity.token.util.TestConstants; -import com.wso2.openbanking.accelerator.identity.token.util.TestUtil; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.http.HttpStatus; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.springframework.mock.web.MockHttpServletRequest; -import org.springframework.mock.web.MockHttpServletResponse; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -import java.io.IOException; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import javax.servlet.FilterChain; -import javax.servlet.ServletException; -import javax.servlet.http.HttpServletResponse; - -import static org.testng.Assert.assertEquals; - -/** - * Test for client authenticator validator. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({IdentityCommonUtil.class}) -public class ClientAuthenticatorValidatorTest extends PowerMockTestCase { - - MockHttpServletResponse response; - MockHttpServletRequest request; - FilterChain filterChain; - - @BeforeMethod - public void beforeMethod() throws ReflectiveOperationException, IOException { - - request = new MockHttpServletRequest(); - response = new MockHttpServletResponse(); - filterChain = Mockito.spy(FilterChain.class); - - } - - @Test(description = "Test whether authentication follows the registered client authentication method PKJWT") - public void privateKeyJWTValidatorTest() throws IOException, ServletException, OpenBankingException { - - ClientAuthenticatorValidator validator = Mockito.spy(ClientAuthenticatorValidator.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - request.setParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION, TestConstants.CLIENT_ASSERTION); - request.setParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION_TYPE, - IdentityCommonConstants.OAUTH_JWT_BEARER_GRANT_TYPE); - request.addHeader(TestConstants.CERTIFICATE_HEADER, TestConstants.CERTIFICATE_CONTENT); - Mockito.doReturn("private_key_jwt").when(validator) - .retrieveRegisteredAuthMethod("iYpRm64b2vmvmKDhdL6KZD9z6fca"); - - List validators = new ArrayList<>(); - validators.add(validator); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - Mockito.doReturn(validators).when(filter).getValidators(); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - - Map configMap = new HashMap<>(); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - filter.doFilter(request, response, filterChain); - - assertEquals(response.getStatus(), HttpServletResponse.SC_OK); - } - - @Test(description = "Test fail scenario authentication follows the registered client authentication method PKJWT") - public void privateKeyJWTValidatorNegativeTest() throws IOException, ServletException, OpenBankingException { - - Map configMap = new HashMap<>(); - PowerMockito.mockStatic(IdentityCommonUtil.class); - configMap.put(IdentityCommonConstants.ENABLE_TRANSPORT_CERT_AS_HEADER, true); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - PowerMockito.mockStatic(IdentityCommonUtil.class); - ClientAuthenticatorValidator validator = Mockito.spy(ClientAuthenticatorValidator.class); - - request.setParameter("client_assertion", TestConstants.CLIENT_ASSERTION); - request.addHeader(TestConstants.CERTIFICATE_HEADER, TestConstants.CERTIFICATE_CONTENT); - Mockito.doReturn("private_key_jwt").when(validator) - .retrieveRegisteredAuthMethod("iYpRm64b2vmvmKDhdL6KZD9z6fca"); - - List validators = new ArrayList<>(); - validators.add(validator); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - Mockito.doReturn(validators).when(filter).getValidators(); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("iYpRm64b2vmvmKDhdL6KZD9z6fca")) - .thenReturn(true); - filter.doFilter(request, response, filterChain); - - Map responseMap = TestUtil.getResponse(response.getOutputStream()); - assertEquals(response.getStatus(), HttpStatus.SC_BAD_REQUEST); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR), IdentityCommonConstants - .OAUTH2_INVALID_REQUEST_MESSAGE); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION), - "Request does not follow the registered token endpoint auth method private_key_jwt"); - } - - @Test(description = "Test client ID enforcement in client authentication method mtls") - public void privateKeyClientIdEnforcementTest() throws IOException, ServletException, OpenBankingException { - - ClientAuthenticatorValidator validator = Mockito.spy(ClientAuthenticatorValidator.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - - request.setParameter("client_assertion_type", IdentityCommonConstants.OAUTH_JWT_BEARER_GRANT_TYPE); - Mockito.doReturn("private_key_jwt").when(validator) - .retrieveRegisteredAuthMethod("iYpRm64b2vmvmKDhdL6KZD9z6fca"); - - List validators = new ArrayList<>(); - validators.add(validator); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - Mockito.doReturn(validators).when(filter).getValidators(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - filter.doFilter(request, response, filterChain); - - Map responseMap = TestUtil.getResponse(response.getOutputStream()); - assertEquals(response.getStatus(), HttpStatus.SC_BAD_REQUEST); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR), IdentityCommonConstants - .OAUTH2_INVALID_REQUEST_MESSAGE); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION), - "Unable to find client id in the request"); - } - - @Test(description = "Test whether authentication follows the registered client authentication method mtls") - public void mtlsValidatorTest() throws IOException, ServletException, OpenBankingException { - - PowerMockito.mockStatic(IdentityCommonUtil.class); - ClientAuthenticatorValidator validator = Mockito.spy(ClientAuthenticatorValidator.class); - request.setParameter("client_id", "iYpRm64b2vmvmKDhdL6KZD9z6fca"); - request.addHeader(TestConstants.CERTIFICATE_HEADER, TestConstants.CERTIFICATE_CONTENT); - Mockito.doReturn("tls_client_auth").when(validator) - .retrieveRegisteredAuthMethod("iYpRm64b2vmvmKDhdL6KZD9z6fca"); - - List validators = new ArrayList<>(); - validators.add(validator); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - Mockito.doReturn(validators).when(filter).getValidators(); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - - Map configMap = new HashMap<>(); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - filter.doFilter(request, response, filterChain); - assertEquals(response.getStatus(), HttpServletResponse.SC_OK); - } - - @Test(description = "Test client ID enforcement in client authentication method mtls") - public void mtlsValidatorClientIdEnforcementTest() throws IOException, ServletException, OpenBankingException { - - ClientAuthenticatorValidator validator = Mockito.spy(ClientAuthenticatorValidator.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - request.addHeader(TestConstants.CERTIFICATE_HEADER, TestConstants.CERTIFICATE_CONTENT); - Mockito.doReturn("tls_client_auth").when(validator) - .retrieveRegisteredAuthMethod("iYpRm64b2vmvmKDhdL6KZD9z6fca"); - - List validators = new ArrayList<>(); - validators.add(validator); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - Mockito.doReturn(validators).when(filter).getValidators(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - filter.doFilter(request, response, filterChain); - - Map responseMap = TestUtil.getResponse(response.getOutputStream()); - assertEquals(response.getStatus(), HttpStatus.SC_BAD_REQUEST); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR), IdentityCommonConstants - .OAUTH2_INVALID_REQUEST_MESSAGE); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION), - "Unable to find client id in the request"); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/validators/MTLSCertificateValidatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/validators/MTLSCertificateValidatorTest.java deleted file mode 100644 index b1ec9332..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/validators/MTLSCertificateValidatorTest.java +++ /dev/null @@ -1,103 +0,0 @@ -/* - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). All Rights Reserved. - * - * This software is the property of WSO2 LLC. and its suppliers, if any. - * Dissemination of any information or reproduction of any material contained - * herein in any form is strictly forbidden, unless permitted by WSO2 expressly. - * You may not alter or remove any copyright or other notice from copies of this content. - */ - -package com.wso2.openbanking.accelerator.identity.token.validators; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.token.DefaultTokenFilter; -import com.wso2.openbanking.accelerator.identity.token.TokenFilter; -import com.wso2.openbanking.accelerator.identity.token.util.TestConstants; -import com.wso2.openbanking.accelerator.identity.token.util.TestUtil; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.http.HttpStatus; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.springframework.mock.web.MockHttpServletRequest; -import org.springframework.mock.web.MockHttpServletResponse; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -import java.io.IOException; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import javax.servlet.FilterChain; -import javax.servlet.ServletException; - -import static org.testng.Assert.assertEquals; - -/** - * class for MTLSCertificateValidator Test. - */ -@PrepareForTest({IdentityCommonUtil.class}) -@PowerMockIgnore({"jdk.internal.reflect.*"}) -public class MTLSCertificateValidatorTest extends PowerMockTestCase { - - MockHttpServletResponse response; - MockHttpServletRequest request; - FilterChain filterChain; - TokenFilter filter; - - @BeforeMethod - public void beforeMethod() throws ReflectiveOperationException, IOException, OpenBankingException { - - request = new MockHttpServletRequest(); - response = new MockHttpServletResponse(); - filterChain = Mockito.spy(FilterChain.class); - - List validators = new ArrayList<>(); - MTLSCertificateValidator mtlsCertificateValidator = Mockito.spy(MTLSCertificateValidator.class); - validators.add(mtlsCertificateValidator); - - filter = Mockito.spy(TokenFilter.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - Mockito.doReturn(validators).when(filter).getValidators(); - PowerMockito.mockStatic(IdentityCommonUtil.class); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - Map configMap = new HashMap<>(); - configMap.put(IdentityCommonConstants.ENABLE_TRANSPORT_CERT_AS_HEADER, true); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - request.setParameter(IdentityCommonConstants.CLIENT_ID, "test"); - request.setAttribute(IdentityCommonConstants.JAVAX_SERVLET_REQUEST_CERTIFICATE, null); - - } - - @Test(description = "Test whether the expired certificate fails") - public void testMTLSCertValidationWithExpiredCertificate() throws IOException, ServletException { - - request.addHeader(TestConstants.CERTIFICATE_HEADER, TestConstants.EXPIRED_CERTIFICATE_CONTENT); - - filter.doFilter(request, response, filterChain); - Map responseMap = TestUtil.getResponse(response.getOutputStream()); - assertEquals(response.getStatus(), HttpStatus.SC_UNAUTHORIZED); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR), "invalid_client"); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION), - "Invalid mutual TLS request. Client certificate is expired"); - - } - - @Test(description = "Test whether the expired certificate fails") - public void testMTLSCertValidationWithValidCertificate() throws IOException, ServletException { - - request.addHeader(TestConstants.CERTIFICATE_HEADER, TestConstants.CERTIFICATE_CONTENT); - - filter.doFilter(request, response, filterChain); - assertEquals(response.getStatus(), HttpStatus.SC_OK); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/validators/MTLSEnforcementValidatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/validators/MTLSEnforcementValidatorTest.java deleted file mode 100644 index f76ba2b8..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/validators/MTLSEnforcementValidatorTest.java +++ /dev/null @@ -1,251 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.token.validators; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.token.DefaultTokenFilter; -import com.wso2.openbanking.accelerator.identity.token.TokenFilter; -import com.wso2.openbanking.accelerator.identity.token.util.TestConstants; -import com.wso2.openbanking.accelerator.identity.token.util.TestUtil; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.http.HttpStatus; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.springframework.mock.web.MockHttpServletRequest; -import org.springframework.mock.web.MockHttpServletResponse; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -import java.io.IOException; -import java.security.cert.X509Certificate; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import javax.servlet.FilterChain; -import javax.servlet.ServletException; -import javax.servlet.http.HttpServletResponse; - -import static org.testng.Assert.assertEquals; - -/** - * Test for MTLS enforcement validator. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({IdentityCommonUtil.class}) -public class MTLSEnforcementValidatorTest extends PowerMockTestCase { - - MockHttpServletResponse response; - MockHttpServletRequest request; - FilterChain filterChain; - - @BeforeMethod - public void beforeMethod() throws ReflectiveOperationException, IOException { - - request = new MockHttpServletRequest(); - response = new MockHttpServletResponse(); - filterChain = Mockito.spy(FilterChain.class); - - } - - @Test(description = "Test the complete flow with certificate header configured") - public void certificateHeaderValidation() throws Exception { - Map configMap = new HashMap<>(); - PowerMockito.mockStatic(IdentityCommonUtil.class); - configMap.put(IdentityCommonConstants.ENABLE_TRANSPORT_CERT_AS_HEADER, true); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - MTLSEnforcementValidator mtlsEnforcementValidator = Mockito.spy(MTLSEnforcementValidator.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - request.setParameter(IdentityCommonConstants.CLIENT_ID, "test"); - request.addHeader(TestConstants.CERTIFICATE_HEADER, TestConstants.CERTIFICATE_CONTENT); - - List validators = new ArrayList<>(); - validators.add(mtlsEnforcementValidator); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - Mockito.doReturn(validators).when(filter).getValidators(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - filter.doFilter(request, response, filterChain); - - assertEquals(response.getStatus(), HttpServletResponse.SC_OK); - } - - @Test(description = "Test the complete flow with certificate passed as a attribute") - public void certificateAttributeValidation() throws Exception { - - PowerMockito.mockStatic(IdentityCommonUtil.class); - MTLSEnforcementValidator mtlsEnforcementValidator = Mockito.spy(MTLSEnforcementValidator.class); - IdentityCommonUtil util = Mockito.mock(IdentityCommonUtil.class); - - X509Certificate cert = TestUtil.getCertificate(TestConstants.CERTIFICATE_CONTENT); - - Map configMap = new HashMap<>(); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - request.setParameter(IdentityCommonConstants.CLIENT_ID, "test"); - request.setAttribute(IdentityCommonConstants.JAVAX_SERVLET_REQUEST_CERTIFICATE, - TestUtil.getCertificate(TestConstants.CERTIFICATE_CONTENT)); - - List validators = new ArrayList<>(); - validators.add(mtlsEnforcementValidator); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - Mockito.doReturn(validators).when(filter).getValidators(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - PowerMockito.when(IdentityCommonUtil.getCertificateFromAttribute(cert)).thenReturn(cert); - - filter.doFilter(request, response, filterChain); - assertEquals(response.getStatus(), HttpServletResponse.SC_OK); - } - - @Test(description = "Test whether the certificate header is present") - public void noCertificateHeaderValidation() throws IOException, OpenBankingException, ServletException { - Map configMap = new HashMap<>(); - PowerMockito.mockStatic(IdentityCommonUtil.class); - configMap.put(IdentityCommonConstants.ENABLE_TRANSPORT_CERT_AS_HEADER, true); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - PowerMockito.mockStatic(IdentityCommonUtil.class); - MTLSEnforcementValidator mtlsEnforcementValidator = Mockito.spy(MTLSEnforcementValidator.class); - - request.setParameter(IdentityCommonConstants.CLIENT_ID, "test"); - List validators = new ArrayList<>(); - validators.add(mtlsEnforcementValidator); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - Mockito.doReturn(validators).when(filter).getValidators(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - - filter.doFilter(request, response, filterChain); - Map responseMap = TestUtil.getResponse(response.getOutputStream()); - assertEquals(response.getStatus(), HttpStatus.SC_BAD_REQUEST); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR), "invalid_request"); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION), - "Transport certificate not found in the request"); - } - - - - @Test(description = "Test the certificate in attribute is passed as a header") - public void certificateIsPresentInAttributeTest() throws IOException, OpenBankingException, ServletException { - MTLSEnforcementValidator mtlsEnforcementValidator = Mockito.spy(MTLSEnforcementValidator.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - - X509Certificate cert = TestUtil.getCertificate(TestConstants.CERTIFICATE_CONTENT); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - request.setParameter(IdentityCommonConstants.CLIENT_ID, "test"); - request.setAttribute(IdentityCommonConstants.JAVAX_SERVLET_REQUEST_CERTIFICATE, - TestUtil.getCertificate(TestConstants.CERTIFICATE_CONTENT)); - - Map configMap = new HashMap<>(); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - List validators = new ArrayList<>(); - validators.add(mtlsEnforcementValidator); - - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - Mockito.doReturn(validators).when(filter).getValidators(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - PowerMockito.when(IdentityCommonUtil.getCertificateFromAttribute(cert)).thenReturn(cert); - - filter.doFilter(request, response, filterChain); - assertEquals(response.getStatus(), HttpServletResponse.SC_OK); - } - - @Test(description = "Test whether the certificate attribute is valid") - public void invalidCertificateHeaderValidation() throws IOException, OpenBankingException, ServletException { - Map configMap = new HashMap<>(); - PowerMockito.mockStatic(IdentityCommonUtil.class); - configMap.put(IdentityCommonConstants.ENABLE_TRANSPORT_CERT_AS_HEADER, true); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - MTLSEnforcementValidator mtlsEnforcementValidator = Mockito.spy(MTLSEnforcementValidator.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - - request.setParameter(IdentityCommonConstants.CLIENT_ID, "test"); - request.setAttribute(IdentityCommonConstants.JAVAX_SERVLET_REQUEST_CERTIFICATE, null); - List validators = new ArrayList<>(); - validators.add(mtlsEnforcementValidator); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - Mockito.doReturn(validators).when(filter).getValidators(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - - filter.doFilter(request, response, filterChain); - Map responseMap = TestUtil.getResponse(response.getOutputStream()); - assertEquals(response.getStatus(), HttpStatus.SC_BAD_REQUEST); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR), "invalid_request"); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION), - "Transport certificate not found in the request"); - - } - - @Test(description = "Test whether the certificate passed through the certificate header is valid") - public void invalidCertificateValidation() throws Exception { - Map configMap = new HashMap<>(); - PowerMockito.mockStatic(IdentityCommonUtil.class); - configMap.put(IdentityCommonConstants.ENABLE_TRANSPORT_CERT_AS_HEADER, true); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - PowerMockito.mockStatic(IdentityCommonUtil.class); - MTLSEnforcementValidator mtlsEnforcementValidator = Mockito.spy(MTLSEnforcementValidator.class); - - request.addHeader(TestConstants.CERTIFICATE_HEADER, "test"); - request.setParameter(IdentityCommonConstants.CLIENT_ID, "test"); - - List validators = new ArrayList<>(); - validators.add(mtlsEnforcementValidator); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - Mockito.doReturn(validators).when(filter).getValidators(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - - filter.doFilter(request, response, filterChain); - Map responseMap = TestUtil.getResponse(response.getOutputStream()); - assertEquals(response.getStatus(), HttpStatus.SC_BAD_REQUEST); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR), "invalid_client"); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION), - "Invalid transport certificate. Certificate passed through the request not valid"); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/validators/SignatureAlgorithmEnforcementValidatorTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/validators/SignatureAlgorithmEnforcementValidatorTest.java deleted file mode 100644 index 2e087bb0..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/java/com/wso2/openbanking/accelerator/identity/token/validators/SignatureAlgorithmEnforcementValidatorTest.java +++ /dev/null @@ -1,215 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.token.validators; - -import com.wso2.openbanking.accelerator.identity.internal.IdentityExtensionsDataHolder; -import com.wso2.openbanking.accelerator.identity.token.DefaultTokenFilter; -import com.wso2.openbanking.accelerator.identity.token.TokenFilter; -import com.wso2.openbanking.accelerator.identity.token.util.TestConstants; -import com.wso2.openbanking.accelerator.identity.token.util.TestUtil; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.http.HttpStatus; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.springframework.mock.web.MockHttpServletRequest; -import org.springframework.mock.web.MockHttpServletResponse; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.oauth.config.OAuthServerConfiguration; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import javax.servlet.FilterChain; -import javax.servlet.http.HttpServletResponse; - -import static org.testng.Assert.assertEquals; - -/** - * Test for signature algorithm enforcement validator. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({IdentityCommonUtil.class, OAuthServerConfiguration.class}) -public class SignatureAlgorithmEnforcementValidatorTest extends PowerMockTestCase { - - MockHttpServletResponse response; - MockHttpServletRequest request; - FilterChain filterChain; - - @BeforeMethod - public void beforeMethod() { - - request = new MockHttpServletRequest(); - response = new MockHttpServletResponse(); - filterChain = Mockito.spy(FilterChain.class); - - } - - @Test(description = "Test the complete signature algorithm validation flow") - public void signatureAlgorithmValidationTest() throws Exception { - - SignatureAlgorithmEnforcementValidator validator = Mockito.spy(SignatureAlgorithmEnforcementValidator.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - request.setParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION, TestConstants.CLIENT_ASSERTION); - request.setAttribute(IdentityCommonConstants.JAVAX_SERVLET_REQUEST_CERTIFICATE, - TestUtil.getCertificate(TestConstants.CERTIFICATE_CONTENT)); - Mockito.doReturn("PS256").when(validator) - .getRegisteredSigningAlgorithm("iYpRm64b2vmvmKDhdL6KZD9z6fca"); - Mockito.doReturn("PS256").when(validator) - .getRequestSigningAlgorithm(TestConstants.CLIENT_ASSERTION); - - List validators = new ArrayList<>(); - validators.add(validator); - - Map configMap = new HashMap<>(); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - Mockito.doReturn(validators).when(filter).getValidators(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - filter.doFilter(request, response, filterChain); - - assertEquals(response.getStatus(), HttpServletResponse.SC_OK); - } - - @Test(description = "Test when registered algorithm and signed algorithm differ") - public void signatureInvalidAlgorithmValidationTest() throws Exception { - Map configMap = new HashMap<>(); - PowerMockito.mockStatic(IdentityCommonUtil.class); - PowerMockito.mockStatic(OAuthServerConfiguration.class); - - OAuthServerConfiguration oAuthServerConfiguration = Mockito.mock(OAuthServerConfiguration.class); - configMap.put(IdentityCommonConstants.ENABLE_TRANSPORT_CERT_AS_HEADER, true); - configMap.put(IdentityCommonConstants.CLIENT_CERTIFICATE_ENCODE, false); - IdentityExtensionsDataHolder.getInstance().setConfigurationMap(configMap); - - SignatureAlgorithmEnforcementValidator validator = Mockito.spy(SignatureAlgorithmEnforcementValidator.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - request.setParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION, TestConstants.CLIENT_ASSERTION); - request.addHeader(TestConstants.CERTIFICATE_HEADER, TestConstants.CERTIFICATE_CONTENT); - - Mockito.doReturn("PS256").when(validator) - .getRegisteredSigningAlgorithm("iYpRm64b2vmvmKDhdL6KZD9z6fca"); - Mockito.doReturn("RS256").when(validator) - .getRequestSigningAlgorithm(TestConstants.CLIENT_ASSERTION); - - List validators = new ArrayList<>(); - validators.add(validator); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - Mockito.doReturn(validators).when(filter).getValidators(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("iYpRm64b2vmvmKDhdL6KZD9z6fca")) - .thenReturn(true); - PowerMockito.when(IdentityCommonUtil.getMTLSAuthHeader()).thenReturn(TestConstants.CERTIFICATE_HEADER); - filter.doFilter(request, response, filterChain); - - Map responseMap = TestUtil.getResponse(response.getOutputStream()); - assertEquals(response.getStatus(), HttpStatus.SC_UNAUTHORIZED); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR), IdentityCommonConstants - .OAUTH2_INVALID_CLIENT_MESSAGE); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION), - "Registered algorithm does not match with the token signed algorithm"); - } - - @Test(description = "Test the validity of the signed assertion without client ID") - public void signatureInvalidAssertionValidationTest() throws Exception { - - SignatureAlgorithmEnforcementValidator validator = Mockito.spy(SignatureAlgorithmEnforcementValidator.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - request.setParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION, TestConstants.CLIENT_ASSERTION_NO_HEADER); - Mockito.doReturn("PS256").when(validator) - .getRegisteredSigningAlgorithm("iYpRm64b2vmvmKDhdL6KZD9z6fca"); - - List validators = new ArrayList<>(); - validators.add(validator); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - Mockito.doReturn(validators).when(filter).getValidators(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - filter.doFilter(request, response, filterChain); - - Map responseMap = TestUtil.getResponse(response.getOutputStream()); - assertEquals(response.getStatus(), HttpStatus.SC_UNAUTHORIZED); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR), "invalid_request"); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION), - "Error occurred while parsing the signed assertion"); - } - - @Test(description = "Test the validity of the signed assertion with client ID") - public void invalidClientAssertionValidationTest() throws Exception { - - SignatureAlgorithmEnforcementValidator validator = Mockito.spy(SignatureAlgorithmEnforcementValidator.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - request.setParameter(IdentityCommonConstants.CLIENT_ID, "test"); - request.setParameter(IdentityCommonConstants.OAUTH_JWT_ASSERTION, "test"); - request.setAttribute(IdentityCommonConstants.JAVAX_SERVLET_REQUEST_CERTIFICATE, - TestUtil.getCertificate(TestConstants.CERTIFICATE_CONTENT)); - - List validators = new ArrayList<>(); - validators.add(validator); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - Mockito.doReturn(validators).when(filter).getValidators(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - filter.doFilter(request, response, filterChain); - - Map responseMap = TestUtil.getResponse(response.getOutputStream()); - assertEquals(response.getStatus(), HttpStatus.SC_UNAUTHORIZED); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR), "invalid_request"); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION), - "Error occurred while parsing the signed assertion"); - } - - @Test(description = "Client ID enforcement test") - public void clientIdEnforcementTest() throws Exception { - - SignatureAlgorithmEnforcementValidator validator = Mockito.spy(SignatureAlgorithmEnforcementValidator.class); - PowerMockito.mockStatic(IdentityCommonUtil.class); - Mockito.doReturn("PS256").when(validator) - .getRegisteredSigningAlgorithm("iYpRm64b2vmvmKDhdL6KZD9z6fca"); - Mockito.doReturn("PS256").when(validator) - .getRequestSigningAlgorithm(TestConstants.CLIENT_ASSERTION); - - List validators = new ArrayList<>(); - validators.add(validator); - - TokenFilter filter = Mockito.spy(TokenFilter.class); - Mockito.doReturn(new DefaultTokenFilter()).when(filter).getDefaultTokenFilter(); - Mockito.doReturn(validators).when(filter).getValidators(); - PowerMockito.when(IdentityCommonUtil.getRegulatoryFromSPMetaData("test")).thenReturn(true); - filter.doFilter(request, response, filterChain); - - Map responseMap = TestUtil.getResponse(response.getOutputStream()); - assertEquals(response.getStatus(), HttpStatus.SC_BAD_REQUEST); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR), "invalid_request"); - assertEquals(responseMap.get(IdentityCommonConstants.OAUTH_ERROR_DESCRIPTION), - "Unable to find client id in the request"); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/resources/common.auth.script.js b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/resources/common.auth.script.js deleted file mode 100644 index 41bb1c33..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/resources/common.auth.script.js +++ /dev/null @@ -1,61 +0,0 @@ -var psuChannel = 'Online Banking'; - -function onLoginRequest(context) { - reportingData(context, "AuthenticationAttempted", false, psuChannel); - - executeStep(1, { - onSuccess: function (context) { - var supportedAcrValues = ['urn:openbanking:psd2:sca', 'urn:openbanking:psd2:ca']; - var selectedAcr = selectAcrFrom(context, supportedAcrValues); - reportingData(context, "AuthenticationSuccessful", false, psuChannel); - - if (isACREnabled()) { - - context.selectedAcr = selectedAcr; - if (isTRAEnabled()) { - if (selectedAcr === 'urn:openbanking:psd2:ca') { - executeTRAFunction(context); - } else { - executeStep(2, { - onSuccess: function (context) { - reportingData(context, "AuthenticationSuccessful", true, psuChannel); - }, - onFail: function (context) { - reportingData(context, "AuthenticationFailed", false, psuChannel); - } - }); - } - } else { - if (selectedAcr == 'urn:openbanking:psd2:sca') { - executeStep(2, { - onSuccess: function (context) { - reportingData(context, "AuthenticationSuccessful", true, psuChannel); - }, - onFail: function (context) { - reportingData(context, "AuthenticationFailed", false, psuChannel); - } - }); - } - } - - } else { - if (isTRAEnabled()) { - executeTRAFunction(context); - } else { - executeStep(2, { - onSuccess: function (context) { - reportingData(context, "AuthenticationSuccessful", true, psuChannel); - }, - onFail: function (context) { - reportingData(context, "AuthenticationFailed", false, psuChannel); - } - }); - } - } - }, - onFail: function (context) { //basic auth fail - reportingData(context, "AuthenticationFailed", false, psuChannel); - //retry - } - }); -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/resources/testng.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/resources/testng.xml deleted file mode 100644 index f2b8a270..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/resources/testng.xml +++ /dev/null @@ -1,151 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/resources/wso2carbon.jks b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.identity/src/test/resources/wso2carbon.jks deleted file mode 100644 index c8775783fb3555d6e88255c1236576e7a4a7df1f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 98874 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 o1{wExgs!Xzw`yH% - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../pom.xml - - 4.0.0 - - com.wso2.openbanking.accelerator.keymanager - bundle - WSO2 Open Banking - KeyManager Extensions - - - - org.wso2.eclipse.osgi - org.eclipse.osgi.services - - - org.eclipse.osgi - org.eclipse.osgi - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - provided - - - org.wso2.carbon.apimgt - org.wso2.carbon.apimgt.api - - - org.wso2.carbon.apimgt - org.wso2.carbon.apimgt.impl - - - org.mockito - mockito-all - - - org.powermock - powermock-api-mockito - test - - - org.testng - testng - test - - - org.jacoco - org.jacoco.agent - runtime - test - - - org.powermock - powermock-module-testng - - - - - - org.apache.maven.plugins - maven-compiler-plugin - - - org.jacoco - jacoco-maven-plugin - - - - **/*Component.class - **/*DataHolder.class - **/OBKeyManagerConfiguration.class - - - - - default-prepare-agent - - prepare-agent - - - - default-prepare-agent-integration - - prepare-agent-integration - - - - default-report - - report - - - - default-report-integration - - report-integration - - - - default-check - - check - - - - - BUNDLE - - - INSTRUCTION - COVEREDRATIO - 0.80 - - - - - - - - - - org.apache.maven.plugins - maven-surefire-plugin - - - src/test/resources/testng.xml - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - false - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - org.apache.felix - maven-bundle-plugin - true - - - - ${project.artifactId} - - - com.wso2.openbanking.accelerator.keymanager.internal - - - org.apache.axis2;version="${axis2.imp.pkg.version.range}", - org.apache.axis2.client;version="${axis2.imp.pkg.version.range}", - org.apache.axis2.context;version="${axis2.imp.pkg.version.range}", - org.apache.commons.logging;version="${commons.logging.version}", - org.osgi.service.component;version="${osgi.service.component.imp.pkg.version.range}", - org.wso2.carbon.apimgt.api;version="${org.wso2.carbon.apimgt.version.range}", - org.wso2.carbon.apimgt.api.model;version="${org.wso2.carbon.apimgt.version.range}", - org.wso2.carbon.apimgt.impl;version="${org.wso2.carbon.apimgt.version.range}", - org.wso2.carbon.apimgt.impl.jwt;version="${org.wso2.carbon.apimgt.version.range}", - org.wso2.carbon.authenticator.stub;version="${carbon.kernel.version.range}", - org.wso2.carbon.user.mgt.stub;version="${carbon.identity.framework.version.range}", - org.wso2.carbon.identity.oauth.stub;version="${org.wso2.carbon.identity.oauth.stub.version.range}", - org.wso2.carbon.identity.oauth.stub.dto;version="${org.wso2.carbon.identity.oauth.stub.version.range}", - - - !com.wso2.openbanking.accelerator.keymanager.internal, - com.wso2.openbanking.accelerator.keymanager.*;version="${project.version}", - - * - <_dsannotations>* - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/KeyManagerUtil.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/KeyManagerUtil.java deleted file mode 100644 index fed1c6be..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/KeyManagerUtil.java +++ /dev/null @@ -1,225 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.keymanager; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.keymanager.internal.KeyManagerDataHolder; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.axis2.client.Options; -import org.apache.axis2.client.ServiceClient; -import org.apache.axis2.context.ServiceContext; -import org.apache.axis2.transport.http.HTTPConstants; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.apimgt.api.APIManagementException; -import org.wso2.carbon.apimgt.api.ExceptionCodes; -import org.wso2.carbon.apimgt.api.model.OAuthAppRequest; -import org.wso2.carbon.apimgt.impl.APIConstants; -import org.wso2.carbon.apimgt.impl.APIManagerConfiguration; -import org.wso2.carbon.authenticator.stub.LoginAuthenticationExceptionException; -import org.wso2.carbon.identity.core.util.IdentityTenantUtil; -import org.wso2.carbon.user.api.UserRealm; -import org.wso2.carbon.user.api.UserStoreException; -import org.wso2.carbon.user.core.UserCoreConstants; -import org.wso2.carbon.user.core.common.AbstractUserStoreManager; - -import java.lang.reflect.InvocationTargetException; -import java.rmi.RemoteException; -import java.util.HashMap; -import java.util.Map; - -/** - * Util class for OB key manager. - */ -public class KeyManagerUtil { - - private static final Log log = LogFactory.getLog(KeyManagerUtil.class); - - /** - * Method to get the session Cookie. - * - * @return Session cookie as a String - * @throws APIManagementException When failed to obtain the session cookie - * @deprecated ApplicationManagementService is used instead of SOAP API calls. - */ - @Deprecated - @Generated(message = "Excluding from unit test coverage") - public static String getSessionCookie() throws APIManagementException { - - String sessionCookie = ""; - APIManagerConfiguration config = KeyManagerDataHolder.getInstance().getApiManagerConfigurationService() - .getAPIManagerConfiguration(); - String adminUsername = config.getFirstProperty(APIConstants.API_KEY_VALIDATOR_USERNAME); - - char[] adminPassword = config.getFirstProperty(APIConstants.API_KEY_VALIDATOR_PASSWORD).toCharArray(); - try { - if (KeyManagerDataHolder.getInstance().getAuthenticationAdminStub().login(adminUsername, - String.valueOf(adminPassword), "localhost")) { - ServiceContext serviceContext = KeyManagerDataHolder.getInstance().getAuthenticationAdminStub() - ._getServiceClient().getLastOperationContext() - .getServiceContext(); - sessionCookie = (String) serviceContext.getProperty(HTTPConstants.COOKIE_STRING); - } - } catch (RemoteException e) { - throw new APIManagementException("Error occurred while making remote call.", e); - } catch (LoginAuthenticationExceptionException e) { - throw new APIManagementException("Error occurred while authenticating user.", e); - } - return sessionCookie; - } - - - /** - * Method to bind session cookie to Admin service client. - * - * @param serviceClient Admin service client - * @param sessionCookie session cookie as a string - * @deprecated ApplicationManagementService is used instead of SOAP API calls. - */ - @Deprecated - @Generated(message = "Excluding from unit test coverage") - public static void setAdminServiceSession(ServiceClient serviceClient, String sessionCookie) { - - Options userAdminOption = serviceClient.getOptions(); - userAdminOption.setManageSession(true); - userAdminOption.setProperty(org.apache.axis2.transport.http.HTTPConstants.COOKIE_STRING, sessionCookie); - } - - /** - * Obtain OB Key Manage Extension Impl class from config. - * - * @return obKeyManagerExtensionInterface - */ - public static OBKeyManagerExtensionInterface getOBKeyManagerExtensionImpl() throws APIManagementException { - OBKeyManagerExtensionInterface obKeyManagerExtensionImpl; - try { - String obKeyManagerExtensionImplName = OpenBankingConfigParser.getInstance() - .getOBKeyManagerExtensionImpl(); - if (!StringUtils.isEmpty(obKeyManagerExtensionImplName)) { - obKeyManagerExtensionImpl = (OBKeyManagerExtensionInterface) - Class.forName(obKeyManagerExtensionImplName).getDeclaredConstructor().newInstance(); - return obKeyManagerExtensionImpl; - } else { - return null; - } - - } catch (InstantiationException | IllegalAccessException | - InvocationTargetException | NoSuchMethodException | ClassNotFoundException e) { - throw new APIManagementException("Failed to obtain OB Key Manager Extension Impl instance", e); - } - } - - /** - * Extract values for additional properties from input. - * - * @param oauthAppRequest OAuthAppRequest object - * @return Additional Property Map - * @throws APIManagementException - */ - public static HashMap getValuesForAdditionalProperties(OAuthAppRequest oauthAppRequest) - throws APIManagementException { - // Get additional properties defined in the config - Map> keyManagerAdditionalProperties = OpenBankingConfigParser.getInstance() - .getKeyManagerAdditionalProperties(); - HashMap additionalProperties = new HashMap<>(); - Object additionalPropertiesJSON; - try { - // Get values for additional properties given at key generation step - additionalPropertiesJSON = new JSONParser(JSONParser.MODE_PERMISSIVE) - .parse((String) oauthAppRequest.getOAuthApplicationInfo() - .getParameter(APIConstants.JSON_ADDITIONAL_PROPERTIES)); - if (!(additionalPropertiesJSON instanceof JSONObject)) { - log.error(APIConstants.JSON_ADDITIONAL_PROPERTIES + " is not a JSON object"); - throw new APIManagementException(ExceptionCodes.JSON_PARSE_ERROR.getErrorMessage(), - ExceptionCodes.JSON_PARSE_ERROR); - } - } catch (ParseException e) { - throw new APIManagementException(ExceptionCodes.JSON_PARSE_ERROR.getErrorMessage(), e, - ExceptionCodes.JSON_PARSE_ERROR); - } - - JSONObject additionalPropertiesJSONObject = (JSONObject) additionalPropertiesJSON; - // Add values of additional properties defined in the config to the default additional property list JSON object - for (String key : keyManagerAdditionalProperties.keySet()) { - additionalProperties.put(key, additionalPropertiesJSONObject.getAsString(key)); - } - return additionalProperties; - } - - /** - * Obtain Application role name using application name. - * @param applicationName Application name - * @return Application role name - */ - protected static String getAppRoleName(String applicationName) { - - return org.wso2.carbon.identity.application.mgt.ApplicationConstants.APPLICATION_DOMAIN + - UserCoreConstants.DOMAIN_SEPARATOR + applicationName; - } - - /** - * Add the application role to the admin so that admin can manipulate app data. - * @param applicationName Application Name - * @throws APIManagementException - */ - @Generated(message = "excluding from coverage because it is a void method with external calls") - protected static void addApplicationRoleToAdmin(String applicationName) throws APIManagementException { - - APIManagerConfiguration config = KeyManagerDataHolder.getInstance().getApiManagerConfigurationService() - .getAPIManagerConfiguration(); - String adminUsername = config.getFirstProperty(APIConstants.API_KEY_VALIDATOR_USERNAME); - String roleName = getAppRoleName(applicationName); - String[] newRoles = {roleName}; - - try { - // assign new application role to the user. - UserRealm realm = getUserRealm(adminUsername); - if (realm != null) { - if (((AbstractUserStoreManager) realm.getUserStoreManager()).isUserInRole(adminUsername, roleName)) { - if (log.isDebugEnabled()) { - log.debug("The user: " + adminUsername + " is already having the role: " + roleName); - } - } else { - realm.getUserStoreManager().updateRoleListOfUser(adminUsername, null, newRoles); - if (log.isDebugEnabled()) { - log.debug("Assigning application role : " + roleName + " to the user : " + adminUsername); - } - } - } - } catch (UserStoreException e) { - throw new APIManagementException("Error while assigning application role: " + roleName + - " to the user: " + adminUsername, e); - } - } - - @Generated(message = "separated for unit testing purposes") - protected static UserRealm getUserRealm(String username) throws APIManagementException { - - try { - int tenantId = IdentityTenantUtil.getTenantIdOfUser(username); - return KeyManagerDataHolder.getInstance().getRealmService().getTenantUserRealm(tenantId); - } catch (UserStoreException e) { - throw new APIManagementException("Error while obtaining user realm for user: " + username, e); - } - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/OBKeyManagerConfiguration.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/OBKeyManagerConfiguration.java deleted file mode 100644 index b02b314f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/OBKeyManagerConfiguration.java +++ /dev/null @@ -1,136 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.keymanager; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import org.apache.commons.lang.StringUtils; -import org.osgi.service.component.annotations.Component; -import org.wso2.carbon.apimgt.api.model.ConfigurationDto; -import org.wso2.carbon.apimgt.api.model.KeyManagerConnectorConfiguration; -import org.wso2.carbon.apimgt.impl.APIConstants; -import org.wso2.carbon.apimgt.impl.jwt.JWTValidatorImpl; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Collections; -import java.util.List; -import java.util.Map; - -/** - * Key manager configuration class to override the default key manager interface implementation. - */ -@Component( - name = "com.wso2.open.banking.keymanager.config", - immediate = true, - service = KeyManagerConnectorConfiguration.class -) -public class OBKeyManagerConfiguration implements KeyManagerConnectorConfiguration { - - @Override - public String getImplementation() { - - return OBKeyManagerImpl.class.getName(); - } - - @Override - public String getJWTValidator() { - - return JWTValidatorImpl.class.getName(); - } - - @Override - public List getConnectionConfigurations() { - - List configurationDtoList = new ArrayList<>(); - configurationDtoList - .add(new ConfigurationDto("Username", "Username", "input", "Username of admin user", "", - true, false, Collections.emptyList(), false)); - configurationDtoList - .add(new ConfigurationDto("Password", "Password", "input", - "Password of Admin user", "", true, true, Collections.emptyList(), false)); - return configurationDtoList; - - } - - @Override - public List getApplicationConfigurations() { - - List applicationConfigurationsList = new ArrayList(); - applicationConfigurationsList - .add(new ConfigurationDto(APIConstants.KeyManager.APPLICATION_ACCESS_TOKEN_EXPIRY_TIME, - "Application Access Token Expiry Time ", "input", "Type Application Access Token Expiry Time " + - "in seconds ", APIConstants.KeyManager.NOT_APPLICABLE_VALUE, false, false, - Collections.EMPTY_LIST, false)); - applicationConfigurationsList - .add(new ConfigurationDto(APIConstants.KeyManager.USER_ACCESS_TOKEN_EXPIRY_TIME, - "User Access Token Expiry Time ", "input", "Type User Access Token Expiry Time " + - "in seconds ", APIConstants.KeyManager.NOT_APPLICABLE_VALUE, false, false, - Collections.EMPTY_LIST, false)); - applicationConfigurationsList - .add(new ConfigurationDto(APIConstants.KeyManager.REFRESH_TOKEN_EXPIRY_TIME, - "Refresh Token Expiry Time ", "input", "Type Refresh Token Expiry Time " + - "in seconds ", APIConstants.KeyManager.NOT_APPLICABLE_VALUE, false, false, - Collections.EMPTY_LIST, false)); - applicationConfigurationsList - .add(new ConfigurationDto(APIConstants.KeyManager.ID_TOKEN_EXPIRY_TIME, - "Id Token Expiry Time", "input", "Type ID Token Expiry Time " + - "in seconds ", APIConstants.KeyManager.NOT_APPLICABLE_VALUE, false, false, - Collections.EMPTY_LIST, false)); - - Map> keyManagerAdditionalProperties = OpenBankingConfigParser.getInstance() - .getKeyManagerAdditionalProperties(); - - for (Map.Entry> propertyElement : keyManagerAdditionalProperties.entrySet()) { - String propertyName = propertyElement.getKey(); - Map property = propertyElement.getValue(); - boolean required = !StringUtils.isEmpty(property.get("required")) - && Boolean.parseBoolean(property.get("required")); - boolean mask = !StringUtils.isEmpty(property.get("mask")) - && Boolean.parseBoolean(property.get("mask")); - boolean multiple = !StringUtils.isEmpty(property.get("multiple")) - && Boolean.parseBoolean(property.get("multiple")); - List values = StringUtils.isEmpty(property.get("values")) ? Collections.EMPTY_LIST - : Arrays.asList(property.get("values").split(",")); - - applicationConfigurationsList.add(new ConfigurationDto(propertyName, property.get("label"), - property.get("type"), property.get("tooltip"), property.get("default"), required , mask - , values, multiple)); - } - - return applicationConfigurationsList; - } - - @Override - public String getType() { - - return OBKeyManagerConstants.CUSTOM_KEYMANAGER_TYPE; - } - - @Override - public String getDefaultScopesClaim() { - - return APIConstants.JwtTokenConstants.SCOPE; - } - - @Override - public String getDefaultConsumerKeyClaim() { - - return APIConstants.JwtTokenConstants.AUTHORIZED_PARTY; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/OBKeyManagerConstants.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/OBKeyManagerConstants.java deleted file mode 100644 index 766f3588..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/OBKeyManagerConstants.java +++ /dev/null @@ -1,28 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.keymanager; - -/** - * OB Key Manager Constants. - */ -public class OBKeyManagerConstants { - - public static final String CUSTOM_KEYMANAGER_TYPE = "ObKeyManager"; - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/OBKeyManagerExtensionInterface.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/OBKeyManagerExtensionInterface.java deleted file mode 100644 index d2fd842e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/OBKeyManagerExtensionInterface.java +++ /dev/null @@ -1,78 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.keymanager; - -import org.wso2.carbon.apimgt.api.APIManagementException; -import org.wso2.carbon.apimgt.api.model.ConfigurationDto; -import org.wso2.carbon.apimgt.api.model.OAuthAppRequest; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.oauth.dto.OAuthConsumerAppDTO; - -import java.util.HashMap; -import java.util.Map; - -/** - * Interface for validation OB Key Manager Additional Properties. - */ -public interface OBKeyManagerExtensionInterface { - - /** - * Validate additional properties. - * - * @param obAdditionalProperties OB Additional Properties Map - * @throws APIManagementException when failed to validate a given property - */ - void validateAdditionalProperties(Map obAdditionalProperties) - throws APIManagementException; - - /** - * Do changes to app request before creating the app at toolkit level. - * - * @param oAuthAppRequest OAuth Application Request - * @param additionalProperties Values for additional property list defined in the config - * @throws APIManagementException when failed to validate a given property - */ - void doPreCreateApplication(OAuthAppRequest oAuthAppRequest, HashMap additionalProperties) - throws APIManagementException; - - /** - * Do changes to app request before updating the app at toolkit level. - * - * @param oAuthAppRequest OAuth Application Request - * @param additionalProperties Values for additional property list defined in the config - * @throws APIManagementException when failed to validate a given property - */ - - void doPreUpdateApplication(OAuthAppRequest oAuthAppRequest, HashMap additionalProperties, - ServiceProvider serviceProvider) - throws APIManagementException; - - /** - * Do changes to service provider before updating the service provider properties. - * - * @param oAuthConsumerAppDTO oAuth application DTO - * @param serviceProvider Service provider application - * @param additionalProperties Values for additional property list defined in the config - * @param isCreateApp Whether this functions is called at app creation - * @throws APIManagementException when failed to validate a given property - */ - void doPreUpdateSpApp(OAuthConsumerAppDTO oAuthConsumerAppDTO, ServiceProvider serviceProvider, - HashMap additionalProperties, boolean isCreateApp) - throws APIManagementException; -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/OBKeyManagerImpl.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/OBKeyManagerImpl.java deleted file mode 100644 index ee86cf1c..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/OBKeyManagerImpl.java +++ /dev/null @@ -1,470 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.keymanager; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.common.util.ServiceProviderUtils; -import com.wso2.openbanking.accelerator.keymanager.internal.KeyManagerDataHolder; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.apimgt.api.APIManagementException; -import org.wso2.carbon.apimgt.api.ExceptionCodes; -import org.wso2.carbon.apimgt.api.model.AccessTokenInfo; -import org.wso2.carbon.apimgt.api.model.AccessTokenRequest; -import org.wso2.carbon.apimgt.api.model.ApplicationConstants; -import org.wso2.carbon.apimgt.api.model.ConfigurationDto; -import org.wso2.carbon.apimgt.api.model.KeyManagerConnectorConfiguration; -import org.wso2.carbon.apimgt.api.model.OAuthAppRequest; -import org.wso2.carbon.apimgt.api.model.OAuthApplicationInfo; -import org.wso2.carbon.apimgt.impl.AMDefaultKeyManagerImpl; -import org.wso2.carbon.apimgt.impl.APIConstants; -import org.wso2.carbon.identity.application.common.IdentityApplicationManagementException; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.application.common.model.ServiceProviderProperty; -import org.wso2.carbon.identity.application.common.util.IdentityApplicationConstants; -import org.wso2.carbon.identity.application.mgt.ApplicationManagementServiceImpl; -import org.wso2.carbon.identity.oauth.IdentityOAuthAdminException; -import org.wso2.carbon.identity.oauth.OAuthAdminService; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -/** - * OB key manager client impl class. - */ -public class OBKeyManagerImpl extends AMDefaultKeyManagerImpl implements OBKeyManagerExtensionInterface { - - private static final Log log = LogFactory.getLog(OBKeyManagerImpl.class); - - public static final String OAUTH2 = "oauth2"; - - @Override - public AccessTokenInfo getNewApplicationAccessToken(AccessTokenRequest tokenRequest) throws APIManagementException { - - try { - ApplicationManagementServiceImpl applicationManagementService = getApplicationMgmtServiceImpl(); - ServiceProvider serviceProvider = applicationManagementService.getServiceProviderByClientId( - tokenRequest.getClientId(), IdentityApplicationConstants.OAuth2.NAME, tenantDomain); - if (serviceProvider != null) { - ServiceProviderProperty regulatoryProperty = Arrays.stream(serviceProvider.getSpProperties()) - .filter(serviceProviderProperty -> serviceProviderProperty.getName() - .equalsIgnoreCase(OpenBankingConstants.REGULATORY)).findAny().orElse(null); - if (regulatoryProperty != null && "true".equalsIgnoreCase(regulatoryProperty.getValue())) { - return null; - } - } - } catch (IdentityApplicationManagementException e) { - log.error("Error while generating keys. ", e); - } - return super.getNewApplicationAccessToken(tokenRequest); - } - - @Override - public String getType() { - - return OBKeyManagerConstants.CUSTOM_KEYMANAGER_TYPE; - } - - /** - * Validate OAuth Application Properties. - * - * @param oAuthApplicationInfo OAuthApplication Information - * @throws APIManagementException when failed to validate the OAuth application properties - */ - @Override - protected void validateOAuthAppCreationProperties(OAuthApplicationInfo oAuthApplicationInfo) - throws APIManagementException { - - String type = getType(); - List missedRequiredValues = new ArrayList<>(); - Map obAdditionalProperties = new HashMap<>(); - - KeyManagerConnectorConfiguration oBKeyManagerConnectorConfiguration = KeyManagerDataHolder.getInstance() - .getKeyManagerConnectorConfiguration(type); - // Obtain additional key manager configurations defined in config - Map> keyManagerAdditionalProperties = OpenBankingConfigParser.getInstance() - .getKeyManagerAdditionalProperties(); - - if (oBKeyManagerConnectorConfiguration != null) { - List applicationConfigurationDtoList = oBKeyManagerConnectorConfiguration - .getApplicationConfigurations(); - Object additionalProperties = oAuthApplicationInfo.getParameter(APIConstants.JSON_ADDITIONAL_PROPERTIES); - if (additionalProperties != null) { - Object additionalPropertiesJson; - try { - additionalPropertiesJson = new JSONParser(JSONParser.MODE_PERMISSIVE) - .parse(additionalProperties.toString()); - if (!(additionalPropertiesJson instanceof JSONObject)) { - String errMsg = "Additional properties is not a valid json object"; - log.error(errMsg); - throw new APIManagementException(errMsg, ExceptionCodes - .from(ExceptionCodes.INVALID_APPLICATION_ADDITIONAL_PROPERTIES, - errMsg)); - } - } catch (ParseException e) { - String errMsg = "Additional properties is not a valid JSON string"; - throw new APIManagementException(errMsg, e, ExceptionCodes - .from(ExceptionCodes.INVALID_APPLICATION_ADDITIONAL_PROPERTIES, - errMsg)); - } - - for (ConfigurationDto configurationDto : applicationConfigurationDtoList) { - String key = configurationDto.getName(); - String values = ((JSONObject) additionalPropertiesJson).getAsString(key); - - if (values == null) { - // AbstractKeyManager Validations - // Check if mandatory parameters are missing - if (configurationDto.isRequired()) { - missedRequiredValues.add(configurationDto.getName()); - } - } else { - // OBKeyManager Validations - if (keyManagerAdditionalProperties.containsKey(key)) { - configurationDto.setValues(Arrays.asList(values)); - obAdditionalProperties.put(key, configurationDto); - } else { - // AMDefaultKeyManager validations - // Check for invalid time periods - if (StringUtils.isNotBlank(values) && !StringUtils - .equals(values, APIConstants.KeyManager.NOT_APPLICABLE_VALUE)) { - try { - Long longValue = Long.parseLong(values); - if (longValue < 0) { - String errMsg = "Application configuration values cannot have negative values."; - throw new APIManagementException(errMsg, ExceptionCodes - .from(ExceptionCodes.INVALID_APPLICATION_ADDITIONAL_PROPERTIES, - errMsg)); - } - } catch (NumberFormatException e) { - String errMsg = "Application configuration values cannot have string values."; - throw new APIManagementException(errMsg, e, ExceptionCodes - .from(ExceptionCodes.INVALID_APPLICATION_ADDITIONAL_PROPERTIES, errMsg)); - } - } - } - } - } - if (!missedRequiredValues.isEmpty()) { - throw new APIManagementException( - "Missing required properties to create/update oauth " + "application", - ExceptionCodes.KEY_MANAGER_MISSING_REQUIRED_PROPERTIES_IN_APPLICATION); - } - // Call external method to validate additional properties - if (obAdditionalProperties.size() != 0) { - validateAdditionalProperties(obAdditionalProperties); - } - } - } else { - throw new APIManagementException("Invalid Key Manager Type " + type, ExceptionCodes.KEY_MANAGER_NOT_FOUND); - } - } - - /** - * Overriding the default create application method with Open Banking requirements. - * - * @param oauthAppRequest OAuthApplicationRequest object - * @return OAuthApplicationInfo object - * @throws APIManagementException when failed to create the application properly in Key Manager - */ - @Override - @Generated(message = "Excluding from code coverage since it is covered from other method") - public OAuthApplicationInfo createApplication(OAuthAppRequest oauthAppRequest) throws APIManagementException { - - HashMap additionalProperties = KeyManagerUtil.getValuesForAdditionalProperties(oauthAppRequest); - if (Boolean.parseBoolean(additionalProperties.get(OpenBankingConstants.REGULATORY))) { - // Adding SP property to identify create request. Will be removed when setting up authenticators. - additionalProperties.put("AppCreateRequest", "true"); - } - doPreCreateApplication(oauthAppRequest, additionalProperties); - OAuthApplicationInfo oAuthApplicationInfo = oauthAppRequest.getOAuthApplicationInfo(); - String username = (String) oAuthApplicationInfo.getParameter(ApplicationConstants.OAUTH_CLIENT_USERNAME); - oAuthApplicationInfo = super.createApplication(oauthAppRequest); - // Need to get the application name after creating the application to obtain the generated app name - String appName = oAuthApplicationInfo.getClientName(); - // Admin needs to have application role to retrieve and edit the app - KeyManagerUtil.addApplicationRoleToAdmin(appName); - - try { - String tenantDomain = ServiceProviderUtils.getSpTenantDomain(oAuthApplicationInfo.getClientId()); - updateSpProperties(appName, tenantDomain, username, additionalProperties, true); - - ServiceProvider appServiceProvider = getApplicationMgmtServiceImpl() - .getServiceProvider(appName, tenantDomain); - ServiceProviderProperty regulatoryProperty = getSpPropertyFromSPMetaData( - OpenBankingConstants.REGULATORY, appServiceProvider.getSpProperties()); - - if (regulatoryProperty != null) { - if (Boolean.parseBoolean(regulatoryProperty.getValue())) { - OAuthAppRequest updatedOauthAppRequest = oauthAppRequest; - ServiceProviderProperty appNameProperty = getSpPropertyFromSPMetaData("DisplayName", - appServiceProvider.getSpProperties()); - if (appNameProperty != null) { - updatedOauthAppRequest.getOAuthApplicationInfo().setClientName(appNameProperty.getValue()); - } - // Assigning null as it is how the tokenScope parameter is used in the updateApplication method - updatedOauthAppRequest.getOAuthApplicationInfo().addParameter("tokenScope", null); - super.updateApplication(updatedOauthAppRequest); - } - } - return oAuthApplicationInfo; - - } catch (OpenBankingException | APIManagementException e) { - throw new APIManagementException(ExceptionCodes.OAUTH2_APP_CREATION_FAILED.getErrorMessage(), - e, ExceptionCodes.OAUTH2_APP_CREATION_FAILED); - } catch (IdentityApplicationManagementException e) { - String errMsg = "error occurred in retrieving service provider for app " + appName; - log.error(errMsg); - throw new APIManagementException(errMsg, e, ExceptionCodes.OAUTH2_APP_UPDATE_FAILED); - } - } - - @Override - @Generated(message = "Excluding from code coverage since it is covered from other method") - public OAuthApplicationInfo updateApplication(OAuthAppRequest oAuthAppRequest) throws APIManagementException { - - HashMap additionalProperties = KeyManagerUtil.getValuesForAdditionalProperties(oAuthAppRequest); - // Adding SP property to identify update request. Will be removed when updating authenticators. - additionalProperties.put("AppCreateRequest", "false"); - OAuthApplicationInfo oAuthApplicationInfo = oAuthAppRequest.getOAuthApplicationInfo(); - String clientId = oAuthApplicationInfo.getClientId(); - // There is no way to identify the client type in here. So we have to hardcode "oauth2" as the client type - try { - ServiceProvider serviceProvider = getApplicationMgmtServiceImpl() - .getServiceProviderByClientId(clientId, OAUTH2, tenantDomain); - doPreUpdateApplication(oAuthAppRequest, additionalProperties, serviceProvider); - String appName = serviceProvider.getApplicationName(); - String username = (String) oAuthApplicationInfo.getParameter(ApplicationConstants.OAUTH_CLIENT_USERNAME); - updateSpProperties(appName, tenantDomain, username, additionalProperties, false); - } catch (IdentityApplicationManagementException e) { - String errMsg = "Cannot find Service provider application for client Id " + clientId; - log.error(errMsg); - throw new APIManagementException(errMsg, ExceptionCodes.OAUTH2_APP_RETRIEVAL_FAILED); - } - - oAuthApplicationInfo = super.updateApplication(oAuthAppRequest); - return oAuthApplicationInfo; - } - - @Override - @Generated(message = "Excluding from code coverage since it is covered from other method") - public OAuthApplicationInfo retrieveApplication(String consumerKey) throws APIManagementException { - - OAuthApplicationInfo oAuthApplicationInfo = super.retrieveApplication(consumerKey); - String name = oAuthApplicationInfo.getClientName(); - try { - String tenantDomain = ServiceProviderUtils.getSpTenantDomain(consumerKey); - org.wso2.carbon.identity.application.common.model.ServiceProvider appServiceProvider = - getApplicationMgmtServiceImpl().getServiceProvider(name, tenantDomain); - // Iterate OB specific additional properties to check whether they override the value of any predefined - // sp properties in application management listeners - List spProperties = - new ArrayList<>(Arrays.asList(appServiceProvider.getSpProperties())); - return updateAdditionalProperties(oAuthApplicationInfo, spProperties); - } catch (IdentityApplicationManagementException | OpenBankingException e) { - throw new APIManagementException(ExceptionCodes.OAUTH2_APP_RETRIEVAL_FAILED.getErrorMessage(), - e, ExceptionCodes.OAUTH2_APP_RETRIEVAL_FAILED); - } - } - - /** - * @param spAppName Generate service provider application name - * @param tenantDomain Tenant domain of the service provider application - * @param username Application owner - * @param additionalProperties new Service provider property map - * @param isCreateApp Whether this function is called at app creation - * @throws APIManagementException - */ - protected void updateSpProperties(String spAppName, String tenantDomain, String username, - HashMap additionalProperties, boolean isCreateApp) - throws APIManagementException { - - try { - org.wso2.carbon.identity.oauth.dto.OAuthConsumerAppDTO oAuthConsumerAppDTO = getOAuthAdminService(). - getOAuthApplicationDataByAppName(spAppName); - ServiceProvider serviceProvider = getApplicationMgmtServiceImpl() - .getServiceProvider(spAppName, tenantDomain); - doPreUpdateSpApp(oAuthConsumerAppDTO, serviceProvider, additionalProperties, isCreateApp); - // Iterate OB specific additional properties to check whether they override the value of any predefined - // sp properties in application management listeners - List spProperties = - new ArrayList<>(Arrays.asList(serviceProvider.getSpProperties())); - for (Map.Entry propertyElement : additionalProperties.entrySet()) { - ServiceProviderProperty overridenSPproperty = spProperties.stream().filter( - serviceProviderProperty -> serviceProviderProperty.getName() - .equalsIgnoreCase(propertyElement.getKey())).findAny().orElse(null); - // If SP property is overridden, remove old SP property and add the new one - if (overridenSPproperty != null) { - spProperties.remove(overridenSPproperty); - overridenSPproperty.setValue(propertyElement.getValue()); - spProperties.add(overridenSPproperty); - } else { - ServiceProviderProperty additionalProperty = new ServiceProviderProperty(); - additionalProperty.setName(propertyElement.getKey()); - additionalProperty.setValue(propertyElement.getValue()); - spProperties.add(additionalProperty); - } - } - serviceProvider.setSpProperties(spProperties.toArray(new ServiceProviderProperty[0])); - try { - getApplicationMgmtServiceImpl().updateApplication(serviceProvider, tenantDomain, username); - if (log.isDebugEnabled()) { - log.debug("Successfully updated service provider properties for app " + spAppName); - } - } catch (IdentityApplicationManagementException e) { - String errMsg = "error occurred while updating service provider " + spAppName; - log.error(errMsg); - throw new APIManagementException(errMsg, e, ExceptionCodes.OAUTH2_APP_UPDATE_FAILED); - } - - try { - getOAuthAdminService().updateConsumerApplication(oAuthConsumerAppDTO); - if (log.isDebugEnabled()) { - log.debug("Successfully updated oAuth application DTO for app " + spAppName); - } - } catch (IdentityOAuthAdminException e) { - String errMsg = "error occurred while updating oAuth Application data for app " + spAppName; - log.error(errMsg); - throw new APIManagementException(errMsg, e, ExceptionCodes.OAUTH2_APP_UPDATE_FAILED); - } - - } catch (IdentityApplicationManagementException | IdentityOAuthAdminException e) { - String errMsg = "error occurred in retrieving service provider or oAuth app " + spAppName; - log.error(errMsg); - throw new APIManagementException(errMsg, e, ExceptionCodes.OAUTH2_APP_UPDATE_FAILED); - } - - } - - /** - * Extract values for additional properties defined in the config from database and add to oAuthApplicationInfo. - * - * @return oAuth application Info - */ - protected OAuthApplicationInfo updateAdditionalProperties(OAuthApplicationInfo oAuthApplicationInfo, - List spProperties) { - - Map> keyManagerAdditionalProperties = OpenBankingConfigParser.getInstance() - .getKeyManagerAdditionalProperties(); - for (String key : keyManagerAdditionalProperties.keySet()) { - for (ServiceProviderProperty spProperty : spProperties) { - if (spProperty.getName().equalsIgnoreCase(key)) { - ((HashMap) oAuthApplicationInfo.getParameter( - APIConstants.JSON_ADDITIONAL_PROPERTIES)).put(key, spProperty.getValue()); - } - } - } - return oAuthApplicationInfo; - } - - /** - * Validate additional properties at toolkit level. - * - * @param obAdditionalProperties Values for additional property list defined in the config - * @throws APIManagementException when failed to validate a given property - */ - @Generated(message = "Excluding from code coverage since the method body is at toolkit") - public void validateAdditionalProperties(Map obAdditionalProperties) - throws APIManagementException { - - OBKeyManagerExtensionInterface obKeyManagerExtensionImpl = KeyManagerUtil.getOBKeyManagerExtensionImpl(); - if (obKeyManagerExtensionImpl != null) { - obKeyManagerExtensionImpl.validateAdditionalProperties(obAdditionalProperties); - } - } - - /** - * Do changes to app request before creating the app at toolkit level. - * - * @param additionalProperties Values for additional property list defined in the config - * @throws APIManagementException when failed to validate a given property - */ - @Generated(message = "Excluding from code coverage since the method body is at toolkit") - public void doPreCreateApplication(OAuthAppRequest oAuthAppRequest, HashMap additionalProperties) - throws APIManagementException { - OBKeyManagerExtensionInterface obKeyManagerExtensionImpl = KeyManagerUtil.getOBKeyManagerExtensionImpl(); - if (obKeyManagerExtensionImpl != null) { - obKeyManagerExtensionImpl.doPreCreateApplication(oAuthAppRequest, additionalProperties); - } - } - - /** - * Do changes to app request before updating the app at toolkit level. - * - * @param additionalProperties Values for additional property list defined in the config - * @throws APIManagementException when failed to validate a given property - */ - @Generated(message = "Excluding from code coverage since the method body is at toolkit") - public void doPreUpdateApplication(OAuthAppRequest oAuthAppRequest, HashMap additionalProperties, - ServiceProvider serviceProvider) throws APIManagementException { - OBKeyManagerExtensionInterface obKeyManagerExtensionImpl = KeyManagerUtil.getOBKeyManagerExtensionImpl(); - if (obKeyManagerExtensionImpl != null) { - obKeyManagerExtensionImpl.doPreUpdateApplication(oAuthAppRequest, additionalProperties, serviceProvider); - } - } - - /** - * Do changes to service provider before updating the service provider properties. - * - * @param oAuthConsumerAppDTO oAuth application DTO - * @param serviceProvider Service provider application - * @param isCreateApp Whether this function is called at app creation - * @throws APIManagementException when failed to validate a given property - */ - @Generated(message = "Excluding from code coverage since the method body is at toolkit") - public void doPreUpdateSpApp(org.wso2.carbon.identity.oauth.dto.OAuthConsumerAppDTO oAuthConsumerAppDTO, - ServiceProvider serviceProvider, - HashMap additionalProperties, boolean isCreateApp) - throws APIManagementException { - - OBKeyManagerExtensionInterface obKeyManagerExtensionImpl = KeyManagerUtil.getOBKeyManagerExtensionImpl(); - if (obKeyManagerExtensionImpl != null) { - obKeyManagerExtensionImpl.doPreUpdateSpApp(oAuthConsumerAppDTO, serviceProvider, additionalProperties, - isCreateApp); - } - } - - @Generated(message = "Added for unit testing purposes") - protected ApplicationManagementServiceImpl getApplicationMgmtServiceImpl() { - - return ApplicationManagementServiceImpl.getInstance(); - } - - @Generated(message = "Added for unit testing purposes") - protected OAuthAdminService getOAuthAdminService() { - return new OAuthAdminService(); - } - - protected ServiceProviderProperty getSpPropertyFromSPMetaData(String propertyName, - ServiceProviderProperty[] spProperties) { - - return Arrays.asList(spProperties).stream().filter(serviceProviderProperty -> serviceProviderProperty.getName() - .equalsIgnoreCase(propertyName)).findAny().orElse(null); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/internal/KeyManagerDataHolder.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/internal/KeyManagerDataHolder.java deleted file mode 100644 index 0e35ac0e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/internal/KeyManagerDataHolder.java +++ /dev/null @@ -1,158 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.keymanager.internal; - -import org.apache.axis2.AxisFault; -import org.wso2.carbon.apimgt.api.model.KeyManagerConnectorConfiguration; -import org.wso2.carbon.apimgt.impl.APIConstants; -import org.wso2.carbon.apimgt.impl.APIManagerConfigurationService; -import org.wso2.carbon.authenticator.stub.AuthenticationAdminStub; -import org.wso2.carbon.identity.oauth.stub.OAuthAdminServiceStub; -import org.wso2.carbon.user.core.service.RealmService; -import org.wso2.carbon.user.mgt.stub.UserAdminStub; - -import java.util.HashMap; -import java.util.Map; - -/** - * Data holder for key manager client extension. - */ -public class KeyManagerDataHolder { - - private APIManagerConfigurationService apiManagerConfigurationService; - private static volatile KeyManagerDataHolder instance; - private static final String IDENTITY_APPLICATION_MGT_SERVICE = "IdentityApplicationManagementService"; - public static final String AUTHENTICATION_ADMIN_SERVICE = "AuthenticationAdmin"; - public static final String USER_ADMIN_SERVICE = "UserAdmin"; - public static final String OAUTH_ADMIN_SERVICE = "OAuthAdminService"; - private AuthenticationAdminStub authenticationAdminStub; - private OAuthAdminServiceStub oAuthAdminServiceStub; - private UserAdminStub userAdminStub; - private String backendServerURL = ""; - private RealmService realmService; - private Map keyManagerConnectorConfigurationMap = new HashMap<>(); - - public static KeyManagerDataHolder getInstance() { - - if (instance == null) { - synchronized (KeyManagerDataHolder.class) { - if (instance == null) { - instance = new KeyManagerDataHolder(); - } - } - } - return instance; - } - - public UserAdminStub getUserAdminStub() throws AxisFault { - - if (userAdminStub == null) { - String userAdminServiceURL = backendServerURL + USER_ADMIN_SERVICE; - userAdminStub = new UserAdminStub(userAdminServiceURL); - } - return userAdminStub; - } - - public void setUserAdminStub(UserAdminStub userAdminStub) { - - this.userAdminStub = userAdminStub; - } - - public AuthenticationAdminStub getAuthenticationAdminStub() throws AxisFault { - - if (authenticationAdminStub == null) { - String authenticationServiceURL = backendServerURL + AUTHENTICATION_ADMIN_SERVICE; - authenticationAdminStub = new AuthenticationAdminStub(authenticationServiceURL); - } - - return authenticationAdminStub; - } - - public void setAuthenticationAdminStub(AuthenticationAdminStub authenticationAdminStub) { - - this.authenticationAdminStub = authenticationAdminStub; - } - - public OAuthAdminServiceStub getOauthAdminServiceStub() throws AxisFault { - - if (oAuthAdminServiceStub == null) { - String oauthAdminServiceURL = backendServerURL + OAUTH_ADMIN_SERVICE; - oAuthAdminServiceStub = new OAuthAdminServiceStub(oauthAdminServiceURL); - } - return oAuthAdminServiceStub; - } - - - public void setOauthAdminServiceStub(OAuthAdminServiceStub oAuthAdminServiceStub) { - - this.oAuthAdminServiceStub = oAuthAdminServiceStub; - } - - public void setApiManagerConfiguration(APIManagerConfigurationService apiManagerConfigurationService) { - - this.apiManagerConfigurationService = apiManagerConfigurationService; - backendServerURL = apiManagerConfigurationService.getAPIManagerConfiguration() - .getFirstProperty(APIConstants.API_KEY_VALIDATOR_URL); - - } - - public APIManagerConfigurationService getApiManagerConfigurationService() { - - return apiManagerConfigurationService; - } - - public String getBackendServerURL() { - - return backendServerURL; - } - - public void addKeyManagerConnectorConfiguration(String type, - KeyManagerConnectorConfiguration keyManagerConnectorConfiguration) { - - keyManagerConnectorConfigurationMap.put(type, keyManagerConnectorConfiguration); - } - - public void removeKeyManagerConnectorConfiguration(String type) { - - keyManagerConnectorConfigurationMap.remove(type); - } - - public KeyManagerConnectorConfiguration getKeyManagerConnectorConfiguration(String type) { - - return keyManagerConnectorConfigurationMap.get(type); - } - - public Map getKeyManagerConnectorConfigurations() { - - return keyManagerConnectorConfigurationMap; - } - - public RealmService getRealmService() { - - if (realmService == null) { - throw new RuntimeException("Realm Service is not available. Component did not start correctly."); - } - return realmService; - } - - void setRealmService(RealmService realmService) { - - this.realmService = realmService; - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/internal/KeyManagerServiceComponent.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/internal/KeyManagerServiceComponent.java deleted file mode 100644 index 993a8320..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/java/com/wso2/openbanking/accelerator/keymanager/internal/KeyManagerServiceComponent.java +++ /dev/null @@ -1,127 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.keymanager.internal; - -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.osgi.service.component.ComponentContext; -import org.osgi.service.component.annotations.Activate; -import org.osgi.service.component.annotations.Component; -import org.osgi.service.component.annotations.Deactivate; -import org.osgi.service.component.annotations.Reference; -import org.osgi.service.component.annotations.ReferenceCardinality; -import org.osgi.service.component.annotations.ReferencePolicy; -import org.wso2.carbon.apimgt.api.model.KeyManagerConnectorConfiguration; -import org.wso2.carbon.apimgt.impl.APIConstants; -import org.wso2.carbon.apimgt.impl.APIManagerConfigurationService; -import org.wso2.carbon.user.core.service.RealmService; - -import java.util.Map; - -/** - * Service component for key manager client. - */ -@Component( - name = "com.wso2.open.banking.keymanager", - immediate = true -) -public class KeyManagerServiceComponent { - - private static final Log log = LogFactory.getLog(KeyManagerServiceComponent.class); - - @Activate - protected void activate(ComponentContext context) { - - log.debug("Open banking key manager extension component is activated "); - } - - @Deactivate - protected void deactivate(ComponentContext context) { - - log.debug("Open banking key manager extension is deactivated "); - } - - @Reference( - service = APIManagerConfigurationService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unSetAPIMConfigs" - ) - public void setAPIMConfig(APIManagerConfigurationService apManagerConfigurationService) { - - KeyManagerDataHolder.getInstance().setApiManagerConfiguration(apManagerConfigurationService); - } - - public void unSetAPIMConfigs(APIManagerConfigurationService apManagerConfigurationService) { - - KeyManagerDataHolder.getInstance().setApiManagerConfiguration(apManagerConfigurationService); - } - - /** - * Initialize the KeyManager Connector configuration Service Service dependency. - * - * @param keyManagerConnectorConfiguration {@link KeyManagerConnectorConfiguration} service reference. - */ - @Reference( - name = "keyManager.connector.service", - service = KeyManagerConnectorConfiguration.class, - cardinality = ReferenceCardinality.MULTIPLE, - policy = ReferencePolicy.DYNAMIC, - unbind = "removeKeyManagerConnectorConfiguration") - protected void addKeyManagerConnectorConfiguration( - KeyManagerConnectorConfiguration keyManagerConnectorConfiguration) { - - KeyManagerDataHolder.getInstance() - .addKeyManagerConnectorConfiguration(keyManagerConnectorConfiguration.getType(), - keyManagerConnectorConfiguration); - - } - - /** - * De-reference the JWTTransformer service. - * - * @param keyManagerConnectorConfiguration - */ - protected void removeKeyManagerConnectorConfiguration( - KeyManagerConnectorConfiguration keyManagerConnectorConfiguration, Map properties) { - if (properties.containsKey(APIConstants.KeyManager.KEY_MANAGER_TYPE)) { - String type = (String) properties.get(APIConstants.KeyManager.KEY_MANAGER_TYPE); - KeyManagerDataHolder.getInstance().removeKeyManagerConnectorConfiguration(type); - } - } - - @Reference( - name = "realm.service", - service = RealmService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetRealmService" - ) - protected void setRealmService(RealmService realmService) { - - log.debug("Setting the Realm Service"); - KeyManagerDataHolder.getInstance().setRealmService(realmService); - } - - protected void unsetRealmService(RealmService realmService) { - - log.debug("UnSetting the Realm Service"); - KeyManagerDataHolder.getInstance().setRealmService(null); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/resources/findbugs-include.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/resources/findbugs-include.xml deleted file mode 100644 index f04bf7e4..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,34 +0,0 @@ - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/test/java/com/wso2/openbanking/accelerator/keymanager/KeyManagerTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/test/java/com/wso2/openbanking/accelerator/keymanager/KeyManagerTest.java deleted file mode 100644 index 565e0d5b..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/test/java/com/wso2/openbanking/accelerator/keymanager/KeyManagerTest.java +++ /dev/null @@ -1,521 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.keymanager; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.util.ServiceProviderUtils; -import com.wso2.openbanking.accelerator.keymanager.internal.KeyManagerDataHolder; -import org.apache.axis2.client.Options; -import org.apache.axis2.client.ServiceClient; -import org.apache.axis2.context.OperationContext; -import org.apache.axis2.context.ServiceContext; -import org.apache.axis2.transport.http.HTTPConstants; -import org.mockito.InjectMocks; -import org.mockito.Mock; -import org.mockito.Mockito; -import org.mockito.MockitoAnnotations; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.DataProvider; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; -import org.wso2.carbon.apimgt.api.APIManagementException; -import org.wso2.carbon.apimgt.api.model.AccessTokenInfo; -import org.wso2.carbon.apimgt.api.model.AccessTokenRequest; -import org.wso2.carbon.apimgt.api.model.ConfigurationDto; -import org.wso2.carbon.apimgt.api.model.KeyManagerConnectorConfiguration; -import org.wso2.carbon.apimgt.api.model.OAuthApplicationInfo; -import org.wso2.carbon.apimgt.impl.APIConstants; -import org.wso2.carbon.apimgt.impl.APIManagerConfiguration; -import org.wso2.carbon.apimgt.impl.APIManagerConfigurationService; -import org.wso2.carbon.authenticator.stub.AuthenticationAdminStub; -import org.wso2.carbon.authenticator.stub.LoginAuthenticationExceptionException; -import org.wso2.carbon.identity.application.common.IdentityApplicationManagementException; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.application.common.model.ServiceProviderProperty; -import org.wso2.carbon.identity.application.mgt.ApplicationManagementServiceImpl; -import org.wso2.carbon.identity.oauth.OAuthAdminService; -import org.wso2.carbon.identity.oauth.stub.OAuthAdminServiceIdentityOAuthAdminException; -import org.wso2.carbon.identity.oauth.stub.OAuthAdminServiceStub; -import org.wso2.carbon.identity.oauth.stub.dto.OAuthConsumerAppDTO; -import org.wso2.carbon.user.mgt.stub.UserAdminStub; - -import java.rmi.RemoteException; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Collections; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import static org.mockito.Matchers.anyString; -import static org.mockito.Mockito.doNothing; -import static org.mockito.Mockito.mock; -import static org.mockito.Mockito.spy; -import static org.powermock.api.mockito.PowerMockito.mockStatic; -import static org.powermock.api.mockito.PowerMockito.when; -/** - * Test class for KeyManager. - */ -@PrepareForTest({OpenBankingConfigParser.class, APIManagementException.class, ServiceProviderUtils.class}) -@PowerMockIgnore("jdk.internal.reflect.*") -public class KeyManagerTest extends PowerMockTestCase { - - @Mock - private OAuthAdminServiceStub oAuthAdminServiceStub; - - @Mock - private AuthenticationAdminStub authenticationAdminStub; - - @Mock - private UserAdminStub userAdminStub; - - @Mock - private KeyManagerDataHolder keyManagerDataHolder; - - @Mock - private ServiceClient serviceClient; - - @Mock - private OperationContext operationContext; - - @Mock - private ServiceContext serviceContext; - - @Mock - private APIManagerConfiguration config; - - @Mock - private APIManagerConfigurationService apiManagerConfigurationService; - - @InjectMocks - OBKeyManagerImpl obKeyManager = new OBKeyManagerImpl(); - - @Mock - OpenBankingConfigParser openBankingConfigParser; - - @Mock - ApplicationManagementServiceImpl applicationManagementServiceImpl; - - @Mock - OAuthAdminService oAuthAdminService; - - @Mock - ServiceProviderUtils serviceProviderUtils; - - @Mock - org.wso2.carbon.identity.application.common.model.ServiceProvider serviceProvider; - - @Mock - org.wso2.carbon.identity.oauth.dto.OAuthConsumerAppDTO oAuthConsumerAppDTO; - - String dummyPropertyName1 = "dummyName1"; - String dummyPropertyName2 = "dummyName2"; - String dummyValue1 = "dummyValue1"; - String dummyValue2 = "dummyValue2"; - - String defaultPropertyName1 = "defaultName1"; - String defaultPropertyName2 = "defaultName2"; - String defaultValue1 = "defaultValue1"; - String defaultValue2 = "defaultValue2"; - - String dummyString = "dummyString"; - - Map property = new HashMap<>(); - - @BeforeClass - public void init() { - - MockitoAnnotations.initMocks(this); - } - - @BeforeMethod() - public void before() { - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - openBankingConfigParser = PowerMockito.mock(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()) - .thenReturn(openBankingConfigParser); - - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - - @Test - public void testGetNewApplicationAccessToken() throws APIManagementException, RemoteException, - OAuthAdminServiceIdentityOAuthAdminException, - LoginAuthenticationExceptionException, IdentityApplicationManagementException { - - OBKeyManagerImpl obKeyManager = spy(new OBKeyManagerImplMock()); - OAuthConsumerAppDTO oAuthConsumerAppDTO = new OAuthConsumerAppDTO(); - oAuthConsumerAppDTO.setApplicationName("AppName"); - - AccessTokenRequest tokenRequest = new AccessTokenRequest(); - tokenRequest.setClientId("0001"); - - ServiceProvider serviceProvider = new ServiceProvider(); - ServiceProviderProperty serviceProviderProperty = new ServiceProviderProperty(); - serviceProviderProperty.setDisplayName(OpenBankingConstants.REGULATORY); - serviceProviderProperty.setName(OpenBankingConstants.REGULATORY); - serviceProviderProperty.setValue("true"); - ServiceProviderProperty[] spPropertyArray = new ServiceProviderProperty[1]; - spPropertyArray[0] = serviceProviderProperty; - serviceProvider.setSpProperties(spPropertyArray); - - KeyManagerDataHolder.getInstance().setUserAdminStub(userAdminStub); - Options userAdminOptions = new Options(); - userAdminOptions.setManageSession(true); - userAdminOptions.setProperty(org.apache.axis2.transport.http.HTTPConstants.COOKIE_STRING, "sessionCookie"); - - Mockito.when(userAdminStub._getServiceClient()).thenReturn(serviceClient); - Mockito.when(serviceClient.getOptions()).thenReturn(userAdminOptions); - Mockito.when(apiManagerConfigurationService.getAPIManagerConfiguration()).thenReturn(config); - Mockito.when(config.getFirstProperty(APIConstants.API_KEY_VALIDATOR_URL)).thenReturn("KmBackEndURL"); - KeyManagerDataHolder.getInstance().setApiManagerConfiguration(apiManagerConfigurationService); - KeyManagerDataHolder.getInstance().setAuthenticationAdminStub(authenticationAdminStub); - KeyManagerDataHolder.getInstance().setOauthAdminServiceStub(oAuthAdminServiceStub); - - AccessTokenRequest accessTokenRequest = new AccessTokenRequest(); - - Mockito.when(keyManagerDataHolder.getApiManagerConfigurationService()) - .thenReturn(apiManagerConfigurationService); - Mockito.when(apiManagerConfigurationService.getAPIManagerConfiguration()).thenReturn(config); - Mockito.when(config.getFirstProperty(APIConstants.API_KEY_VALIDATOR_USERNAME)).thenReturn("userName"); - Mockito.when(config.getFirstProperty(APIConstants.API_KEY_VALIDATOR_PASSWORD)).thenReturn("password"); - - Mockito.when(keyManagerDataHolder.getAuthenticationAdminStub()).thenReturn(authenticationAdminStub); - Mockito.when(authenticationAdminStub.login(anyString(), anyString(), anyString())).thenReturn(true); - Mockito.when(authenticationAdminStub._getServiceClient()).thenReturn(serviceClient); - Mockito.when(serviceClient.getLastOperationContext()).thenReturn(operationContext); - Mockito.when(operationContext.getServiceContext()).thenReturn(serviceContext); - Mockito.when(serviceContext.getProperty(HTTPConstants.COOKIE_STRING)).thenReturn("cookie"); - - Mockito.when(keyManagerDataHolder.getOauthAdminServiceStub()).thenReturn(oAuthAdminServiceStub); - Mockito.when(oAuthAdminServiceStub._getServiceClient()).thenReturn(serviceClient); - Mockito.when(oAuthAdminServiceStub.getOAuthApplicationData(anyString())) - .thenReturn(oAuthConsumerAppDTO); - - Mockito.when(obKeyManager.getApplicationMgmtServiceImpl()).thenReturn(applicationManagementServiceImpl); - Mockito.when(applicationManagementServiceImpl.getServiceProviderByClientId( - anyString(), anyString(), anyString())).thenReturn(serviceProvider); - AccessTokenInfo accessTokenInfo = obKeyManager.getNewApplicationAccessToken(accessTokenRequest); - Assert.assertTrue(accessTokenInfo == null); - - } - - @Test(description = "Add the values for additional properties defined in the config to oAuthApplicationInfo") - public void testUpdateAdditionalProperties() { - - Map> keyManagerAdditionalProperties = new HashMap<>(); - keyManagerAdditionalProperties.put(dummyPropertyName1, property); - keyManagerAdditionalProperties.put(dummyPropertyName2, property); - - when(openBankingConfigParser.getKeyManagerAdditionalProperties()).thenReturn(keyManagerAdditionalProperties); - spy(ServiceProvider.class); - - List spProperties = - new ArrayList<>(); - org.wso2.carbon.identity.application.common.model.ServiceProviderProperty spProperty1 = - new org.wso2.carbon.identity.application.common.model.ServiceProviderProperty(); - org.wso2.carbon.identity.application.common.model.ServiceProviderProperty spProperty2 = - new org.wso2.carbon.identity.application.common.model.ServiceProviderProperty(); - spProperty1.setName(dummyPropertyName1); - spProperty1.setValue(dummyValue1); - spProperty2.setName(dummyPropertyName2); - spProperty2.setValue(dummyValue2); - - spProperties.add(spProperty1); - spProperties.add(spProperty2); - - HashMap additionalProperties = new HashMap<>(); - additionalProperties.put(defaultPropertyName1, defaultValue1); - additionalProperties.put(defaultPropertyName2, defaultValue2); - - OAuthApplicationInfo oAuthApplicationInfo = new OAuthApplicationInfo(); - oAuthApplicationInfo.addParameter(APIConstants.JSON_ADDITIONAL_PROPERTIES, - additionalProperties); - oAuthApplicationInfo = obKeyManager.updateAdditionalProperties(oAuthApplicationInfo, spProperties); - - additionalProperties.put(dummyPropertyName1, dummyValue1); - additionalProperties.put(dummyPropertyName2, dummyValue2); - - Assert.assertEquals(additionalProperties, (HashMap) oAuthApplicationInfo. - getParameter(APIConstants.JSON_ADDITIONAL_PROPERTIES)); - } - - @Test - public void testUpdateSpProperties() throws Exception { - - OBKeyManagerImpl obKeyManager = spy(new OBKeyManagerImplMock()); - - Mockito.when(obKeyManager.getApplicationMgmtServiceImpl()).thenReturn(applicationManagementServiceImpl); - Mockito.when(obKeyManager.getOAuthAdminService()).thenReturn(oAuthAdminService); - Mockito.when(oAuthAdminService.getOAuthApplicationDataByAppName(Mockito.anyString())) - .thenReturn(oAuthConsumerAppDTO); - - org.wso2.carbon.identity.application.common.model.ServiceProviderProperty[] spProperties = - new org.wso2.carbon.identity.application.common.model.ServiceProviderProperty[2]; - org.wso2.carbon.identity.application.common.model.ServiceProviderProperty spProperty1 = - new org.wso2.carbon.identity.application.common.model.ServiceProviderProperty(); - org.wso2.carbon.identity.application.common.model.ServiceProviderProperty spProperty2 = - new org.wso2.carbon.identity.application.common.model.ServiceProviderProperty(); - spProperty1.setName(defaultPropertyName1); - spProperty1.setValue(defaultValue1); - spProperty2.setName(defaultPropertyName2); - spProperty2.setValue(defaultValue2); - - spProperties[0] = (spProperty1); - spProperties[1] = (spProperty2); - - ServiceProvider serviceProvider = spy(ServiceProvider.class); - doNothing().when(applicationManagementServiceImpl).updateApplication(Mockito.anyObject(), Mockito.anyString(), - Mockito.anyString()); - - serviceProvider.setSpProperties(spProperties); - serviceProvider.setApplicationName(dummyString); - - Mockito.when(applicationManagementServiceImpl.getServiceProvider(dummyString, dummyString)) - .thenReturn(serviceProvider); - - String overriddenDummyValue = "overriddenDummyValue"; - HashMap additionalProperties = new HashMap<>(); - additionalProperties.put(dummyPropertyName1, dummyValue1); - additionalProperties.put(dummyPropertyName2, dummyValue2); - additionalProperties.put(defaultPropertyName2, overriddenDummyValue); - - List updatedSpProperties = - new ArrayList<>(Arrays.asList(spProperties)); - org.wso2.carbon.identity.application.common.model.ServiceProviderProperty spProperty3 = - new org.wso2.carbon.identity.application.common.model.ServiceProviderProperty(); - org.wso2.carbon.identity.application.common.model.ServiceProviderProperty spProperty4 = - new org.wso2.carbon.identity.application.common.model.ServiceProviderProperty(); - org.wso2.carbon.identity.application.common.model.ServiceProviderProperty spProperty5 = - new org.wso2.carbon.identity.application.common.model.ServiceProviderProperty(); - spProperty3.setName(dummyPropertyName1); - spProperty3.setValue(dummyValue1); - spProperty4.setName(dummyPropertyName2); - spProperty4.setValue(dummyValue2); - spProperty5.setName(defaultPropertyName2); - spProperty5.setValue(overriddenDummyValue); - - updatedSpProperties.add(spProperty3); - updatedSpProperties.add(spProperty4); - updatedSpProperties.remove(spProperty2); - updatedSpProperties.add(spProperty5); - - org.wso2.carbon.identity.oauth.dto.OAuthConsumerAppDTO oAuthConsumerAppDTOdummy = - new org.wso2.carbon.identity.oauth.dto.OAuthConsumerAppDTO(); - ServiceProvider serviceProviderDummy = new ServiceProvider(); - HashMap dummyMap = new HashMap<>(); - - doNothing().when(obKeyManager).doPreUpdateSpApp(oAuthConsumerAppDTOdummy, serviceProviderDummy, dummyMap, - true); - - obKeyManager.updateSpProperties( - dummyString, dummyString, dummyString, additionalProperties, true); - - List - updatedSpPropertiesFromFunction = new ArrayList<>(Arrays.asList(serviceProvider.getSpProperties())); - - // Two sp property arrays have same sp property elements but as different object. Therefore, - // their comparison needs to be explicitly done - Assert.assertTrue(compareSpPropertyList(updatedSpProperties, updatedSpPropertiesFromFunction)); - } - - /** - * Compare two arraylist for equality of size and equality of the attributes of each object in the array. - * - * @param originalList original list - * @param comparedList compared list - * @return whether elements in the array are equal or not - */ - private Boolean compareSpPropertyList(List originalList, List comparedList) { - - if (originalList.size() != comparedList.size()) { - return false; - } else { - int equalElementCount = 0; - for (org.wso2.carbon.identity.application.common.model.ServiceProviderProperty originalProperty - : originalList) { - org.wso2.carbon.identity.application.common.model.ServiceProviderProperty updatedProperty - = comparedList.stream().filter(serviceProviderProperty -> serviceProviderProperty.getName() - .equalsIgnoreCase(originalProperty.getName())).findAny().orElse(null); - if (originalProperty.getValue() == updatedProperty.getValue()) { - equalElementCount++; - } - } - return originalList.size() == equalElementCount; - } - } - - @DataProvider - public Object[][] validateOAuthAppCreationPropertiesDataProvider() { - - Map> keyManagerAdditionalProperties = new HashMap<>(); - keyManagerAdditionalProperties.put(dummyPropertyName1, property); - keyManagerAdditionalProperties.put(dummyPropertyName2, property); - - Map> keyManagerWithoutAdditionalProperties = new HashMap<>(); - - List applicationConfigurationsList = new ArrayList(); - ConfigurationDto optionalApplicationConfiguration = new ConfigurationDto(dummyPropertyName1, - dummyPropertyName1, "", "", APIConstants.KeyManager.NOT_APPLICABLE_VALUE, true, false, - Collections.EMPTY_LIST, false); - ConfigurationDto mandatoryApplicationConfiguration = new ConfigurationDto(dummyPropertyName2, - dummyPropertyName2, "", "", APIConstants.KeyManager.NOT_APPLICABLE_VALUE, false, false, - Collections.EMPTY_LIST, false); - // Configurations defined in the config - applicationConfigurationsList - .add(optionalApplicationConfiguration); - applicationConfigurationsList - .add(mandatoryApplicationConfiguration); - - List applicationConfigurationsListWithCorrectDefaultValue = new ArrayList(); - ConfigurationDto applicationConfigurationWithCorrectDefaultValue = new ConfigurationDto(dummyPropertyName1, - dummyPropertyName1, "", "", "500", false, false, - Collections.EMPTY_LIST, false); - applicationConfigurationsListWithCorrectDefaultValue.add(applicationConfigurationWithCorrectDefaultValue); - - // Input values for properties from the UI - String inputJsonWithValuesForMandatoryProperties = - "{\"dummyName1\" : \"dummyValue1\" , \"dummyName2\" : \"dummyValue2\"}"; - String inputJsonWithoutValuesForMandatoryProperties = "{\"dummyName2\" : \"dummyValue2\"}"; - String inputNonJsonStringForAdditionalProperties = "dummy string"; - // Only N/A or numbers greater than zero are allowed for additional property values if they are not defined - // separately in config - String inputJsonWithIncorrectValueForDefaultProperties = - "{\"dummyName1\" : \"dummyValue1\" , \"dummyName2\" : \"dummyValue2\"}"; - String inputJsonWithCorrectValueForDefaultProperties = "{\"dummyName1\" : \"800\"}"; - String inputJsonWithEmptyValueForDefaultProperties = "{\"dummyName1\" : \"\"}"; - String inputJsonWithIncorrectNumberValueForDefaultProperties = "{\"dummyName1\" : \"-800\"}"; - String inputWithInvalidJsonString = "\"dummyName1\" : \"-800\""; - - return new Object[][]{ - {keyManagerAdditionalProperties, applicationConfigurationsList, - inputJsonWithValuesForMandatoryProperties, null}, - {keyManagerAdditionalProperties, applicationConfigurationsList, - inputJsonWithoutValuesForMandatoryProperties, APIManagementException.class}, - {keyManagerAdditionalProperties, applicationConfigurationsList, - inputWithInvalidJsonString, APIManagementException.class}, - {keyManagerAdditionalProperties, applicationConfigurationsList, - inputNonJsonStringForAdditionalProperties, APIManagementException.class}, - {keyManagerWithoutAdditionalProperties, applicationConfigurationsList, - inputJsonWithIncorrectValueForDefaultProperties, APIManagementException.class}, - {keyManagerWithoutAdditionalProperties, applicationConfigurationsList, - inputJsonWithIncorrectValueForDefaultProperties, APIManagementException.class}, - {keyManagerWithoutAdditionalProperties, applicationConfigurationsList, - inputJsonWithCorrectValueForDefaultProperties, null}, - {keyManagerWithoutAdditionalProperties, applicationConfigurationsList, - inputJsonWithCorrectValueForDefaultProperties, null}, - {keyManagerWithoutAdditionalProperties, applicationConfigurationsList, - inputJsonWithIncorrectNumberValueForDefaultProperties, APIManagementException.class}, - {keyManagerWithoutAdditionalProperties, applicationConfigurationsList, null, null}, - {keyManagerWithoutAdditionalProperties, applicationConfigurationsList, - inputJsonWithEmptyValueForDefaultProperties, null} - }; - } - - @Test(dataProvider = "validateOAuthAppCreationPropertiesDataProvider", - description = "Validate user inputs for application creation") - public void testValidateOAuthAppCreationProperties(Map> - keyManagerAdditionalProperties, - List applicationConfigurationsList, - String valuesForProperties, - Class exceptionType) { - - try { - Mockito.when(openBankingConfigParser.getKeyManagerAdditionalProperties()) - .thenReturn(keyManagerAdditionalProperties); - KeyManagerConnectorConfiguration keyManagerConnectorConfiguration = - mock(KeyManagerConnectorConfiguration.class); - KeyManagerDataHolder.getInstance().addKeyManagerConnectorConfiguration(obKeyManager.getType(), - keyManagerConnectorConfiguration); - - Mockito.when(keyManagerDataHolder.getKeyManagerConnectorConfiguration(obKeyManager.getType())) - .thenReturn(keyManagerConnectorConfiguration); - Mockito.when(keyManagerConnectorConfiguration.getApplicationConfigurations()) - .thenReturn(applicationConfigurationsList); - - String dummyString = "dummy"; - - mockStatic(ServiceProviderUtils.class); - Mockito.when(serviceProviderUtils.getSpTenantDomain(dummyString)).thenReturn(dummyString); - OBKeyManagerImpl obKeyManager = spy(new OBKeyManagerImplMock()); - - Mockito.when(obKeyManager.getApplicationMgmtServiceImpl()).thenReturn(applicationManagementServiceImpl); - Mockito.when(applicationManagementServiceImpl.getServiceProvider(dummyString, dummyString)) - .thenReturn(serviceProvider); - - OAuthApplicationInfo oAuthApplicationInfo = new OAuthApplicationInfo(); - oAuthApplicationInfo.addParameter(APIConstants.JSON_ADDITIONAL_PROPERTIES, - valuesForProperties); - - obKeyManager.validateOAuthAppCreationProperties(oAuthApplicationInfo); - Assert.assertTrue(exceptionType == null); - } catch (Exception e) { - Assert.assertEquals(e.getClass(), exceptionType); - } - } - - @Test - public void testGetSpPropertyFromSPMetaData() { - - org.wso2.carbon.identity.application.common.model.ServiceProviderProperty[] spProperties = - new org.wso2.carbon.identity.application.common.model.ServiceProviderProperty[2]; - org.wso2.carbon.identity.application.common.model.ServiceProviderProperty spProperty1 = - new org.wso2.carbon.identity.application.common.model.ServiceProviderProperty(); - org.wso2.carbon.identity.application.common.model.ServiceProviderProperty spProperty2 = - new org.wso2.carbon.identity.application.common.model.ServiceProviderProperty(); - spProperty1.setName(defaultPropertyName1); - spProperty1.setValue(defaultValue1); - spProperty2.setName(defaultPropertyName2); - spProperty2.setValue(defaultValue2); - - spProperties[0] = (spProperty1); - spProperties[1] = (spProperty2); - - org.wso2.carbon.identity.application.common.model.ServiceProviderProperty property = - obKeyManager.getSpPropertyFromSPMetaData(defaultPropertyName1, spProperties); - - Assert.assertTrue(property != null); - } - -} - -class OBKeyManagerImplMock extends OBKeyManagerImpl { - - @Override - protected OAuthAdminService getOAuthAdminService() { - - return mock(OAuthAdminService.class); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/test/java/com/wso2/openbanking/accelerator/keymanager/KeyManagerUtilTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/test/java/com/wso2/openbanking/accelerator/keymanager/KeyManagerUtilTest.java deleted file mode 100644 index 193c8064..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/test/java/com/wso2/openbanking/accelerator/keymanager/KeyManagerUtilTest.java +++ /dev/null @@ -1,202 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.keymanager; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.keymanager.internal.KeyManagerDataHolder; -import org.mockito.Mock; -import org.mockito.Mockito; -import org.mockito.MockitoAnnotations; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; -import org.wso2.carbon.apimgt.api.APIManagementException; -import org.wso2.carbon.apimgt.api.model.OAuthAppRequest; -import org.wso2.carbon.apimgt.api.model.OAuthApplicationInfo; -import org.wso2.carbon.apimgt.impl.APIConstants; -import org.wso2.carbon.apimgt.impl.APIManagerConfiguration; -import org.wso2.carbon.apimgt.impl.APIManagerConfigurationService; -import org.wso2.carbon.identity.core.util.IdentityTenantUtil; -import org.wso2.carbon.user.api.RealmConfiguration; -import org.wso2.carbon.user.api.UserRealm; -import org.wso2.carbon.user.api.UserStoreException; -import org.wso2.carbon.user.core.common.AbstractUserStoreManager; -import org.wso2.carbon.user.core.service.RealmService; - -import java.util.HashMap; -import java.util.Map; - -import static org.mockito.Mockito.when; -import static org.powermock.api.mockito.PowerMockito.mockStatic; -/** - * Test class for KeyManagerUtil. - */ -@PrepareForTest({OpenBankingConfigParser.class, KeyManagerDataHolder.class, IdentityTenantUtil.class}) -@PowerMockIgnore("jdk.internal.reflect.*") -public class KeyManagerUtilTest extends PowerMockTestCase { - - String dummyPropertyName1 = "dummyName1"; - String dummyPropertyName2 = "dummyName2"; - String dummyValue1 = "dummyValue1"; - String dummyValue2 = "dummyValue2"; - - Map property = new HashMap<>(); - - @Mock - OpenBankingConfigParser openBankingConfigParser; - - @Mock - RealmService realmService; - - @Mock - private KeyManagerDataHolder keyManagerDataHolder; - - @Mock - private APIManagerConfigurationService apiManagerConfigurationService; - - @Mock - private APIManagerConfiguration config; - - @Mock - private UserRealm userRealm; - - @Mock - private AbstractUserStoreManager abstractUserStoreManager; - - @Mock - private RealmConfiguration realmConfiguration; - - @BeforeClass - public void init() { - - MockitoAnnotations.initMocks(this); - } - - @BeforeMethod() - public void before() { - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - openBankingConfigParser = PowerMockito.mock(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()) - .thenReturn(openBankingConfigParser); - - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - - - @Test - public void getEmptyOBKeyManagerExtensionImplTest() throws APIManagementException { - - when(openBankingConfigParser.getOBKeyManagerExtensionImpl()) - .thenReturn(""); - Assert.assertNull(KeyManagerUtil.getOBKeyManagerExtensionImpl()); - } - - @Test - public void getCorrectOBKeyManagerExtensionImplTest() throws APIManagementException { - - when(openBankingConfigParser.getOBKeyManagerExtensionImpl()) - .thenReturn("com.wso2.openbanking.accelerator.keymanager.OBKeyManagerImpl"); - Assert.assertTrue(KeyManagerUtil.getOBKeyManagerExtensionImpl() instanceof OBKeyManagerImpl); - } - - @Test(description = "Get the value from the JSON input for the properties defined in the config") - public void testGetValuesForAdditionalProperties() throws Exception { - - Map> keyManagerAdditionalProperties = new HashMap<>(); - keyManagerAdditionalProperties.put(dummyPropertyName1, property); - keyManagerAdditionalProperties.put(dummyPropertyName2, property); - - PowerMockito.when(openBankingConfigParser.getKeyManagerAdditionalProperties()) - .thenReturn(keyManagerAdditionalProperties); - - HashMap result = new HashMap<>(); - result.put(dummyPropertyName1, dummyValue1); - result.put(dummyPropertyName2, dummyValue2); - - OAuthAppRequest oAuthAppRequest = new OAuthAppRequest(); - OAuthApplicationInfo oAuthApplicationInfo = new OAuthApplicationInfo(); - oAuthApplicationInfo.addParameter(APIConstants.JSON_ADDITIONAL_PROPERTIES, - "{\"dummyName1\" : \"dummyValue1\" , \"dummyName2\" : \"dummyValue2\"}"); - oAuthAppRequest.setOAuthApplicationInfo(oAuthApplicationInfo); - - Assert.assertEquals(result, KeyManagerUtil.getValuesForAdditionalProperties(oAuthAppRequest)); - } - - @Test(description = "Get the value from the invalid JSON input for the properties defined in the config") - public void testGetValuesForAdditionalPropertiesFailure() { - - Map> keyManagerAdditionalProperties = new HashMap<>(); - keyManagerAdditionalProperties.put(dummyPropertyName1, property); - keyManagerAdditionalProperties.put(dummyPropertyName2, property); - - PowerMockito.when(openBankingConfigParser.getKeyManagerAdditionalProperties()) - .thenReturn(keyManagerAdditionalProperties); - - HashMap result = new HashMap<>(); - result.put(dummyPropertyName1, dummyValue1); - result.put(dummyPropertyName2, dummyValue2); - - OAuthAppRequest oAuthAppRequest = new OAuthAppRequest(); - OAuthApplicationInfo oAuthApplicationInfo = new OAuthApplicationInfo(); - oAuthApplicationInfo.addParameter(APIConstants.JSON_ADDITIONAL_PROPERTIES, - "\"dummyName1\" : \"dummyValue1\" , \"dummyName2\" : \"dummyValue2\""); - oAuthAppRequest.setOAuthApplicationInfo(oAuthApplicationInfo); - try { - KeyManagerUtil.getValuesForAdditionalProperties(oAuthAppRequest); - } catch (Exception e) { - Assert.assertEquals(e.getClass(), APIManagementException.class); - } - - } - - @Test(description = "Add existing role to admin") - private void testAddExistingApplicationRoleToAdmin() throws UserStoreException, APIManagementException { - - int dummyTenantId = 1; - mockStatic(IdentityTenantUtil.class); - mockStatic(KeyManagerDataHolder.class); - PowerMockito.when(IdentityTenantUtil.getTenantId(Mockito.anyString())).thenReturn(dummyTenantId); - - when(KeyManagerDataHolder.getInstance()).thenReturn(keyManagerDataHolder); - Mockito.when(keyManagerDataHolder.getApiManagerConfigurationService()) - .thenReturn(apiManagerConfigurationService); - Mockito.when(apiManagerConfigurationService.getAPIManagerConfiguration()).thenReturn(config); - Mockito.when(config.getFirstProperty(APIConstants.API_KEY_VALIDATOR_USERNAME)).thenReturn("userName"); - - PowerMockito.when(keyManagerDataHolder.getRealmService()).thenReturn(realmService); - PowerMockito.when(realmService.getTenantUserRealm(Mockito.anyInt())).thenReturn(userRealm); - PowerMockito.when(userRealm.getUserStoreManager()).thenReturn(abstractUserStoreManager); - - Mockito.when(abstractUserStoreManager.isUserInRole(Mockito.anyString(), Mockito.anyString())).thenReturn(true); - KeyManagerUtil.addApplicationRoleToAdmin("dummy"); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/test/resources/testng.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/test/resources/testng.xml deleted file mode 100644 index b62f8f57..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.keymanager/src/test/resources/testng.xml +++ /dev/null @@ -1,28 +0,0 @@ - - - - - - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.runtime/com.wso2.openbanking.accelerator.runtime.identity.authn.filter/pom.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.runtime/com.wso2.openbanking.accelerator.runtime.identity.authn.filter/pom.xml deleted file mode 100644 index 1b934f36..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.runtime/com.wso2.openbanking.accelerator.runtime.identity.authn.filter/pom.xml +++ /dev/null @@ -1,105 +0,0 @@ - - - - - com.wso2.openbanking.accelerator.runtime - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../pom.xml - - - 4.0.0 - com.wso2.openbanking.accelerator.runtime.identity.authn.filter - jar - WSO2 Open Banking - Identity Authentication Filter - Proxy Filter Which Invokes OAuth Client Authenticators - - - - org.apache.cxf - cxf-core - - - org.apache.cxf - cxf-rt-frontend-jaxrs - - - org.wso2.carbon.identity.inbound.auth.oauth2 - org.wso2.carbon.identity.oauth - provided - - - org.wso2.carbon.identity.inbound.auth.oauth2 - org.wso2.carbon.identity.oauth.client.authn.filter - provided - - - org.springframework - spring-web - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.identity - provided - - - - - - - org.apache.maven.plugins - maven-compiler-plugin - - 8 - 8 - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - true - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.runtime/com.wso2.openbanking.accelerator.runtime.identity.authn.filter/src/main/java/com/wso2/openbanking/accelerator/runtime/identity/authn/filter/OBOAuthClientAuthenticatorProxy.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.runtime/com.wso2.openbanking.accelerator.runtime.identity.authn.filter/src/main/java/com/wso2/openbanking/accelerator/runtime/identity/authn/filter/OBOAuthClientAuthenticatorProxy.java deleted file mode 100644 index e38d8398..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.runtime/com.wso2.openbanking.accelerator.runtime.identity.authn.filter/src/main/java/com/wso2/openbanking/accelerator/runtime/identity/authn/filter/OBOAuthClientAuthenticatorProxy.java +++ /dev/null @@ -1,106 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.runtime.identity.authn.filter; - -import com.wso2.openbanking.accelerator.identity.common.IdentityServiceExporter; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.cxf.jaxrs.impl.MetadataMap; -import org.apache.cxf.message.Message; -import org.wso2.carbon.identity.oauth.client.authn.filter.OAuthClientAuthenticatorProxy; -import org.wso2.carbon.identity.oauth.common.OAuth2ErrorCodes; -import org.wso2.carbon.identity.oauth.common.OAuthConstants; -import org.wso2.carbon.identity.oauth2.bean.OAuthClientAuthnContext; -import org.wso2.carbon.identity.oauth2.client.authentication.OAuthClientAuthnService; - -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import javax.servlet.http.HttpServletRequest; - -/** - * JAX-RS interceptor which intercepts requests. This interceptor will act as a proxy for OAuth2 Client Authenticators. - * This will pick correct authenticator which can handle OAuth client authentication and engage it. - */ -public class OBOAuthClientAuthenticatorProxy extends OAuthClientAuthenticatorProxy { - - private static final Log log = LogFactory.getLog(OBOAuthClientAuthenticatorProxy.class); - private static final String HTTP_REQUEST = "HTTP.REQUEST"; - private OAuthClientAuthnService oAuthClientAuthnService; - - /** - * Handles the incoming JAX-RS message for the purpose of OAuth2 client authentication. - * - * @param message JAX-RS message - */ - @Override - public void handleMessage(Message message) { - - Map bodyContentParams = getContentParams(message); - HttpServletRequest request = ((HttpServletRequest) message.get(HTTP_REQUEST)); - if (oAuthClientAuthnService == null) { - oAuthClientAuthnService = IdentityServiceExporter.getOAuthClientAuthnService(); - } - OAuthClientAuthnContext oAuthClientAuthnContext = oAuthClientAuthnService.authenticateClient(request, - bodyContentParams); - if (!oAuthClientAuthnContext.isPreviousAuthenticatorEngaged()) { - oAuthClientAuthnContext.setErrorCode(OAuth2ErrorCodes.INVALID_CLIENT); - oAuthClientAuthnContext.setErrorMessage("Unsupported client authentication mechanism"); - } - setContextToRequest(request, oAuthClientAuthnContext); - } - - /** - * Retrieve body content as a String, List map. - * - * @param message JAX-RS incoming message - * @return Body parameter of the incoming request message - */ - protected Map getContentParams(Message message) { - - Map contentMap = new HashMap<>(); - List contentList = message.getContent(List.class); - contentList.forEach(item -> { - if (item instanceof MetadataMap) { - MetadataMap metadataMap = (MetadataMap) item; - metadataMap.forEach((key, value) -> { - if (key instanceof String && value instanceof List) { - contentMap.put((String) key, (List) value); - } - }); - } - }); - return contentMap; - } - - /** - * Set client authentication context to the request. - * - * @param request - Request - * @param oAuthClientAuthnContext - Context - */ - private void setContextToRequest(HttpServletRequest request, OAuthClientAuthnContext oAuthClientAuthnContext) { - - log.debug("Setting OAuth client authentication context to request"); - request.setAttribute(OAuthConstants.CLIENT_AUTHN_CONTEXT, - oAuthClientAuthnContext); - } - -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.runtime/com.wso2.openbanking.accelerator.runtime.identity.authn.filter/src/main/resources/findbugs-include.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.runtime/com.wso2.openbanking.accelerator.runtime.identity.authn.filter/src/main/resources/findbugs-include.xml deleted file mode 100644 index 649d044e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.runtime/com.wso2.openbanking.accelerator.runtime.identity.authn.filter/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,23 +0,0 @@ - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.runtime/pom.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.runtime/pom.xml deleted file mode 100644 index edc3b4c5..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.runtime/pom.xml +++ /dev/null @@ -1,36 +0,0 @@ - - - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../pom.xml - - - 4.0.0 - com.wso2.openbanking.accelerator.runtime - WSO2 Open Banking - Runtime Components - pom - - - com.wso2.openbanking.accelerator.runtime.identity.authn.filter - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/pom.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/pom.xml deleted file mode 100644 index b80a60de..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/pom.xml +++ /dev/null @@ -1,191 +0,0 @@ - - - - 4.0.0 - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../pom.xml - - - com.wso2.openbanking.accelerator.service.activator - bundle - WSO2 Open Banking - Service Activator Component - - - - org.wso2.carbon - org.wso2.carbon.core - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - - - - - org.testng - testng - test - - - org.mockito - mockito-all - test - - - - - - - org.apache.maven.plugins - maven-compiler-plugin - - 1.8 - 1.8 - - - - - org.jacoco - jacoco-maven-plugin - - - - **/ServiceRegisterComponent.class - - - - - default-prepare-agent - - prepare-agent - - - - default-prepare-agent-integration - - prepare-agent-integration - - - - default-report - - report - - - - default-report-integration - - report-integration - - - - default-check - - check - - - - - BUNDLE - - - INSTRUCTION - COVEREDRATIO - 0.80 - - - - - - - - - - - org.apache.maven.plugins - maven-surefire-plugin - - - src/test/resources/testng.xml - - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - true - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - - org.apache.felix - maven-bundle-plugin - true - - - - ${project.artifactId} - - - com.wso2.openbanking.accelerator.service.activator.internal, - - - org.osgi.framework; version="${osgi.framework.imp.pkg.version.range}", - org.osgi.service.component; version="${osgi.service.component.imp.pkg.version.range}", - - - !com.wso2.openbanking.accelerator.service.activator.internal, - com.wso2.openbanking.accelerator.service.activator.*; version="${project.version}", - - * - <_dsannotations>* - - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/main/java/com/wso2/openbanking/accelerator/service/activator/OBServiceObserver.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/main/java/com/wso2/openbanking/accelerator/service/activator/OBServiceObserver.java deleted file mode 100644 index a274c661..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/main/java/com/wso2/openbanking/accelerator/service/activator/OBServiceObserver.java +++ /dev/null @@ -1,30 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.service.activator; - -/** - * OBServiceObserver - *

- * When the ServiceActivator OSGI bundle is activated, the implementations of OBServiceObserver interface - * will get notified. Once implemented add the FQN of the implemented class to the configuration. - */ -public interface OBServiceObserver { - - void activate(); -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/main/java/com/wso2/openbanking/accelerator/service/activator/internal/ServiceObservable.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/main/java/com/wso2/openbanking/accelerator/service/activator/internal/ServiceObservable.java deleted file mode 100644 index f5c881d6..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/main/java/com/wso2/openbanking/accelerator/service/activator/internal/ServiceObservable.java +++ /dev/null @@ -1,64 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.service.activator.internal; - -import com.wso2.openbanking.accelerator.service.activator.OBServiceObserver; - -import java.util.ArrayList; -import java.util.List; -import java.util.Objects; - -/** - * ServiceObservable. - *

- * Maintains one-to-many dependency with OBServiceObservers in such a way that whenever ServiceActivator - * OSGI bundle changes its status to ACTIVE, the dependents get notified - */ -public class ServiceObservable { - - private static volatile ServiceObservable instance; - private final List obServiceObservers; - - private ServiceObservable() { - this.obServiceObservers = new ArrayList<>(); - } - - public static ServiceObservable getInstance() { - if (instance == null) { - synchronized (ServiceObservable.class) { - if (instance == null) { - instance = new ServiceObservable(); - } - } - } - return instance; - } - - public synchronized void registerServiceObserver(OBServiceObserver obServiceObserver) { - this.obServiceObservers.add(obServiceObserver); - } - - public synchronized void activateAllServiceObservers() { - this.obServiceObservers.parallelStream() - .filter(Objects::nonNull) - .forEach(OBServiceObserver::activate); - - this.obServiceObservers.clear(); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/main/java/com/wso2/openbanking/accelerator/service/activator/internal/ServiceRegisterComponent.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/main/java/com/wso2/openbanking/accelerator/service/activator/internal/ServiceRegisterComponent.java deleted file mode 100644 index 656e196f..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/main/java/com/wso2/openbanking/accelerator/service/activator/internal/ServiceRegisterComponent.java +++ /dev/null @@ -1,71 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.service.activator.internal; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.service.activator.OBServiceObserver; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.osgi.service.component.ComponentContext; -import org.osgi.service.component.annotations.Activate; -import org.osgi.service.component.annotations.Component; -import org.osgi.service.component.annotations.Deactivate; - -import java.util.Objects; - -/** - * ServiceRegisterComponent. - * - * OSGI Component class to register and activate subscriber (observer) classes - */ -@Component -public class ServiceRegisterComponent { - - private static final Log LOG = LogFactory.getLog(ServiceRegisterComponent.class); - - @Activate - protected void activate(ComponentContext context) { - ServiceObservable serviceObservable = ServiceObservable.getInstance(); - - OpenBankingConfigParser.getInstance().getServiceActivatorSubscribers() - .stream() - .map(this::getInstanceFromFQN) - .filter(Objects::nonNull) - .forEach(serviceObservable::registerServiceObserver); - - serviceObservable.activateAllServiceObservers(); - LOG.debug("All OB service observers are activated"); - } - - @Deactivate - protected void deactivate(ComponentContext context) { - LOG.debug("Metadata Updater bundle is deactivated"); - } - - private OBServiceObserver getInstanceFromFQN(String fqn) { - try { - return (OBServiceObserver) Class.forName(fqn).newInstance(); - } catch (ClassNotFoundException e) { - LOG.error("Unable to find the OBServiceObserver class implementation", e); - } catch (InstantiationException | IllegalAccessException e) { - LOG.error("Error occurred while loading the OBServiceObserver class implementation", e); - } - return null; - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/main/resources/findbugs-include.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/main/resources/findbugs-include.xml deleted file mode 100644 index 8932a22e..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/test/java/com/wso2/openbanking/accelerator/service/activator/internal/ServiceObservableTest.java b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/test/java/com/wso2/openbanking/accelerator/service/activator/internal/ServiceObservableTest.java deleted file mode 100644 index 7e4a8d79..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/test/java/com/wso2/openbanking/accelerator/service/activator/internal/ServiceObservableTest.java +++ /dev/null @@ -1,54 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.service.activator.internal; - -import com.wso2.openbanking.accelerator.service.activator.OBServiceObserver; -import org.mockito.Mockito; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -/** - * ServiceObservableTest. - *

- * Tests for ServiceObservable class - */ -public class ServiceObservableTest { - - ServiceObservable uut; - - @BeforeClass - public void init() { - uut = ServiceObservable.getInstance(); - } - - @Test - public void testActivateAllServiceObservers() { - OBServiceObserver obServiceObserverMock = Mockito.mock(OBServiceObserver.class); - Mockito.doNothing().when(obServiceObserverMock).activate(); - OBServiceObserver obServiceObserverMock1 = Mockito.mock(OBServiceObserver.class); - Mockito.doNothing().when(obServiceObserverMock1).activate(); - - uut.registerServiceObserver(obServiceObserverMock); - uut.registerServiceObserver(obServiceObserverMock1); - uut.activateAllServiceObservers(); - - Mockito.verify(obServiceObserverMock, Mockito.times(1)).activate(); - Mockito.verify(obServiceObserverMock1, Mockito.times(1)).activate(); - } -} diff --git a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/test/resources/testng.xml b/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/test/resources/testng.xml deleted file mode 100644 index 77515f81..00000000 --- a/open-banking-accelerator/components/com.wso2.openbanking.accelerator.service.activator/src/test/resources/testng.xml +++ /dev/null @@ -1,27 +0,0 @@ - - - - - - - - - - - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/pom.xml b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/pom.xml deleted file mode 100644 index 61986ede..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/pom.xml +++ /dev/null @@ -1,251 +0,0 @@ - - - - 4.0.0 - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../../pom.xml - - - com.wso2.openbanking.accelerator.consent.extensions - bundle - WSO2 Open Banking - Consent Extensions - - - - net.minidev - json-smart - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.identity - - - commons-logging - commons-logging - - - org.testng - testng - test - - - org.mockito - mockito-all - test - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.consent.service - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.consent.dao - provided - - - org.quartz-scheduler.wso2 - quartz - ${quartz.version} - - - org.powermock - powermock-api-mockito - - - org.powermock - powermock-module-testng - - - org.wso2.carbon.identity.local.auth.api - org.wso2.carbon.identity.local.auth.api.core - - - org.slf4j - slf4j-api - - - provided - - - org.wso2.carbon.identity.outbound.auth.push - org.wso2.carbon.identity.application.authenticator.push.common - - - org.wso2.carbon.identity.outbound.auth.push - org.wso2.carbon.identity.application.authenticator.push - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.event.notifications.service - - - - - - - org.apache.maven.plugins - maven-compiler-plugin - - - org.jacoco - jacoco-maven-plugin - - - - **/*Constants.class - **/*Component.class - **/*DataHolder.class - **/*Exception.class - **/*Step.class - **/*Data.class - **/*Builder.class - **/*ConsentExtensionExporter.class - **/*ConsentValidationResult.class - **/*ConsentValidator.class - **/*Default.class - - - - - default-prepare-agent - - prepare-agent - - - - default-prepare-agent-integration - - prepare-agent-integration - - - - default-report - - report - - - - default-report-integration - - report-integration - - - - default-check - - check - - - - - BUNDLE - - - INSTRUCTION - COVEREDRATIO - - - - - - - - - - - org.apache.maven.plugins - maven-surefire-plugin - - - src/test/resources/testng.xml - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - false - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-exclude.xml - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - org.apache.felix - maven-bundle-plugin - true - - - - ${project.artifactId} - - - com.wso2.openbanking.accelerator.consent.extensions.internal - - - com.wso2.openbanking.accelerator.common.*;version="${project.version}", - org.osgi.framework;version="${osgi.framework.imp.pkg.version.range}", - org.osgi.service.component;version="${osgi.service.component.imp.pkg.version.range}", - net.minidev.json.*;version="${json-smart}", - javax.servlet.http; version="${imp.pkg.version.javax.servlet}", - com.wso2.openbanking.accelerator.consent.mgt.service.*;version="${project.version}", - - - !com.wso2.openbanking.accelerator.consent.extensions.internal, - com.wso2.openbanking.accelerator.consent.extensions.*;version="${project.version}", - - - * - <_dsannotations>* - - - - - - - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/admin/builder/ConsentAdminBuilder.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/admin/builder/ConsentAdminBuilder.java deleted file mode 100644 index d2ce9137..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/admin/builder/ConsentAdminBuilder.java +++ /dev/null @@ -1,48 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.admin.builder; - -import com.wso2.openbanking.accelerator.common.util.OpenBankingUtils; -import com.wso2.openbanking.accelerator.consent.extensions.admin.model.ConsentAdminHandler; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -/** - * Builder class for consent admin handler. - */ -public class ConsentAdminBuilder { - - private static final Log log = LogFactory.getLog(ConsentAdminBuilder.class); - private ConsentAdminHandler consentAdminHandler = null; - private static String adminBuilderConfigPath = "Consent.AdminHandler"; - - public void build() { - - String handlerConfig = (String) ConsentExtensionsDataHolder.getInstance().getOpenBankingConfigurationService(). - getConfigurations().get(adminBuilderConfigPath); - consentAdminHandler = (ConsentAdminHandler) OpenBankingUtils.getClassInstanceFromFQN(handlerConfig); - - log.debug("Admin handler loaded successfully"); - } - - public ConsentAdminHandler getConsentAdminHandler() { - return consentAdminHandler; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/admin/impl/DefaultConsentAdminHandler.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/admin/impl/DefaultConsentAdminHandler.java deleted file mode 100644 index 85c236a8..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/admin/impl/DefaultConsentAdminHandler.java +++ /dev/null @@ -1,417 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.admin.impl; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.consent.extensions.admin.model.ConsentAdminData; -import com.wso2.openbanking.accelerator.consent.extensions.admin.model.ConsentAdminHandler; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionUtils; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import com.wso2.openbanking.accelerator.consent.extensions.util.jobs.ExpiredConsentStatusUpdateJob; -import com.wso2.openbanking.accelerator.consent.extensions.util.jobs.RetentionDatabaseSyncJob; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentFile; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentHistoryResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentStatusAuditRecord; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.ConsentCoreService; -import com.wso2.openbanking.accelerator.consent.mgt.service.constants.ConsentCoreServiceConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.handler.EventNotificationPersistenceServiceHandler; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Map; - -/** - * Consent admin handler default implementation. - */ -public class DefaultConsentAdminHandler implements ConsentAdminHandler { - private static final Log log = LogFactory.getLog(DefaultConsentAdminHandler.class); - private static final String AUTHORISED = "authorised"; - private static final String FETCH_FROM_RETENTION_DB_QUERY_PARAM = "fetchFromRetentionDatabase"; - - @Override - public void handleSearch(ConsentAdminData consentAdminData) throws ConsentException { - - JSONObject response = new JSONObject(); - - ArrayList consentIDs = null; - ArrayList clientIDs = null; - ArrayList consentTypes = null; - ArrayList consentStatuses = null; - ArrayList userIDs = null; - Long fromTime = null; - Long toTime = null; - Integer limit = null; - Integer offset = null; - boolean fetchFromRetentionDatabase = false; - - Map queryParams = consentAdminData.getQueryParams(); - - if (validateAndGetQueryParam(queryParams, "consentIDs") != null) { - consentIDs = new ArrayList<>(Arrays.asList(validateAndGetQueryParam(queryParams, "consentIDs"). - split(","))); - } - if (validateAndGetQueryParam(queryParams, "clientIDs") != null) { - clientIDs = new ArrayList<>(Arrays.asList(validateAndGetQueryParam(queryParams, "clientIDs"). - split(","))); - } - if (validateAndGetQueryParam(queryParams, "consentTypes") != null) { - consentTypes = new ArrayList<>(Arrays.asList(validateAndGetQueryParam(queryParams, "consentTypes"). - split(","))); - } - if (validateAndGetQueryParam(queryParams, "consentStatuses") != null) { - consentStatuses = new ArrayList<>(Arrays.asList(validateAndGetQueryParam(queryParams, "consentStatuses"). - split(","))); - - } - if (validateAndGetQueryParam(queryParams, "userIDs") != null) { - userIDs = new ArrayList<>(Arrays.asList(validateAndGetQueryParam(queryParams, "userIDs"). - split(","))); - } - if (validateAndGetQueryParam(queryParams, "fromTime") != null) { - try { - fromTime = Long.parseLong(validateAndGetQueryParam(queryParams, "fromTime")); - } catch (NumberFormatException e) { - log.error("Number format incorrect in search for parameter fromTime. Ignoring parameter"); - } - } - if (validateAndGetQueryParam(queryParams, "toTime") != null) { - try { - toTime = Long.parseLong(validateAndGetQueryParam(queryParams, "toTime")); - } catch (NumberFormatException e) { - log.error("Number format incorrect in search for parameter toTime. Ignoring parameter"); - } - } - if (validateAndGetQueryParam(queryParams, "limit") != null) { - try { - limit = Integer.parseInt(validateAndGetQueryParam(queryParams, "limit")); - } catch (NumberFormatException e) { - log.error("Number format incorrect in search for parameter limit. Ignoring parameter"); - } - } - if (validateAndGetQueryParam(queryParams, "offset") != null) { - try { - offset = Integer.parseInt(validateAndGetQueryParam(queryParams, "offset")); - } catch (NumberFormatException e) { - log.error("Number format incorrect in search for parameter offset. Ignoring parameter"); - } - } - if (validateAndGetQueryParam(queryParams, FETCH_FROM_RETENTION_DB_QUERY_PARAM) != null) { - fetchFromRetentionDatabase = Boolean.parseBoolean(validateAndGetQueryParam(queryParams, - FETCH_FROM_RETENTION_DB_QUERY_PARAM)); - } - int count, total = 0; - - try { - ArrayList results = ConsentExtensionsDataHolder.getInstance() - .getConsentCoreService().searchDetailedConsents(consentIDs, clientIDs, consentTypes, - consentStatuses, userIDs, fromTime, toTime, limit, offset, fetchFromRetentionDatabase); - JSONArray searchResults = new JSONArray(); - for (DetailedConsentResource result : results) { - searchResults.add(ConsentExtensionUtils.detailedConsentToJSON(result)); - } - response.appendField("data", searchResults); - count = searchResults.size(); - total = results.size(); - } catch (ConsentManagementException e) { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, e.getMessage()); - } - - //retrieve the total of the data set queried - if (limit != null || offset != null) { - try { - ArrayList results = ConsentExtensionsDataHolder.getInstance() - .getConsentCoreService().searchDetailedConsents(consentIDs, clientIDs, consentTypes, - consentStatuses, userIDs, fromTime, toTime, null, null, fetchFromRetentionDatabase); - total = results.size(); - } catch (ConsentManagementException e) { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, e.getMessage()); - } - } - - JSONObject metadata = new JSONObject(); - metadata.appendField("count", count); - metadata.appendField("offset", offset); - metadata.appendField("limit", limit); - metadata.appendField("total", total); - - response.appendField("metadata", metadata); - consentAdminData.setResponseStatus(ResponseStatus.OK); - consentAdminData.setResponsePayload(response); - } - - private String validateAndGetQueryParam(Map queryParams, String key) { - if (queryParams.containsKey(key) && (((ArrayList) queryParams.get(key)).get(0) instanceof String)) { - return (String) ((ArrayList) queryParams.get(key)).get(0); - } - return null; - } - - @Override - public void handleRevoke(ConsentAdminData consentAdminData) throws ConsentException { - - try { - Map queryParams = consentAdminData.getQueryParams(); - - String consentId = validateAndGetQueryParam(queryParams, "consentID"); - if (consentId == null) { - throw new ConsentException(ResponseStatus.BAD_REQUEST, "Mandatory parameter consent ID not available"); - } else { - ConsentResource consentResource = ConsentExtensionsDataHolder.getInstance().getConsentCoreService() - .getConsent(consentId, false); - - if (!AUTHORISED.equalsIgnoreCase(consentResource.getCurrentStatus())) { - throw new ConsentException(ResponseStatus.BAD_REQUEST, - "Consent is not in a revocable status"); - } else { - boolean success = ConsentExtensionsDataHolder.getInstance().getConsentCoreService() - .revokeConsentWithReason(validateAndGetQueryParam(queryParams, "consentID"), "revoked", - validateAndGetQueryParam(queryParams, "userID"), - ConsentCoreServiceConstants.CONSENT_REVOKE_FROM_DASHBOARD_REASON); - if (success) { - // persist a new notification to the DB - // This is a sample event notification persisting. This can be modified in the Toolkit level - if (OpenBankingConfigParser.getInstance().isRealtimeEventNotificationEnabled()) { - JSONObject notificationInfo = new JSONObject(); - notificationInfo.put("consentID", consentId); - notificationInfo.put("status", "Consent Revocation"); - notificationInfo.put("timeStamp", System.currentTimeMillis()); - EventNotificationPersistenceServiceHandler.getInstance().persistRevokeEvent( - consentResource.getClientID(), consentId, - "Consent Revocation", notificationInfo); - } - } - } - } - consentAdminData.setResponseStatus(ResponseStatus.OK); - consentAdminData.setResponseStatus(ResponseStatus.NO_CONTENT); - } catch (ConsentManagementException e) { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - "Exception occurred while revoking consents"); - } - } - - public void handleConsentAmendmentHistoryRetrieval(ConsentAdminData consentAdminData) throws ConsentException { - - JSONObject response = new JSONObject(); - String consentID = null; - Map queryParams = consentAdminData.getQueryParams(); - - if (validateAndGetQueryParam(queryParams, "consentId") != null) { - consentID = validateAndGetQueryParam(queryParams, "consentId"); - } - - if (StringUtils.isBlank(consentID)) { - log.error("Request missing the mandatory query parameter consentId"); - throw new ConsentException(ResponseStatus.BAD_REQUEST, "Mandatory query parameter consentId " + - "not available"); - } - - int count = 0; - - try { - ConsentCoreService consentCoreService = ConsentExtensionsDataHolder.getInstance().getConsentCoreService(); - Map results = consentCoreService.getConsentAmendmentHistoryData(consentID); - - JSONArray consentHistory = new JSONArray(); - for (Map.Entry result : results.entrySet()) { - JSONObject consentResourceJSON = new JSONObject(); - ConsentHistoryResource consentHistoryResource = result.getValue(); - DetailedConsentResource detailedConsentHistory = consentHistoryResource.getDetailedConsentResource(); - consentResourceJSON.appendField("historyId", result.getKey()); - consentResourceJSON.appendField("amendedReason", consentHistoryResource.getReason()); - consentResourceJSON.appendField("amendedTime", detailedConsentHistory.getUpdatedTime()); - consentResourceJSON.appendField("consentData", - ConsentExtensionUtils.detailedConsentToJSON(detailedConsentHistory)); - consentHistory.add(consentResourceJSON); - } - response.appendField("consentID", consentID); - response.appendField("currentConsent", - ConsentExtensionUtils.detailedConsentToJSON(consentCoreService.getDetailedConsent(consentID))); - response.appendField("consentAmendmentHistory", consentHistory); - count = consentHistory.size(); - } catch (ConsentManagementException e) { - log.error("Error while retrieving consent amendment history data", e); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, e.getMessage()); - } - - JSONObject metadata = new JSONObject(); - metadata.appendField("amendmentCount", count); - response.appendField("metadata", metadata); - consentAdminData.setResponseStatus(ResponseStatus.OK); - consentAdminData.setResponsePayload(response); - } - - @Override - public void handleConsentExpiry(ConsentAdminData consentAdminData) throws ConsentException { - - try { - ExpiredConsentStatusUpdateJob.updateExpiredStatues(); - consentAdminData.setResponseStatus(ResponseStatus.OK); - consentAdminData.setResponseStatus(ResponseStatus.NO_CONTENT); - } catch (ConsentManagementException e) { - log.error("Error while retrieving expiring consents", e); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, e.getMessage()); - } - - } - - @Override - public void handleTemporaryRetentionDataSyncing(ConsentAdminData consentAdminData) throws ConsentException { - - if (OpenBankingConfigParser.getInstance().isRetentionDataDBSyncEnabled()) { - consentAdminData.setResponseStatus(ResponseStatus.BAD_REQUEST); - log.error("Retention data DB sync periodical job is already enabled"); - throw new ConsentException(ResponseStatus.BAD_REQUEST, - "Retention data DB sync periodical job is already enabled"); - } - try { - RetentionDatabaseSyncJob.syncRetentionDatabase(); - consentAdminData.setResponseStatus(ResponseStatus.NO_CONTENT); - } catch (ConsentManagementException e) { - log.error("Error while triggering retention data DB sync method", e); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, e.getMessage()); - } - } - - @Override - public void handleConsentStatusAuditSearch(ConsentAdminData consentAdminData) throws ConsentException { - - JSONObject response = new JSONObject(); - ArrayList consentIDs = null; - Integer limit = null; - Integer offset = null; - boolean fetchFromRetentionDatabase = false; - - Map queryParams = consentAdminData.getQueryParams(); - - if (validateAndGetQueryParam(queryParams, "consentIDs") != null) { - consentIDs = new ArrayList<>(Arrays.asList(validateAndGetQueryParam(queryParams, "consentIDs"). - split(","))); - } - if (validateAndGetQueryParam(queryParams, "limit") != null) { - try { - limit = Integer.parseInt(validateAndGetQueryParam(queryParams, "limit")); - } catch (NumberFormatException e) { - log.error("Number format incorrect in search for parameter limit. Ignoring parameter"); - } - } - if (validateAndGetQueryParam(queryParams, "offset") != null) { - try { - offset = Integer.parseInt(validateAndGetQueryParam(queryParams, "offset")); - } catch (NumberFormatException e) { - log.error("Number format incorrect in search for parameter offset. Ignoring parameter"); - } - } - if (validateAndGetQueryParam(queryParams, FETCH_FROM_RETENTION_DB_QUERY_PARAM) != null) { - fetchFromRetentionDatabase = Boolean.parseBoolean(validateAndGetQueryParam(queryParams, - FETCH_FROM_RETENTION_DB_QUERY_PARAM)); - } - int count, total = 0; - - try { - ConsentCoreService consentCoreService = ConsentExtensionsDataHolder.getInstance().getConsentCoreService(); - ArrayList results = consentCoreService.getConsentStatusAuditRecords(consentIDs, - limit, offset, fetchFromRetentionDatabase); - - JSONArray consentAuditRecords = new JSONArray(); - for (ConsentStatusAuditRecord statusAuditRecord : results) { - JSONObject statusAuditRecordJSON = new JSONObject(); - statusAuditRecordJSON.appendField("statusAuditId", statusAuditRecord.getStatusAuditID()); - statusAuditRecordJSON.appendField("consentId", statusAuditRecord.getConsentID()); - statusAuditRecordJSON.appendField("currentStatus", statusAuditRecord.getCurrentStatus()); - statusAuditRecordJSON.appendField("actionTime", statusAuditRecord.getActionTime()); - statusAuditRecordJSON.appendField("reason", statusAuditRecord.getReason()); - statusAuditRecordJSON.appendField("actionBy", statusAuditRecord.getActionBy()); - statusAuditRecordJSON.appendField("previousStatus", statusAuditRecord.getPreviousStatus()); - consentAuditRecords.add(statusAuditRecordJSON); - } - response.appendField("data", consentAuditRecords); - count = consentAuditRecords.size(); - total = results.size(); - } catch (ConsentManagementException e) { - log.error("Error while retrieving consent status audit data"); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, e.getMessage()); - } - - //retrieve the total of the data set queried - if (limit != null || offset != null) { - try { - ArrayList results = ConsentExtensionsDataHolder.getInstance() - .getConsentCoreService().getConsentStatusAuditRecords(consentIDs, - null, null, fetchFromRetentionDatabase); - total = results.size(); - } catch (ConsentManagementException e) { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, e.getMessage()); - } - } - - JSONObject metadata = new JSONObject(); - metadata.appendField("count", count); - metadata.appendField("offset", offset); - metadata.appendField("limit", limit); - metadata.appendField("total", total); - response.appendField("metadata", metadata); - consentAdminData.setResponseStatus(ResponseStatus.OK); - consentAdminData.setResponsePayload(response); - } - - @Override - public void handleConsentFileSearch(ConsentAdminData consentAdminData) throws ConsentException { - - JSONObject response = new JSONObject(); - String consentID = null; - boolean fetchFromRetentionDatabase = false; - Map queryParams = consentAdminData.getQueryParams(); - - if (validateAndGetQueryParam(queryParams, "consentId") != null) { - consentID = validateAndGetQueryParam(queryParams, "consentId"); - } - - if (StringUtils.isBlank(consentID)) { - log.error("Request missing the mandatory query parameter consentId"); - throw new ConsentException(ResponseStatus.BAD_REQUEST, "Mandatory query parameter consentId " + - "not available"); - } - if (validateAndGetQueryParam(queryParams, FETCH_FROM_RETENTION_DB_QUERY_PARAM) != null) { - fetchFromRetentionDatabase = Boolean.parseBoolean(validateAndGetQueryParam(queryParams, - FETCH_FROM_RETENTION_DB_QUERY_PARAM)); - } - - try { - ConsentCoreService consentCoreService = ConsentExtensionsDataHolder.getInstance().getConsentCoreService(); - ConsentFile file = consentCoreService.getConsentFile(consentID, fetchFromRetentionDatabase); - response.appendField("consentFile", file.getConsentFile()); - } catch (ConsentManagementException e) { - log.error("Error while retrieving consent file"); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, e.getMessage()); - } - consentAdminData.setResponseStatus(ResponseStatus.OK); - consentAdminData.setResponsePayload(response); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/admin/model/ConsentAdminData.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/admin/model/ConsentAdminData.java deleted file mode 100644 index 39fdf9a4..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/admin/model/ConsentAdminData.java +++ /dev/null @@ -1,117 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.admin.model; - -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import net.minidev.json.JSONObject; - -import java.util.Map; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; - -/** - * Data wrapper for consent admin data. - */ -public class ConsentAdminData { - - private Map headers; - private JSONObject payload; - private Map queryParams; - private String absolutePath; - private HttpServletRequest request; - private HttpServletResponse response; - private ResponseStatus responseStatus; - private JSONObject responsePayload; - - public ConsentAdminData(Map headers, JSONObject payload, Map queryParams, - String absolutePath, HttpServletRequest request, HttpServletResponse response) { - this.headers = headers; - this.payload = payload; - this.queryParams = queryParams; - this.absolutePath = absolutePath; - this.request = request; - this.response = response; - } - - public ConsentAdminData(Map headers, Map queryParams, String absolutePath, - HttpServletRequest request, HttpServletResponse response) { - this.headers = headers; - payload = null; - this.queryParams = queryParams; - this.absolutePath = absolutePath; - this.request = request; - this.response = response; - } - - - public HttpServletRequest getRequest() { - return request; - } - - public HttpServletResponse getResponse() { - return response; - } - - public Map getQueryParams() { - return queryParams; - } - - public JSONObject getPayload() { - return payload; - } - - public Map getHeaders() { - return headers; - } - - public void setResponseStatus(ResponseStatus responseStatus) { - this.responseStatus = responseStatus; - } - - public void setResponsePayload(JSONObject responsePayload) { - this.responsePayload = responsePayload; - } - - public JSONObject getResponsePayload() { - return responsePayload; - } - - public ResponseStatus getResponseStatus() { - return responseStatus; - } - - public void setResponseHeader(String key, String value) { - response.setHeader(key, value); - } - - public void setResponseHeaders(Map headers) { - for (Map.Entry header : headers.entrySet()) { - setResponseHeader(header.getKey(), header.getValue()); - } - } - - public String getAbsolutePath() { - return absolutePath; - } - - public void setAbsolutePath(String absolutePath) { - this.absolutePath = absolutePath; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/admin/model/ConsentAdminHandler.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/admin/model/ConsentAdminHandler.java deleted file mode 100644 index 65f514bf..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/admin/model/ConsentAdminHandler.java +++ /dev/null @@ -1,63 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.admin.model; - -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; - -/** - * Consent admin handler interface. - */ -public interface ConsentAdminHandler { - - public void handleSearch(ConsentAdminData consentAdminData) throws ConsentException; - - public void handleRevoke(ConsentAdminData consentAdminData) throws ConsentException; - - /** - * This method is used to handle the consent amendment history retrieval request. - * - * @param consentAdminData Data wrapper for consent admin data that holds the request context data - * @throws ConsentException thrown if any error occurs in the process - */ - public void handleConsentAmendmentHistoryRetrieval(ConsentAdminData consentAdminData) throws ConsentException; - - public void handleConsentExpiry(ConsentAdminData consentAdminData) throws ConsentException; - - /** - * Method to handle the temporary retention data syncing with the retention database. - * @param consentAdminData consentAdminData - * @throws ConsentException if any error occurs while syncing the retention database - */ - public void handleTemporaryRetentionDataSyncing(ConsentAdminData consentAdminData) throws ConsentException; - - /** - * Method to handle the consent status audit search. - * @param consentAdminData consentAdminData - * @throws ConsentException if any error occurs while searching the consent status audit - */ - public void handleConsentStatusAuditSearch(ConsentAdminData consentAdminData) throws ConsentException; - - /** - * Method to handle the consent file search. - * @param consentAdminData consentAdminData - * @throws ConsentException if any error occurs while searching the consent file - */ - public void handleConsentFileSearch(ConsentAdminData consentAdminData) throws ConsentException; - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/builder/ConsentStepsBuilder.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/builder/ConsentStepsBuilder.java deleted file mode 100644 index 7a84bf78..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/builder/ConsentStepsBuilder.java +++ /dev/null @@ -1,77 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authorize.builder; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingRuntimeException; -import com.wso2.openbanking.accelerator.common.util.OpenBankingUtils; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentPersistStep; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentRetrievalStep; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.List; -import java.util.Map; -import java.util.stream.Collectors; - -/** - * Builder class for consent steps. - */ -public class ConsentStepsBuilder { - - private static final Log log = LogFactory.getLog(ConsentStepsBuilder.class); - private List consentPersistSteps = null; - private List consentRetrievalSteps = null; - private static final String RETRIEVE = "Retrieve"; - private static final String PERSIST = "Persist"; - - public void build() { - - try { - Map> stepsConfig = - ConsentExtensionsDataHolder.getInstance().getOpenBankingConfigurationService().getAuthorizeSteps(); - Map persistIntegerStringMap = stepsConfig.get(PERSIST); - if (persistIntegerStringMap != null) { - consentPersistSteps = persistIntegerStringMap.keySet().stream() - .map(integer -> (ConsentPersistStep) OpenBankingUtils - .getClassInstanceFromFQN(persistIntegerStringMap.get(integer))) - .collect(Collectors.toList()); - log.debug("Persistence steps loaded successfully"); - } - Map retrieveIntegerStringMap = stepsConfig.get(RETRIEVE); - if (retrieveIntegerStringMap != null) { - consentRetrievalSteps = retrieveIntegerStringMap.keySet().stream() - .map(integer -> (ConsentRetrievalStep) OpenBankingUtils - .getClassInstanceFromFQN(retrieveIntegerStringMap.get(integer))) - .collect(Collectors.toList()); - log.debug("Retrieval steps loaded successfully"); - } - } catch (OpenBankingRuntimeException e) { - log.error("Authorize steps not loaded successfully. Please verify configurations. " + e.getMessage()); - } - } - - public List getConsentPersistSteps() { - return consentPersistSteps; - } - - public List getConsentRetrievalSteps() { - return consentRetrievalSteps; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/CIBAConsentPersistStep.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/CIBAConsentPersistStep.java deleted file mode 100644 index a7062374..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/CIBAConsentPersistStep.java +++ /dev/null @@ -1,69 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authorize.impl; - -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentData; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentPersistData; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentPersistStep; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; - -import java.io.Serializable; -import java.util.HashMap; -import java.util.Map; - -/** - * Consent persistence step for CIBA flow. - */ -public class CIBAConsentPersistStep implements ConsentPersistStep { - - @Override - public void execute(ConsentPersistData consentPersistData) throws ConsentException { - - try { - ConsentData consentData = consentPersistData.getConsentData(); - Map sensitiveDataMap = consentData.getSensitiveDataMap(); - - ConsentResource consentResource; - - if (consentData.getConsentResource() == null) { - consentResource = ConsentExtensionsDataHolder.getInstance().getConsentCoreService() - .getConsent(consentData.getConsentId(), false); - } else { - consentResource = consentData.getConsentResource(); - } - - if (sensitiveDataMap != null) { - //Storing mapping to be used to bind consent id scope for CIBA flows - if (sensitiveDataMap.containsKey("auth_req_id")) { - Map consentAttributes = new HashMap<>(); - consentAttributes.put("auth_req_id", (String) consentData.getSensitiveDataMap().get("auth_req_id")); - ConsentExtensionsDataHolder.getInstance().getConsentCoreService().storeConsentAttributes - (consentResource.getConsentID(), consentAttributes); - } - } - } catch (ConsentManagementException e) { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - "Exception occured while persisting consent"); - } - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/CIBAConsentRetrievalStep.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/CIBAConsentRetrievalStep.java deleted file mode 100644 index b5fe62c8..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/CIBAConsentRetrievalStep.java +++ /dev/null @@ -1,74 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authorize.impl; - -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentData; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentRetrievalStep; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionUtils; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import net.minidev.json.JSONObject; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -/** - * Consent retrieval step for CIBA flow. - */ -public class CIBAConsentRetrievalStep implements ConsentRetrievalStep { - - private static final Log log = LogFactory.getLog(CIBAConsentRetrievalStep.class); - private static final String OPENBANKING_INTENT_ID = "openbanking_intent_id"; - private static final String VALUE = "value"; - - @Override - public void execute(ConsentData consentData, JSONObject jsonObject) throws ConsentException { - - //If query params are null it should be a CIBA flow - // Run this step only if it is a CIBA flow - if (consentData.getSpQueryParams() == null) { - if (consentData.getSensitiveDataMap().containsKey("request")) { - String requestObject = (String) consentData.getSensitiveDataMap().get("request"); - String consentId = validateCibaRequestObjectAndExtractConsentId(requestObject); - consentData.setConsentId(consentId); - } else { - throw new ConsentException(ResponseStatus.BAD_REQUEST, "Request object unavailable"); - } - } - - } - - private String validateCibaRequestObjectAndExtractConsentId(String requestObject) { - - String consentId = null; - JSONObject jsonObject = ConsentExtensionUtils.getRequestObjectPayload(requestObject); - - if (jsonObject.containsKey(OPENBANKING_INTENT_ID)) { - JSONObject intentObject = (JSONObject) jsonObject.get(OPENBANKING_INTENT_ID); - if (intentObject.containsKey(VALUE)) { - consentId = (String) intentObject.get(VALUE); - } - } - - if (consentId == null) { - log.error("intent_id not found in request object"); - throw new ConsentException(ResponseStatus.BAD_REQUEST, "intent_id not found in request object"); - } - return consentId; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/DefaultConsentPersistStep.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/DefaultConsentPersistStep.java deleted file mode 100644 index 47a06698..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/DefaultConsentPersistStep.java +++ /dev/null @@ -1,126 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authorize.impl; - -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentData; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentPersistData; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentPersistStep; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.ArrayList; - - -/** - * Consent persist step default implementation. - */ -public class DefaultConsentPersistStep implements ConsentPersistStep { - - private static final Log log = LogFactory.getLog(DefaultConsentPersistStep.class); - - @Override - public void execute(ConsentPersistData consentPersistData) throws ConsentException { - - try { - ConsentData consentData = consentPersistData.getConsentData(); - ConsentResource consentResource; - - if (consentData.getConsentId() == null && consentData.getConsentResource() == null) { - log.error("Consent ID not available in consent data"); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - "Consent ID not available in consent data"); - } - - if (consentData.getConsentResource() == null) { - consentResource = ConsentExtensionsDataHolder.getInstance().getConsentCoreService() - .getConsent(consentData.getConsentId(), false); - } else { - consentResource = consentData.getConsentResource(); - } - - if (consentData.getAuthResource() == null) { - log.error("Auth resource not available in consent data"); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - "Auth resource not available in consent data"); - } - - consentPersist(consentPersistData, consentResource); - - - } catch (ConsentManagementException e) { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - "Exception occured while persisting consent"); - } - } - - /** - * This method defined to handle consent persistence based on the consent type. - * - * @param consentPersistData Consent Persist Data Object - * @param consentResource Consent Resource Object - * @throws ConsentManagementException - */ - public static void consentPersist(ConsentPersistData consentPersistData, ConsentResource consentResource) - throws ConsentManagementException { - - ConsentData consentData = consentPersistData.getConsentData(); - - JSONObject payload = consentPersistData.getPayload(); - - if (payload.get(ConsentExtensionConstants.ACCOUNT_IDS) == null || - !(payload.get(ConsentExtensionConstants.ACCOUNT_IDS) instanceof JSONArray)) { - log.error(ErrorConstants.ACCOUNT_ID_NOT_FOUND_ERROR); - throw new ConsentException(ResponseStatus.BAD_REQUEST, - ErrorConstants.ACCOUNT_ID_NOT_FOUND_ERROR); - } - - JSONArray accountIds = (JSONArray) payload.get(ConsentExtensionConstants.ACCOUNT_IDS); - ArrayList accountIdsString = new ArrayList<>(); - for (Object account : accountIds) { - if (!(account instanceof String)) { - log.error(ErrorConstants.ACCOUNT_ID_FORMAT_ERROR); - throw new ConsentException(ResponseStatus.BAD_REQUEST, - ErrorConstants.ACCOUNT_ID_FORMAT_ERROR); - } - accountIdsString.add((String) account); - } - String consentStatus; - - if (consentPersistData.getApproval()) { - consentStatus = ConsentExtensionConstants.AUTHORIZED_STATUS; - } else { - consentStatus = ConsentExtensionConstants.REJECTED_STATUS; - } - - ConsentExtensionsDataHolder.getInstance().getConsentCoreService() - .bindUserAccountsToConsent(consentResource, consentData.getUserId(), - consentData.getAuthResource().getAuthorizationID(), accountIdsString, consentStatus, - consentStatus); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/DefaultConsentRetrievalStep.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/DefaultConsentRetrievalStep.java deleted file mode 100644 index 0deca7b5..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/DefaultConsentRetrievalStep.java +++ /dev/null @@ -1,122 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authorize.impl; - -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentData; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentRetrievalStep; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.utils.ConsentRetrievalUtil; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentServiceUtil; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -/** - * Consent retrieval step default implementation. - */ -public class DefaultConsentRetrievalStep implements ConsentRetrievalStep { - - private static final Log log = LogFactory.getLog(DefaultConsentRetrievalStep.class); - - @Override - public void execute(ConsentData consentData, JSONObject jsonObject) throws ConsentException { - - if (!consentData.isRegulatory()) { - return; - } - - ConsentCoreServiceImpl consentCoreService = ConsentServiceUtil.getConsentService(); - - try { - // If this is a CIBA flow, consent ID is already set at the CIBAConsentRetrievalStep - String consentId = consentData.getConsentId(); - if (consentId == null) { - // If query params are null, this is a CIBA flow. Therefore consent ID should be set at CIBA consent - // retrieval step - if (consentData.getSpQueryParams() == null) { - throw new ConsentException(ResponseStatus.BAD_REQUEST, "CIBA consent retrieval step has not been " + - "executed successfully before default consent persist step"); - } - String requestObject = ConsentRetrievalUtil.extractRequestObject(consentData.getSpQueryParams()); - consentId = ConsentRetrievalUtil.extractConsentId(requestObject); - consentData.setConsentId(consentId); - } - ConsentResource consentResource = consentCoreService.getConsent(consentId, false); - - if (!consentResource.getCurrentStatus().equals(ConsentExtensionConstants.AWAITING_AUTH_STATUS)) { - log.error("Consent not in authorizable state"); - //Currently throwing error as 400 response. Developer also have the option of appending a field IS_ERROR - // to the jsonObject and showing it to the user in the webapp. If so, the IS_ERROR have to be checked in - // any later steps. - throw new ConsentException(ResponseStatus.BAD_REQUEST, "Consent not in authorizable state"); - } - - AuthorizationResource authorizationResource = ConsentExtensionsDataHolder.getInstance() - .getConsentCoreService().searchAuthorizations(consentId).get(0); - if (!authorizationResource.getAuthorizationStatus().equals(ConsentExtensionConstants.CREATED_STATUS)) { - log.error("Authorization not in authorizable state"); - //Currently throwing error as 400 response. Developer also have the option of appending a field IS_ERROR - // to the jsonObject and showing it to the user in the webapp. If so, the IS_ERROR have to be checked in - // any later steps. - throw new ConsentException(ResponseStatus.BAD_REQUEST, "Authorization not in authorizable state"); - } - - consentData.setType(consentResource.getConsentType()); - consentData.setAuthResource(authorizationResource); - consentData.setConsentResource(consentResource); - - //Appending Consent Data - JSONArray consentDataJSON = getConsentDataSet(consentResource); - jsonObject.appendField(ConsentExtensionConstants.CONSENT_DATA, consentDataJSON); - - //Appending Dummy data for Accounts consent. Ideally should be separate step calling accounts service - JSONArray accountsJSON = ConsentRetrievalUtil.appendDummyAccountID(); - jsonObject.appendField(ConsentExtensionConstants.ACCOUNTS, accountsJSON); - - } catch (ConsentException e) { - JSONObject errorObj = (JSONObject) e.getPayload(); - JSONArray errorList = (JSONArray) errorObj.get("Errors"); - jsonObject.put(ConsentExtensionConstants.IS_ERROR, - ((JSONObject) errorList.get(0)).getAsString("Message")); - return; - } catch (ConsentManagementException e) { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - "Exception occurred while getting consent data"); - } - } - - /** - * Method to retrieve consent related data from the initiation payload. - * @param consentResource Consent Resource - * @return consent - */ - public JSONArray getConsentDataSet(ConsentResource consentResource) { - - return ConsentRetrievalUtil.getConsentData(consentResource); - } - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/NonRegulatoryConsentStep.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/NonRegulatoryConsentStep.java deleted file mode 100644 index 30b0475f..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/NonRegulatoryConsentStep.java +++ /dev/null @@ -1,91 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authorize.impl; - -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentData; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentRetrievalStep; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import org.apache.commons.lang3.StringUtils; - -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.stream.Collectors; -import java.util.stream.Stream; - -/** - * Default retrieval step to get comprehensive consent. - */ -public class NonRegulatoryConsentStep implements ConsentRetrievalStep { - - private static final String OPENID_SCOPES = "openid_scopes"; - private static final String OPENID = "openid"; - private static final String CONSENT_MGT = "consentmgt"; - private static final String DEFAULT = "default"; - private static final String CONSENT_READ_ALL_SCOPE = "consents:read_all"; - private static final String CONSENT_READ_SELF_SCOPE = "consents:read_self"; - private static final Map CONSENT_SCOPES = new HashMap<>(); - - - @Override - public void execute(ConsentData consentData, JSONObject jsonObject) { - if (consentData.isRegulatory()) { - return; - } - - if (consentData.getScopeString().toLowerCase().contains(CONSENT_MGT)) { - consentData.setType(CONSENT_MGT); - } else if (consentData.getScopeString().contains(OPENID)) { - consentData.setType(DEFAULT); - } - String scopeString = consentData.getScopeString(); - addScopesArray(scopeString, jsonObject); - } - - /** - * Add scopes to json object. - * - * @param scopeString scopes string - * @param jsonObject json object - */ - private void addScopesArray(String scopeString, JSONObject jsonObject) { - - if (StringUtils.isNotBlank(scopeString)) { - // Remove "openid" from the scope list to display. - List openIdScopes = Stream.of(scopeString.split(" ")) - .filter(x -> !StringUtils.equalsIgnoreCase(x, OPENID)) - // if scope found in CONSENT_SCOPES map return meaningful scope value, else return scope - .map(scope -> getConsentScopes().getOrDefault(scope, scope)) - .collect(Collectors.toList()); - JSONArray scopeArray = new JSONArray(); - scopeArray.addAll(openIdScopes); - jsonObject.put(OPENID_SCOPES, scopeArray); - } - } - - protected synchronized Map getConsentScopes() { - if (CONSENT_SCOPES.isEmpty()) { - // Add meaningful string value to scopes - CONSENT_SCOPES.put(CONSENT_READ_ALL_SCOPE, "Manage all consents"); - CONSENT_SCOPES.put(CONSENT_READ_SELF_SCOPE, "Manage your consents"); - } - return CONSENT_SCOPES; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/RequestObjectCheckStep.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/RequestObjectCheckStep.java deleted file mode 100644 index 3de0a34a..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/impl/RequestObjectCheckStep.java +++ /dev/null @@ -1,57 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authorize.impl; - -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentData; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentRetrievalStep; -import com.wso2.openbanking.accelerator.consent.extensions.common.AuthErrorCode; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import net.minidev.json.JSONObject; -/** - * Step to check whether the request object is sent in the authorization reques for - * regulatory app. - */ -public class RequestObjectCheckStep implements ConsentRetrievalStep { - - @Override - public void execute(ConsentData consentData, JSONObject jsonObject) throws ConsentException { - - if (consentData.isRegulatory() && !checkRequestObject(consentData.getSpQueryParams())) { - JSONObject json = new JSONObject(); - json.put("error", AuthErrorCode.INVALID_REQUEST.toString()); - json.put("redirect_uri", consentData.getRedirectURI().toString()); - throw new ConsentException(ResponseStatus.BAD_REQUEST, json); - } - } - - private boolean checkRequestObject(String spQueryParams) { - - boolean requestObjectExist = false; - if (spQueryParams != null && !spQueryParams.trim().isEmpty()) { - String requestObject = null; - String[] spQueries = spQueryParams.split("&"); - for (String param : spQueries) { - if (param.contains("request=")) { - requestObjectExist = true; - } - } - } - return requestObjectExist; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/model/ConsentData.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/model/ConsentData.java deleted file mode 100644 index ea3efe1b..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/model/ConsentData.java +++ /dev/null @@ -1,208 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authorize.model; - -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; - -import java.io.Serializable; -import java.net.URI; -import java.util.HashMap; -import java.util.Map; - -/** - * Data wrapper for consent retrieve flow data. - * - */ -public class ConsentData implements Serializable { - - private static final long serialVersionUID = -568639746792395972L; - private String sessionDataKey; - private String userId; - private String spQueryParams; - private String scopeString; - private String application; - private String consentId; - private String clientId; - private Boolean regulatory; - private Map requestHeaders; - private ConsentResource consentResource; - private AuthorizationResource authResource; - private Map metaDataMap; - private Map sensitiveDataMap; - private URI redirectURI; - private String state; - - //Mandatory parameter to be set by extension. This will be considered "DEFAULT" if not set. - private String type; - - public ConsentData(String sessionDataKey, String userId, String spQueryParams, String scopeString, - String application, Map requestHeaders) { - - this.sessionDataKey = sessionDataKey; - this.userId = userId; - this.spQueryParams = spQueryParams; - this.scopeString = scopeString; - this.application = application; - this.requestHeaders = requestHeaders; - this.metaDataMap = new HashMap<>(); - } - - public String getSessionDataKey() { - return sessionDataKey; - } - - public void setSessionDataKey(String sessionDataKey) { - this.sessionDataKey = sessionDataKey; - } - - public String getUserId() { - - return userId; - } - - public void setUserId(String userId) { - - this.userId = userId; - } - - public String getApplication() { - - return application; - } - - public void setApplication(String application) { - - this.application = application; - } - - public String getScopeString() { - - return scopeString; - } - - public void setScopeString(String scopeString) { - - this.scopeString = scopeString; - } - - public String getSpQueryParams() { - - return spQueryParams; - } - - public void setSpQueryParams(String spQueryParams) { - - this.spQueryParams = spQueryParams; - } - - public Map getRequestHeaders() { - return requestHeaders; - } - - public String getType() { - return type; - } - - public void setType(String type) { - this.type = type; - } - - public ConsentResource getConsentResource() { - return consentResource; - } - - public void setConsentResource(ConsentResource consentResource) { - this.consentResource = consentResource; - } - - public String getConsentId() { - return consentId; - } - - public void setConsentId(String consentId) { - this.consentId = consentId; - } - - public AuthorizationResource getAuthResource() { - return authResource; - } - - public void setAuthResource(AuthorizationResource authResource) { - this.authResource = authResource; - } - - public Boolean isRegulatory() { - return regulatory; - } - - public void setRegulatory(Boolean regulatory) { - this.regulatory = regulatory; - } - - public String getClientId() { - return clientId; - } - - public void setClientId(String clientId) { - this.clientId = clientId; - } - - public Map getMetaDataMap() { - return metaDataMap; - } - - public void setMetaDataMap(Map metaDataMap) { - this.metaDataMap = metaDataMap; - } - - public void addData(String key, Object value) { - - this.metaDataMap.put(key, value); - } - - public void addAllData(Map data) { - - this.metaDataMap.putAll(data); - } - - public URI getRedirectURI() { - return redirectURI; - } - - public void setRedirectURI(URI redirectURI) { - this.redirectURI = redirectURI; - } - - public String getState() { - return state; - } - - public void setState(String state) { - this.state = state; - } - - public Map getSensitiveDataMap() { - return sensitiveDataMap; - } - - public void setSensitiveDataMap(Map sensitiveDataMap) { - this.sensitiveDataMap = sensitiveDataMap; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/model/ConsentPersistData.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/model/ConsentPersistData.java deleted file mode 100644 index c73455b0..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/model/ConsentPersistData.java +++ /dev/null @@ -1,109 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authorize.model; - -import net.minidev.json.JSONObject; - -import java.util.HashMap; -import java.util.Map; - -/** - * Data wrapper for consent persist flow data. - */ -public class ConsentPersistData { - - //Payload of the persist request - private JSONObject payload; - //Request headers of the persist request - private Map headers; - //Consent data object used in the retrieval flow populated via a cache - private ConsentData consentData; - //Boolean value representing whether the approval was granted by the user - private boolean approval; - //Additional metadata - private Map metadata; - //Browser cookies - private Map browserCookies; - - public ConsentPersistData(JSONObject payload, Map headers, boolean approval, - ConsentData consentData) { - this.payload = payload; - this.headers = headers; - this.approval = approval; - this.consentData = consentData; - metadata = new HashMap<>(); - browserCookies = new HashMap<>(); - } - - public JSONObject getPayload() { - return payload; - } - - public void setPayload(JSONObject payload) { - this.payload = payload; - } - - public Map getHeaders() { - return headers; - } - - public void setHeaders(Map headers) { - this.headers = headers; - } - - public boolean getApproval() { - return approval; - } - - public void setApproval(boolean approval) { - this.approval = approval; - } - - public ConsentData getConsentData() { - return consentData; - } - - public void setConsentData(ConsentData consentData) { - this.consentData = consentData; - } - - public Map getMetadata() { - return metadata; - } - - public void addMetadata(String key, Object value) { - - this.metadata.put(key, value); - } - - public void addMultipleMetadata(Map data) { - - this.metadata.putAll(data); - } - - public Map getBrowserCookies() { - - return browserCookies; - } - - public void setBrowserCookies(Map cookies) { - - this.browserCookies.putAll(cookies); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/model/ConsentPersistStep.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/model/ConsentPersistStep.java deleted file mode 100644 index 0b36dffd..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/model/ConsentPersistStep.java +++ /dev/null @@ -1,35 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authorize.model; - -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; - -/** - * Consent persist step interface. - */ -public interface ConsentPersistStep { - - /** - * Method to be implemented as a step for consent persistence. Once implemented add the step to the configuration. - * - * @param consentPersistData Includes all the generic data of the consents. - * @throws ConsentException Exception thrown in case of failure. - */ - void execute(ConsentPersistData consentPersistData) throws ConsentException; -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/model/ConsentRetrievalStep.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/model/ConsentRetrievalStep.java deleted file mode 100644 index 6d4d5cc2..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/model/ConsentRetrievalStep.java +++ /dev/null @@ -1,38 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authorize.model; - - -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import net.minidev.json.JSONObject; - -/** - * Consent retrieval step interface. - */ -public interface ConsentRetrievalStep { - - /** - * Method to be implemented as a step for consent retrieval. Once implemented add the step to the configuration. - * - * @param consentData Includes all the data that is received to the consent page. - * @param jsonObject Passed on through each step where the response body sent to the page is built. - * @throws ConsentException Exception thrown in case of failure. - */ - void execute(ConsentData consentData, JSONObject jsonObject) throws ConsentException; -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/utils/ConsentRetrievalUtil.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/utils/ConsentRetrievalUtil.java deleted file mode 100644 index f0cc17b4..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/utils/ConsentRetrievalUtil.java +++ /dev/null @@ -1,700 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authorize.utils; - - -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.nio.charset.StandardCharsets; -import java.time.OffsetDateTime; -import java.time.format.DateTimeParseException; -import java.util.Base64; - -/** - * Util class for Consent authorize implementation. - */ -public class ConsentRetrievalUtil { - - private static final Log log = LogFactory.getLog(ConsentRetrievalUtil.class); - - /** - * Method to extract request object from query params. - * - * @param spQueryParams Query params - * @return requestObject - */ - public static String extractRequestObject(String spQueryParams) { - - if (StringUtils.isNotBlank(spQueryParams)) { - String requestObject = null; - String[] spQueries = spQueryParams.split("&"); - for (String param : spQueries) { - if (param.contains("request=")) { - requestObject = (param.substring("request=".length())).replaceAll( - "\\r\\n|\\r|\\n|\\%20", ""); - } - } - if (requestObject != null) { - return requestObject; - } - } - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, ErrorConstants.REQUEST_OBJ_EXTRACT_ERROR); - } - - /** - * Method to validate the request object and extract consent ID. - * - * @param requestObject Request object - * @return consentId - */ - public static String extractConsentId(String requestObject) { - - String consentId = null; - try { - // validate request object and get the payload - String requestObjectPayload; - String[] jwtTokenValues = requestObject.split("\\."); - if (jwtTokenValues.length == ConsentExtensionConstants.NUMBER_OF_PARTS_IN_JWS) { - requestObjectPayload = new String(Base64.getUrlDecoder().decode(jwtTokenValues[1]), - StandardCharsets.UTF_8); - } else { - log.error(ErrorConstants.REQUEST_OBJ_NOT_SIGNED); - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.REQUEST_OBJ_NOT_SIGNED); - } - Object payload = new JSONParser(JSONParser.MODE_PERMISSIVE).parse(requestObjectPayload); - if (!(payload instanceof JSONObject)) { - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.NOT_JSON_PAYLOAD); - } - JSONObject jsonObject = (JSONObject) payload; - - // get consent id from the request object - if (jsonObject.containsKey(ConsentExtensionConstants.CLAIMS)) { - JSONObject claims = (JSONObject) jsonObject.get(ConsentExtensionConstants.CLAIMS); - for (String claim : ConsentExtensionConstants.CLAIM_FIELDS) { - if (claims.containsKey(claim)) { - JSONObject claimObject = (JSONObject) claims.get(claim); - if (claimObject.containsKey(ConsentExtensionConstants.OPENBANKING_INTENT_ID)) { - JSONObject intentObject = (JSONObject) claimObject - .get(ConsentExtensionConstants.OPENBANKING_INTENT_ID); - if (intentObject.containsKey(ConsentExtensionConstants.VALUE)) { - consentId = (String) intentObject.get(ConsentExtensionConstants.VALUE); - break; - } - } - } - } - } - - if (consentId == null) { - log.error(ErrorConstants.INTENT_ID_NOT_FOUND); - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.INTENT_ID_NOT_FOUND); - } - return consentId; - - } catch (ParseException e) { - log.error(ErrorConstants.REQUEST_OBJ_PARSE_ERROR, e); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, ErrorConstants.REQUEST_OBJ_PARSE_ERROR); - } - } - - - /** - * Check if the expiry date time of the consent has elapsed. - * - * @param expiryDate The expiry date/time of consent - * @return boolean result of validation - */ - public static boolean validateExpiryDateTime(String expiryDate) throws ConsentException { - - try { - OffsetDateTime expDate = OffsetDateTime.parse(expiryDate); - if (log.isDebugEnabled()) { - log.debug(String.format(ErrorConstants.DATE_PARSE_MSG, expDate, OffsetDateTime.now())); - } - return OffsetDateTime.now().isBefore(expDate); - } catch (DateTimeParseException e) { - log.error(ErrorConstants.EXP_DATE_PARSE_ERROR, e); - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.ACC_CONSENT_RETRIEVAL_ERROR); - } - } - - /** - * Method to add debtor account details to consent data to send it to the consent page. - * - * @param initiation Initiation object from the request - * @param consentDataJSON Consent information object - */ - public static void populateDebtorAccount(JSONObject initiation, JSONArray consentDataJSON) { - if (initiation.get(ConsentExtensionConstants.DEBTOR_ACC) != null) { - JSONObject debtorAccount = (JSONObject) initiation.get(ConsentExtensionConstants.DEBTOR_ACC); - JSONArray debtorAccountArray = new JSONArray(); - - //Adding Debtor Account Scheme Name - if (debtorAccount.getAsString(ConsentExtensionConstants.SCHEME_NAME) != null) { - debtorAccountArray.add(ConsentExtensionConstants.SCHEME_NAME_TITLE + " : " + - debtorAccount.getAsString(ConsentExtensionConstants.SCHEME_NAME)); - } - - //Adding Debtor Account Identification - if (debtorAccount.getAsString(ConsentExtensionConstants.IDENTIFICATION) != null) { - debtorAccountArray.add(ConsentExtensionConstants.IDENTIFICATION_TITLE + " : " + - debtorAccount.getAsString(ConsentExtensionConstants.IDENTIFICATION)); - } - - //Adding Debtor Account Name - if (debtorAccount.getAsString(ConsentExtensionConstants.NAME) != null) { - debtorAccountArray.add(ConsentExtensionConstants.NAME_TITLE + " : " + - debtorAccount.getAsString(ConsentExtensionConstants.NAME)); - } - - //Adding Debtor Account Secondary Identification - if (debtorAccount.getAsString(ConsentExtensionConstants.SECONDARY_IDENTIFICATION) != null) { - debtorAccountArray.add(ConsentExtensionConstants.SECONDARY_IDENTIFICATION_TITLE + " : " + - debtorAccount.getAsString(ConsentExtensionConstants.SECONDARY_IDENTIFICATION)); - } - - - JSONObject jsonElementDebtor = new JSONObject(); - jsonElementDebtor.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.DEBTOR_ACC_TITLE); - jsonElementDebtor.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), debtorAccountArray); - consentDataJSON.add(jsonElementDebtor); - } - } - - - /** - * Method to add debtor account details to consent data to send it to the consent page. - * - * @param initiation Initiation object from the request - * @param consentDataJSON Consent information object - */ - public static void populateCreditorAccount(JSONObject initiation, JSONArray consentDataJSON) { - if (initiation.get(ConsentExtensionConstants.CREDITOR_ACC) != null) { - JSONObject creditorAccount = (JSONObject) initiation.get(ConsentExtensionConstants.CREDITOR_ACC); - JSONArray creditorAccountArray = new JSONArray(); - //Adding Debtor Account Scheme Name - if (creditorAccount.getAsString(ConsentExtensionConstants.SCHEME_NAME) != null) { - creditorAccountArray.add(ConsentExtensionConstants.SCHEME_NAME_TITLE + " : " + - creditorAccount.getAsString(ConsentExtensionConstants.SCHEME_NAME)); - } - //Adding Debtor Account Identification - if (creditorAccount.getAsString(ConsentExtensionConstants.IDENTIFICATION) != null) { - creditorAccountArray.add(ConsentExtensionConstants.IDENTIFICATION_TITLE + " : " + - creditorAccount.getAsString(ConsentExtensionConstants.IDENTIFICATION)); - } - //Adding Debtor Account Name - if (creditorAccount.getAsString(ConsentExtensionConstants.NAME) != null) { - creditorAccountArray.add(ConsentExtensionConstants.NAME_TITLE + " : " + - creditorAccount.getAsString(ConsentExtensionConstants.NAME)); - } - //Adding Debtor Account Secondary Identification - if (creditorAccount.getAsString(ConsentExtensionConstants.SECONDARY_IDENTIFICATION) != null) { - creditorAccountArray.add(ConsentExtensionConstants.SECONDARY_IDENTIFICATION_TITLE + " : " + - creditorAccount.getAsString(ConsentExtensionConstants.SECONDARY_IDENTIFICATION)); - } - - JSONObject jsonElementCreditor = new JSONObject(); - jsonElementCreditor.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.CREDITOR_ACC_TITLE); - jsonElementCreditor.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), - creditorAccountArray); - consentDataJSON.add(jsonElementCreditor); - } - } - - - /** - * Method to append Dummy data for Account ID. Ideally should be separate step calling accounts service - * - * @return accountsJSON - */ - public static JSONArray appendDummyAccountID() { - - JSONArray accountsJSON = new JSONArray(); - JSONObject accountOne = new JSONObject(); - accountOne.appendField("account_id", "12345"); - accountOne.appendField("display_name", "Salary Saver Account"); - - accountsJSON.add(accountOne); - - JSONObject accountTwo = new JSONObject(); - accountTwo.appendField("account_id", "67890"); - accountTwo.appendField("account_id", "67890"); - accountTwo.appendField("display_name", "Max Bonus Account"); - - accountsJSON.add(accountTwo); - - return accountsJSON; - - } - - /** - * Method that consists the implementation for the validation of payload and the consent, - * this method also invokes the relevant methods to populate data for each flow. - * - * @param consentResource Consent Resource parameter containing consent related information retrieved - * from database. - * @return ConsentDataJson array - */ - public static JSONArray getConsentData(ConsentResource consentResource) throws ConsentException { - - JSONArray consentDataJSON = new JSONArray(); - try { - - String receiptString = consentResource.getReceipt(); - Object receiptJSON = new JSONParser(JSONParser.MODE_PERMISSIVE).parse(receiptString); - - // Checking whether the request body is in JSON format - if (!(receiptJSON instanceof JSONObject)) { - log.error(ErrorConstants.NOT_JSON_OBJECT_ERROR); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - ErrorConstants.NOT_JSON_OBJECT_ERROR); - } - - if (!ConsentExtensionConstants.AWAITING_AUTH_STATUS.equals(consentResource.getCurrentStatus())) { - log.error(ErrorConstants.STATE_INVALID_ERROR); - // Currently throwing an error as a 400 response. - // Developers have the option of appending a field IS_ERROR to the jsonObject - // and showing it to the user in the webapp.If so,the IS_ERROR has to be checked in any later steps. - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.STATE_INVALID_ERROR); - } - - JSONObject receipt = (JSONObject) receiptJSON; - - // Checks if 'data' object is present in the receipt - if (receipt.containsKey(ConsentExtensionConstants.DATA)) { - JSONObject data = (JSONObject) receipt.get(ConsentExtensionConstants.DATA); - - String type = consentResource.getConsentType(); - switch (type) { - case ConsentExtensionConstants.ACCOUNTS: - populateAccountData(data, consentDataJSON); - break; - case ConsentExtensionConstants.PAYMENTS: - populatePaymentData(data, consentDataJSON); - break; - case ConsentExtensionConstants.FUNDSCONFIRMATIONS: - populateCofData(data, consentDataJSON); - break; - case ConsentExtensionConstants.VRP: - populateVRPData(data, consentDataJSON); - break; - default: - break; - } - } else { - log.error(ErrorConstants.DATA_OBJECT_MISSING_ERROR); - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.DATA_OBJECT_MISSING_ERROR); - } - } catch (ParseException e) { - log.error(ErrorConstants.CONSENT_RETRIEVAL_ERROR, e); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - ErrorConstants.CONSENT_RETRIEVAL_ERROR); - } - return consentDataJSON; - } - - /** - * Populate Domestic and international Payment Details. - * - * @param data data request from the request - * @param consentDataJSON Consent information - */ - private static void populatePaymentData(JSONObject data, JSONArray consentDataJSON) { - - JSONArray paymentTypeArray = new JSONArray(); - JSONObject jsonElementPaymentType = new JSONObject(); - - if (data.containsKey(ConsentExtensionConstants.INITIATION)) { - JSONObject initiation = (JSONObject) data.get(ConsentExtensionConstants.INITIATION); - - if (initiation.containsKey(ConsentExtensionConstants.CURRENCY_OF_TRANSFER)) { - //For International Payments - //Adding Payment Type - paymentTypeArray.add(ConsentExtensionConstants.INTERNATIONAL_PAYMENTS); - - jsonElementPaymentType.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.PAYMENT_TYPE_TITLE); - jsonElementPaymentType.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), - paymentTypeArray); - consentDataJSON.add(jsonElementPaymentType); - - //Adding Currency Of Transfer - JSONArray currencyTransferArray = new JSONArray(); - currencyTransferArray.add(initiation.getAsString(ConsentExtensionConstants.CURRENCY_OF_TRANSFER)); - - JSONObject jsonElementCurTransfer = new JSONObject(); - jsonElementCurTransfer.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.CURRENCY_OF_TRANSFER_TITLE); - jsonElementCurTransfer.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), - currencyTransferArray); - consentDataJSON.add(jsonElementCurTransfer); - } else { - //Adding Payment Type - paymentTypeArray.add(ConsentExtensionConstants.DOMESTIC_PAYMENTS); - - jsonElementPaymentType.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.PAYMENT_TYPE_TITLE); - jsonElementPaymentType.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), - paymentTypeArray); - consentDataJSON.add(jsonElementPaymentType); - } - - //Adding InstructionIdentification - JSONArray identificationArray = new JSONArray(); - identificationArray.add(initiation.getAsString(ConsentExtensionConstants.INSTRUCTION_IDENTIFICATION)); - - JSONObject jsonElementIdentification = new JSONObject(); - jsonElementIdentification.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.INSTRUCTION_IDENTIFICATION_TITLE); - jsonElementIdentification.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), - identificationArray); - consentDataJSON.add(jsonElementIdentification); - - //Adding EndToEndIdentification - JSONArray endToEndIdentificationArray = new JSONArray(); - endToEndIdentificationArray - .add(initiation.getAsString(ConsentExtensionConstants.END_TO_END_IDENTIFICATION)); - - JSONObject jsonElementEndToEndIdentification = new JSONObject(); - jsonElementEndToEndIdentification.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.END_TO_END_IDENTIFICATION_TITLE); - jsonElementEndToEndIdentification.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), - endToEndIdentificationArray); - consentDataJSON.add(jsonElementEndToEndIdentification); - - //Adding InstructedAmount - JSONObject instructedAmount = (JSONObject) initiation.get(ConsentExtensionConstants.INSTRUCTED_AMOUNT); - JSONArray instructedAmountArray = new JSONArray(); - - if (instructedAmount.getAsString(ConsentExtensionConstants.AMOUNT_TITLE) != null) { - instructedAmountArray.add(ConsentExtensionConstants.AMOUNT_TITLE + " : " + - instructedAmount.getAsString(ConsentExtensionConstants.AMOUNT)); - } - - if (instructedAmount.getAsString(ConsentExtensionConstants.CURRENCY) != null) { - instructedAmountArray.add(ConsentExtensionConstants.CURRENCY_TITLE + " : " + - instructedAmount.getAsString(ConsentExtensionConstants.CURRENCY)); - } - - JSONObject jsonElementInstructedAmount = new JSONObject(); - jsonElementInstructedAmount.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.INSTRUCTED_AMOUNT_TITLE); - jsonElementInstructedAmount.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), - instructedAmountArray); - consentDataJSON.add(jsonElementInstructedAmount); - - // Adding Debtor Account - populateDebtorAccount(initiation, consentDataJSON); - // Adding Creditor Account - populateCreditorAccount(initiation, consentDataJSON); - - } - } - - /** - * Populate account Details. - * - * @param data data request from the request - * @param consentDataJSON Consent information - */ - private static void populateAccountData(JSONObject data, JSONArray consentDataJSON) { - - //Adding Permissions - JSONArray permissions = (JSONArray) data.get(ConsentExtensionConstants.PERMISSIONS); - if (permissions != null) { - JSONObject jsonElementPermissions = new JSONObject(); - jsonElementPermissions.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.PERMISSIONS); - jsonElementPermissions.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), - permissions); - consentDataJSON.add(jsonElementPermissions); - } - - //Adding Expiration Date Time - String expirationDate = data.getAsString(ConsentExtensionConstants.EXPIRATION_DATE); - if (expirationDate != null) { - if (!ConsentRetrievalUtil.validateExpiryDateTime(expirationDate)) { - log.error(ErrorConstants.CONSENT_EXPIRED); - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.CONSENT_EXPIRED); - } - JSONArray expiryArray = new JSONArray(); - expiryArray.add(expirationDate); - - JSONObject jsonElementExpiry = new JSONObject(); - jsonElementExpiry.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.EXPIRATION_DATE_TITLE); - jsonElementExpiry.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), - expiryArray); - consentDataJSON.add(jsonElementExpiry); - } - - //Adding Transaction From Date Time - String fromDateTime = data.getAsString(ConsentExtensionConstants.TRANSACTION_FROM_DATE); - if (fromDateTime != null) { - JSONArray fromDateTimeArray = new JSONArray(); - fromDateTimeArray.add(fromDateTime); - - JSONObject jsonElementFromDateTime = new JSONObject(); - jsonElementFromDateTime.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.TRANSACTION_FROM_DATE_TITLE); - jsonElementFromDateTime.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), - fromDateTimeArray); - consentDataJSON.add(jsonElementFromDateTime); - } - - //Adding Transaction To Date Time - String toDateTime = data.getAsString(ConsentExtensionConstants.TRANSACTION_TO_DATE); - if (toDateTime != null) { - JSONArray toDateTimeArray = new JSONArray(); - toDateTimeArray.add(toDateTime); - - JSONObject jsonElementToDateTime = new JSONObject(); - jsonElementToDateTime.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.TRANSACTION_TO_DATE_TITLE); - jsonElementToDateTime.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), - toDateTimeArray); - consentDataJSON.add(jsonElementToDateTime); - } - } - - /** - * Populate funds confirmation Details. - * - * @param initiation data from the request - * @param consentDataJSON Consent information - */ - private static void populateCofData(JSONObject initiation, JSONArray consentDataJSON) { - - //Adding Expiration Date Time - if (initiation.getAsString(ConsentExtensionConstants.EXPIRATION_DATE) != null) { - - if (!ConsentRetrievalUtil - .validateExpiryDateTime(initiation.getAsString(ConsentExtensionConstants.EXPIRATION_DATE))) { - log.error(ErrorConstants.CONSENT_EXPIRED); - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.CONSENT_EXPIRED); - } - - String expiry = initiation.getAsString(ConsentExtensionConstants.EXPIRATION_DATE); - JSONArray expiryArray = new JSONArray(); - expiryArray.add(expiry); - - JSONObject jsonElementExpiry = new JSONObject(); - jsonElementExpiry.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.EXPIRATION_DATE_TITLE); - jsonElementExpiry.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), expiryArray); - consentDataJSON.add(jsonElementExpiry); - } else { - JSONArray expiryArray = new JSONArray(); - expiryArray.add(ConsentExtensionConstants.OPEN_ENDED_AUTHORIZATION); - - JSONObject jsonElementExpiry = new JSONObject(); - jsonElementExpiry.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.EXPIRATION_DATE_TITLE); - jsonElementExpiry.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), expiryArray); - consentDataJSON.add(jsonElementExpiry); - } - - if (initiation.get(ConsentExtensionConstants.DEBTOR_ACC) != null) { - //Adding Debtor Account - populateDebtorAccount(initiation, consentDataJSON); - } - } - - /** - * Populate VRP Details. - * - * @param data Control Parameters from the request - * @param consentDataJSON Consent information object - */ - private static void populateVRPData(JSONObject data, JSONArray consentDataJSON) { - - if (!data.containsKey(ConsentExtensionConstants.CONTROL_PARAMETERS)) { - log.error(ErrorConstants.CONTROL_PARAMETERS_MISSING_ERROR); - throw new ConsentException(ResponseStatus.BAD_REQUEST, - ErrorConstants.CONTROL_PARAMETERS_MISSING_ERROR); - } else { - - JSONObject controlParameters = (JSONObject) data. - get(ConsentExtensionConstants.CONTROL_PARAMETERS); - - //Adding Payment Type - JSONArray paymentTypeArray = new JSONArray(); - JSONObject jsonElementPaymentType = new JSONObject(); - paymentTypeArray.add(ConsentExtensionConstants.DOMESTIC_VRP); - jsonElementPaymentType.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.PAYMENT_TYPE_TITLE); - jsonElementPaymentType.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), - paymentTypeArray); - consentDataJSON.add(jsonElementPaymentType); - - String validToDateTime = controlParameters.getAsString(ConsentExtensionConstants.VALID_TO_DATE_TIME); - if (validToDateTime != null) { - // Constructing jsonElementValidToDataTime - JSONObject jsonElementValidToDateTime = new JSONObject(); - jsonElementValidToDateTime.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.CONTROL_PARAMETER_VALID_TO_DATE_TITLE); - JSONArray dateControlParameterArray = new JSONArray(); - dateControlParameterArray.add((controlParameters). - get(ConsentExtensionConstants.VALID_TO_DATE_TIME)); - jsonElementValidToDateTime.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), - dateControlParameterArray); - - consentDataJSON.add(jsonElementValidToDateTime); - } - - String validFromDateTime = controlParameters.getAsString - (ConsentExtensionConstants.VALID_FROM_DATE_TIME); - if (validFromDateTime != null) { - // Constructing jsonElementValidFromDataTime - JSONObject jsonElementValidFromDateTime = new JSONObject(); - jsonElementValidFromDateTime.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.CONTROL_PARAMETER_VALID_FROM_DATE_TITLE); - JSONArray dateTimeControlParameterArray = new JSONArray(); - dateTimeControlParameterArray.add((controlParameters). - get(ConsentExtensionConstants.VALID_FROM_DATE_TIME)); - jsonElementValidFromDateTime.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), - dateTimeControlParameterArray); - consentDataJSON.add(jsonElementValidFromDateTime); - } - - Object maxAmount = controlParameters.get(ConsentExtensionConstants.MAXIMUM_INDIVIDUAL_AMOUNT); - - if (maxAmount instanceof JSONObject) { - JSONObject jsonElementControlParameter = new JSONObject(); - jsonElementControlParameter.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.CONTROL_PARAMETER_MAX_INDIVIDUAL_AMOUNT_TITLE); - JSONArray controlParameterArray = new JSONArray(); - - JSONObject maximumIndividualAmount = (JSONObject) maxAmount; - - String formattedAmount = String.format("%s %s", - maximumIndividualAmount.getAsString(ConsentExtensionConstants.CURRENCY), - maximumIndividualAmount.getAsString(ConsentExtensionConstants.AMOUNT)); - controlParameterArray.add(formattedAmount); - jsonElementControlParameter.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), - controlParameterArray); - - consentDataJSON.add(jsonElementControlParameter); - } else { - log.error(ErrorConstants.MAX_AMOUNT_NOT_JSON_OBJECT_ERROR); - throw new ConsentException(ResponseStatus.BAD_REQUEST, - ErrorConstants.MAX_AMOUNT_NOT_JSON_OBJECT_ERROR); - } - - Object periodicLimit = controlParameters.get(ConsentExtensionConstants.PERIODIC_LIMITS); - - if (periodicLimit instanceof JSONArray) { - JSONArray periodicLimitsArrays = (JSONArray) periodicLimit; - - for (Object periodicLimitObject : periodicLimitsArrays) { - if (periodicLimitObject instanceof JSONObject) { - JSONObject jsonObject = (JSONObject) periodicLimitObject; - - Object periodAlignmentObject = jsonObject.get(ConsentExtensionConstants.PERIOD_ALIGNMENT); - - if (periodAlignmentObject instanceof String) { - // Constructing jsonElementPeriodAlignment - JSONObject jsonElementPeriodAlignment = new JSONObject(); - jsonElementPeriodAlignment.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.CONTROL_PARAMETER_PERIOD_ALIGNMENT_TITLE); - - JSONArray periodAlignmentArray = new JSONArray(); - periodAlignmentArray.add(periodAlignmentObject); - - jsonElementPeriodAlignment.appendField(StringUtils. - lowerCase(ConsentExtensionConstants.DATA), periodAlignmentArray); - consentDataJSON.add(jsonElementPeriodAlignment); - } else { - log.error(ErrorConstants.PERIOD_ALIGNMENT_NOT_STRING_ERROR); - throw new ConsentException(ResponseStatus.BAD_REQUEST, - ErrorConstants.PERIOD_ALIGNMENT_NOT_STRING_ERROR); - } - - Object periodTypeObject = jsonObject.get(ConsentExtensionConstants.PERIOD_TYPE); - - if (periodTypeObject instanceof String) { - - JSONObject jsonElementPeriodType = new JSONObject(); - jsonElementPeriodType.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.CONTROL_PARAMETER_PERIOD_TYPE_TITLE); - - JSONArray periodTypeArray = new JSONArray(); - periodTypeArray.add(periodTypeObject); - - jsonElementPeriodType.appendField(StringUtils.lowerCase(ConsentExtensionConstants.DATA), - periodTypeArray); - - consentDataJSON.add(jsonElementPeriodType); - - } else { - log.error(ErrorConstants.PERIOD_TYPE_NOT_STRING_ERROR); - throw new ConsentException(ResponseStatus.BAD_REQUEST, - ErrorConstants.PERIOD_TYPE_NOT_STRING_ERROR); - } - // Constructing jsonElementPeriodicLimitsAmountCurrency - periodicLimits amount and currency - Object amount = jsonObject.get(ConsentExtensionConstants.AMOUNT); - Object currency = jsonObject.get(ConsentExtensionConstants.CURRENCY); - - if (amount instanceof String && currency instanceof String) { - String periodTypeString = (String) periodTypeObject; - - JSONObject jsonElementPeriodicLimitsAmountCurrency = new JSONObject(); - jsonElementPeriodicLimitsAmountCurrency.appendField(ConsentExtensionConstants.TITLE, - ConsentExtensionConstants.CONTROL_PARAMETER_AMOUNT_TITLE + - periodTypeString); - - JSONArray periodicLimitsArray = new JSONArray(); - - String amountString = (String) amount; - String currencyString = (String) currency; - - String formattedPeriodicAmount = String.format("%s %s", currencyString, amountString); - periodicLimitsArray.add(formattedPeriodicAmount); - - jsonElementPeriodicLimitsAmountCurrency.appendField(StringUtils. - lowerCase(ConsentExtensionConstants.DATA), periodicLimitsArray); - consentDataJSON.add(jsonElementPeriodicLimitsAmountCurrency); - - } else { - log.error(ErrorConstants.NOT_STRING_ERROR); - throw new ConsentException(ResponseStatus.BAD_REQUEST, - ErrorConstants.NOT_STRING_ERROR); - } - } - } - } else { - log.error(ErrorConstants.NOT_JSON_ARRAY_ERROR); - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.NOT_JSON_ARRAY_ERROR); - } - } - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/ConsentMgrAuthServletImpl.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/ConsentMgrAuthServletImpl.java deleted file mode 100644 index bd53588a..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/ConsentMgrAuthServletImpl.java +++ /dev/null @@ -1,104 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authservlet.impl; - -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.consent.extensions.authservlet.impl.util.Constants; -import com.wso2.openbanking.accelerator.consent.extensions.authservlet.model.OBAuthServletInterface; -import org.json.JSONArray; -import org.json.JSONObject; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.ResourceBundle; - -import javax.servlet.http.HttpServletRequest; - -import static com.wso2.openbanking.accelerator.consent.extensions.authservlet.impl.util.Utils.i18n; - - -/** - * ConsentMgrAuthServletImpl - *

- * The consent management implementation of servlet extension that handles self-care portal use cases. - */ -public class ConsentMgrAuthServletImpl implements OBAuthServletInterface { - - @Override - public Map updateRequestAttribute(HttpServletRequest request, JSONObject dataSet, - ResourceBundle resourceBundle) { - - Map updatedRequestData = new HashMap<>(); - - boolean userClaimsConsentOnly = Boolean.parseBoolean(request.getParameter(Constants.USER_CLAIMS_CONSENT_ONLY)); - updatedRequestData.put("userClaimsConsentOnly", userClaimsConsentOnly); - - boolean displayScopes = (boolean) request.getSession().getAttribute("displayScopes"); - if (displayScopes) { - JSONArray openIdScopesArray = dataSet.getJSONArray("openid_scopes"); - if (openIdScopesArray != null) { - List oidScopes = new ArrayList<>(); - for (int scopeIndex = 0; scopeIndex < openIdScopesArray.length(); scopeIndex++) { - oidScopes.add(openIdScopesArray.getString(scopeIndex)); - } - updatedRequestData.put(Constants.OIDC_SCOPES, oidScopes); - } - } - - // Strings - updatedRequestData.put("openidUserClaims", i18n(resourceBundle, "openid.user.claims")); - updatedRequestData.put("requestAccessProfile", i18n(resourceBundle, "request.access.profile")); - updatedRequestData.put("requestedAttributes", i18n(resourceBundle, "requested.attributes")); - updatedRequestData.put("bySelectingFollowingAttributes", - i18n(resourceBundle, "by.selecting.following.attributes")); - updatedRequestData.put("mandatoryClaimsRecommendation", - i18n(resourceBundle, "mandatory.claims.recommendation")); - updatedRequestData.put("continueDefault", i18n(resourceBundle, "continue")); - updatedRequestData.put("deny", i18n(resourceBundle, "deny")); - - return updatedRequestData; - } - - @Generated(message = "ignoring since method doesn't contain a logic") - @Override - public Map updateSessionAttribute(HttpServletRequest request, JSONObject dataSet, - ResourceBundle resourceBundle) { - return new HashMap<>(); - } - - @Generated(message = "ignoring since method doesn't contain a logic") - @Override - public Map updateConsentData(HttpServletRequest request) { - return new HashMap<>(); - } - - @Generated(message = "ignoring since method doesn't contain a logic") - @Override - public Map updateConsentMetaData(HttpServletRequest request) { - return new HashMap<>(); - } - - @Generated(message = "ignoring since method doesn't contain a logic") - @Override - public String getJSPPath() { - return "/default_consent.jsp"; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/ISDefaultAuthServletImpl.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/ISDefaultAuthServletImpl.java deleted file mode 100644 index 8c60e950..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/ISDefaultAuthServletImpl.java +++ /dev/null @@ -1,125 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authservlet.impl; - -import com.wso2.openbanking.accelerator.consent.extensions.authservlet.impl.util.Constants; -import com.wso2.openbanking.accelerator.consent.extensions.authservlet.impl.util.Utils; -import com.wso2.openbanking.accelerator.consent.extensions.authservlet.model.OBAuthServletInterface; -import org.json.JSONArray; -import org.json.JSONObject; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.ResourceBundle; - -import javax.servlet.http.HttpServletRequest; - -/** - * The default implementation of servlet extension that handles non OB use cases. - * Required in other vanilla auth flows. - */ -public class ISDefaultAuthServletImpl implements OBAuthServletInterface { - - @Override - public Map updateRequestAttribute(HttpServletRequest request, JSONObject dataSet, - ResourceBundle resourceBundle) { - - Map returnMaps = new HashMap<>(); - - // Claims - if (request.getParameter(Constants.REQUESTED_CLAIMS) != null) { - - String[] requestedClaimList = request.getParameter(Constants.REQUESTED_CLAIMS) - .split(Constants.CLAIM_SEPARATOR); - - returnMaps.put("requestedClaims", Utils.splitClaims(requestedClaimList)); - } - - if (request.getParameter(Constants.MANDATORY_CLAIMS) != null) { - String[] mandatoryClaimList = request.getParameter(Constants.MANDATORY_CLAIMS) - .split(Constants.CLAIM_SEPARATOR); - - returnMaps.put("mandatoryClaims", Utils.splitClaims(mandatoryClaimList)); - } - - - // Scopes - /*This parameter decides whether the consent page will only be used to get consent for sharing claims with the - Service Provider. If this param is 'true' and user has already given consents for the OIDC scopes, we will be - hiding the scopes being displayed and the approve always button. - */ - boolean userClaimsConsentOnly = Boolean.parseBoolean(request.getParameter(Constants.USER_CLAIMS_CONSENT_ONLY)); - returnMaps.put("userClaimsConsentOnly", userClaimsConsentOnly); - - List oidScopes = new ArrayList<>(); - boolean displayScopes = (boolean) request.getSession().getAttribute("displayScopes"); - - if (userClaimsConsentOnly) { - // If we are getting consent for user claims only, we don't need to display OIDC scopes in the consent page - } else { - if (displayScopes) { - JSONArray openIdScopesArray = dataSet.getJSONArray("openid_scopes"); - if (openIdScopesArray != null) { - for (int scopeIndex = 0; scopeIndex < openIdScopesArray.length(); scopeIndex++) { - oidScopes.add(openIdScopesArray.getString(scopeIndex)); - } - returnMaps.put(Constants.OIDC_SCOPES, oidScopes); - } - } - } - - - // Strings - returnMaps.put("openidUserClaims", Utils.i18n(resourceBundle, "openid.user.claims")); - returnMaps.put("requestAccessProfile", Utils.i18n(resourceBundle, "request.access.profile")); - returnMaps.put("requestedAttributes", Utils.i18n(resourceBundle, "requested.attributes")); - returnMaps.put("bySelectingFollowingAttributes", - Utils.i18n(resourceBundle, "by.selecting.following.attributes")); - returnMaps.put("mandatoryClaimsRecommendation", - Utils.i18n(resourceBundle, "mandatory.claims.recommendation")); - returnMaps.put("continueDefault", Utils.i18n(resourceBundle, "continue")); - returnMaps.put("deny", Utils.i18n(resourceBundle, "deny")); - - return returnMaps; - - } - - @Override - public Map updateSessionAttribute(HttpServletRequest request, JSONObject dataSet, - ResourceBundle resourceBundle) { - return new HashMap<>(); - } - - @Override - public Map updateConsentData(HttpServletRequest request) { - return new HashMap<>(); - } - - @Override - public Map updateConsentMetaData(HttpServletRequest request) { - return new HashMap<>(); - } - - @Override - public String getJSPPath() { - return "/default_consent.jsp"; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/OBDefaultAuthServletImpl.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/OBDefaultAuthServletImpl.java deleted file mode 100644 index 6e33d788..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/OBDefaultAuthServletImpl.java +++ /dev/null @@ -1,96 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authservlet.impl; - -import com.wso2.openbanking.accelerator.consent.extensions.authservlet.impl.util.Utils; -import com.wso2.openbanking.accelerator.consent.extensions.authservlet.model.OBAuthServletInterface; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import org.json.JSONArray; -import org.json.JSONObject; - -import java.util.HashMap; -import java.util.Map; -import java.util.ResourceBundle; - -import javax.servlet.http.HttpServletRequest; - -/** - * The default implementation for OB flow. - */ -public class OBDefaultAuthServletImpl implements OBAuthServletInterface { - - private String jspPath; - @Override - public Map updateRequestAttribute(HttpServletRequest request, JSONObject dataSet, - ResourceBundle resourceBundle) { - - String consentType = dataSet.getString("type"); - - //store consent type in a global variable to return required jsp file in getJSPPath() method - jspPath = consentType; - - switch (consentType) { - - case ConsentExtensionConstants.ACCOUNTS: - return Utils.populateAccountsData(request, dataSet); - case ConsentExtensionConstants.PAYMENTS: - return Utils.populatePaymentsData(request, dataSet); - case ConsentExtensionConstants.FUNDSCONFIRMATIONS: - return Utils.populateCoFData(request, dataSet); - case ConsentExtensionConstants.VRP: - return Utils.populateVRPDataRetrieval(request, dataSet); - default: - return new HashMap<>(); - } - } - - @Override - public Map updateSessionAttribute(HttpServletRequest request, JSONObject dataSet, - ResourceBundle resourceBundle) { - - return new HashMap<>(); - } - - @Override - public Map updateConsentData(HttpServletRequest request) { - - Map returnMaps = new HashMap<>(); - - String[] accounts = request.getParameter("accounts[]").split(":"); - returnMaps.put("accountIds", new JSONArray(accounts)); - return returnMaps; - } - - @Override - public Map updateConsentMetaData(HttpServletRequest request) { - - return new HashMap<>(); - } - - @Override - public String getJSPPath() { - - if (jspPath.equalsIgnoreCase(ConsentExtensionConstants.ACCOUNTS) || - jspPath.equalsIgnoreCase(ConsentExtensionConstants.VRP)) { - return "/ob_default.jsp"; - } else { - return "/default_displayconsent.jsp"; - } - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/util/Constants.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/util/Constants.java deleted file mode 100644 index d94f25ca..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/util/Constants.java +++ /dev/null @@ -1,35 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authservlet.impl.util; - -/** - * Constants required for auth servlet implementations. - */ -public class Constants { - - private Constants() { - // do not required to create instances - } - - public static final String REQUESTED_CLAIMS = "requestedClaims"; - public static final String MANDATORY_CLAIMS = "mandatoryClaims"; - public static final String CLAIM_SEPARATOR = ","; - public static final String USER_CLAIMS_CONSENT_ONLY = "userClaimsConsentOnly"; - public static final String OIDC_SCOPES = "OIDCScopes"; -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/util/Utils.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/util/Utils.java deleted file mode 100644 index 73c6a91b..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/util/Utils.java +++ /dev/null @@ -1,296 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authservlet.impl.util; - -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.json.JSONArray; -import org.json.JSONObject; -import org.owasp.encoder.Encode; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.LinkedHashMap; -import java.util.List; -import java.util.Map; -import java.util.ResourceBundle; - -import javax.servlet.http.HttpServletRequest; - -/** - * Utility methods. - */ -public class Utils { - - private static final Log log = LogFactory.getLog(Utils.class); - - /** - * To get the property value for the given key from the ResourceBundle. - * Retrieve the value of property entry for key, return key if a value is not found for key - * - * @param resourceBundle ResourceBundle - * @param key Key - * @return Value of the property entry for key - */ - public static String i18n(ResourceBundle resourceBundle, String key) { - - try { - return Encode.forHtml((StringUtils.isNotBlank(resourceBundle.getString(key)) ? - resourceBundle.getString(key) : key)); - } catch (Exception e) { - // Intentionally catching Exception and if something goes wrong while finding the value for key, return - // default, not to break the UI - return Encode.forHtml(key); - } - } - - /** - * Split claims based on a deliminator and create map of claimID and displayName. - * - * @param requestedClaimList Requested claim list - * @return List of claims - */ - public static List> splitClaims(String[] requestedClaimList) { - - List> requestedClaims = new ArrayList<>(); - - for (String claim : requestedClaimList) { - String[] requestedClaimData = claim.split("_", 2); - if (requestedClaimData.length == 2) { - Map data = new HashMap<>(); - data.put("claimId", requestedClaimData[0]); - data.put("displayName", requestedClaimData[1]); - requestedClaims.add(data); - } - } - return requestedClaims; - } - - /** - * Method to populate accounts data to be sent to consent page. - * - * @param request HttpServletRequest - * @param dataSet Request payload JSONObject - * @return Map of Accounts data - */ - public static Map populateAccountsData(HttpServletRequest request, JSONObject dataSet) { - - Map returnMaps = new HashMap<>(); - - //Sets "data_requested" that contains the human-readable scope-requested information - JSONArray dataRequestedJsonArray = dataSet.getJSONArray(ConsentExtensionConstants.CONSENT_DATA); - Map> dataRequested = new LinkedHashMap<>(); - - for (int requestedDataIndex = 0; requestedDataIndex < dataRequestedJsonArray.length(); requestedDataIndex++) { - JSONObject dataObj = dataRequestedJsonArray.getJSONObject(requestedDataIndex); - String title = dataObj.getString(ConsentExtensionConstants.TITLE); - JSONArray dataArray = dataObj.getJSONArray(StringUtils.lowerCase(ConsentExtensionConstants.DATA)); - - ArrayList listData = new ArrayList<>(); - for (int dataIndex = 0; dataIndex < dataArray.length(); dataIndex++) { - listData.add(dataArray.getString(dataIndex)); - } - dataRequested.put(title, listData); - } - returnMaps.put(ConsentExtensionConstants.DATA_REQUESTED, dataRequested); - - // add accounts list - request.setAttribute(ConsentExtensionConstants.ACCOUNT_DATA, addAccList(dataSet)); - request.setAttribute(ConsentExtensionConstants.CONSENT_TYPE, ConsentExtensionConstants.ACCOUNTS); - - return returnMaps; - - } - - /** - * Method to populate payments data to be sent to consent page. - * - * @param request HttpServletRequest - * @param dataSet Request payload JSONObject - * @return Map of Payments data - */ - public static Map populatePaymentsData(HttpServletRequest request, JSONObject dataSet) { - - String selectedAccount = null; - Map returnMaps = new HashMap<>(); - - //Sets "data_requested" that contains the human-readable scope-requested information - JSONArray dataRequestedJsonArray = dataSet.getJSONArray(ConsentExtensionConstants.CONSENT_DATA); - Map> dataRequested = new LinkedHashMap<>(); - - for (int requestedDataIndex = 0; requestedDataIndex < dataRequestedJsonArray.length(); requestedDataIndex++) { - JSONObject dataObj = dataRequestedJsonArray.getJSONObject(requestedDataIndex); - String title = dataObj.getString(ConsentExtensionConstants.TITLE); - JSONArray dataArray = dataObj.getJSONArray(StringUtils.lowerCase(ConsentExtensionConstants.DATA)); - - ArrayList listData = new ArrayList<>(); - for (int dataIndex = 0; dataIndex < dataArray.length(); dataIndex++) { - listData.add(dataArray.getString(dataIndex)); - } - dataRequested.put(title, listData); - } - returnMaps.put(ConsentExtensionConstants.DATA_REQUESTED, dataRequested); - - //Assigning value of the "Debtor Account" key in the map to the variable "selectedAccount". - if (dataRequested.containsKey("Debtor Account")) { - selectedAccount = getDebtorAccFromConsentData(dataRequestedJsonArray); - } else { - // add accounts list - request.setAttribute(ConsentExtensionConstants.ACCOUNT_DATA, addAccList(dataSet)); - } - - request.setAttribute(ConsentExtensionConstants.SELECTED_ACCOUNT, selectedAccount); - request.setAttribute(ConsentExtensionConstants.CONSENT_TYPE, ConsentExtensionConstants.PAYMENTS); - - return returnMaps; - - } - /** - * Method to populate Confirmation of Funds data to be sent to consent page. - * - * @param httpServletRequest HttpServletRequest - * @param dataSet Request payload JSONObject - * @return Map of Confirmation of Funds data - */ - public static Map populateCoFData(HttpServletRequest httpServletRequest, JSONObject dataSet) { - - Map returnMaps = new HashMap<>(); - - //Sets "data_requested" that contains the human-readable scope-requested information - JSONArray dataRequestedJsonArray = dataSet.getJSONArray(ConsentExtensionConstants.CONSENT_DATA); - Map> dataRequested = new LinkedHashMap<>(); - - for (int requestedDataIndex = 0; requestedDataIndex < dataRequestedJsonArray.length(); requestedDataIndex++) { - JSONObject dataObj = dataRequestedJsonArray.getJSONObject(requestedDataIndex); - String title = dataObj.getString(ConsentExtensionConstants.TITLE); - JSONArray dataArray = dataObj.getJSONArray(StringUtils.lowerCase(ConsentExtensionConstants.DATA)); - - ArrayList listData = new ArrayList<>(); - for (int dataIndex = 0; dataIndex < dataArray.length(); dataIndex++) { - listData.add(dataArray.getString(dataIndex)); - } - dataRequested.put(title, listData); - - } - returnMaps.put(ConsentExtensionConstants.DATA_REQUESTED, dataRequested); - - //Assigning value of the "Debtor Account" key in the map to the variable "selectedAccount". - if (dataRequested.containsKey("Debtor Account")) { - httpServletRequest.setAttribute(ConsentExtensionConstants.DEBTOR_ACCOUNT_ID, - getDebtorAccFromConsentData(dataRequestedJsonArray)); - } - - httpServletRequest.setAttribute(ConsentExtensionConstants.CONSENT_TYPE, - ConsentExtensionConstants.FUNDSCONFIRMATIONS); - return returnMaps; - } - - /** - * Method to retrieve debtor account from consent data object. - * - * @param consentDataObject Object containing consent related data - * @return Debtor account - */ - public static String getDebtorAccFromConsentData(JSONArray consentDataObject) { - - for (int requestedDataIndex = 0; requestedDataIndex < consentDataObject.length(); requestedDataIndex++) { - JSONObject dataObj = consentDataObject.getJSONObject(requestedDataIndex); - String title = dataObj.getString(ConsentExtensionConstants.TITLE); - - if (ConsentExtensionConstants.DEBTOR_ACC_TITLE.equals(title)) { - JSONArray dataArray = dataObj.getJSONArray(StringUtils.lowerCase(ConsentExtensionConstants.DATA)); - - for (int dataIndex = 0; dataIndex < dataArray.length(); dataIndex++) { - String data = (String) dataArray.get(dataIndex); - if (data.contains(ConsentExtensionConstants.IDENTIFICATION_TITLE)) { - - //Values are set to the array as {name:value} Strings in Consent Retrieval step, - // hence splitting by : and getting the 2nd element to get the value - return (((String) dataArray.get(dataIndex)).split(":")[1]).trim(); - } - } - } - } - return null; - } - - private static List> addAccList (JSONObject dataSet) { - // add accounts list - List> accountData = new ArrayList<>(); - JSONArray accountsArray = dataSet.getJSONArray("accounts"); - for (int accountIndex = 0; accountIndex < accountsArray.length(); accountIndex++) { - JSONObject object = accountsArray.getJSONObject(accountIndex); - String accountId = object.getString(ConsentExtensionConstants.ACCOUNT_ID); - String displayName = object.getString(ConsentExtensionConstants.DISPLAY_NAME); - Map data = new HashMap<>(); - data.put(ConsentExtensionConstants.ACCOUNT_ID, accountId); - data.put(ConsentExtensionConstants.DISPLAY_NAME, displayName); - accountData.add(data); - } - - return accountData; - } - - /** - * Method to populate vrp data to be sent to consent page. - * - * @param request HttpServletRequest - * @param dataSet Request payload JSONObject - * @return Map of VRP data - */ - public static Map populateVRPDataRetrieval(HttpServletRequest request, JSONObject dataSet) { - - String selectedAccount = null; - Map returnMaps = new HashMap<>(); - - // Populates "consentDataArray" with the scope information in a readable format - JSONArray consentDataArray = dataSet.getJSONArray(ConsentExtensionConstants.CONSENT_DATA); - Map> dataRequested = new LinkedHashMap<>(); - - for (int requestedDataIndex = 0; requestedDataIndex < consentDataArray.length(); requestedDataIndex++) { - JSONObject dataObj = consentDataArray.getJSONObject(requestedDataIndex); - String title = dataObj.getString(ConsentExtensionConstants.TITLE); - JSONArray dataArray = dataObj.getJSONArray(StringUtils.lowerCase(ConsentExtensionConstants.DATA)); - - ArrayList listData = new ArrayList<>(); - for (int dataIndex = 0; dataIndex < dataArray.length(); dataIndex++) { - listData.add(dataArray.getString(dataIndex)); - } - dataRequested.put(title, listData); - } - returnMaps.put(ConsentExtensionConstants.DATA_REQUESTED, dataRequested); - - //Assigning value of the "Debtor Account" key in the map to the variable "selectedAccount". - if (dataRequested.containsKey("Debtor Account")) { - selectedAccount = getDebtorAccFromConsentData(consentDataArray); - } else { - // add accounts list - request.setAttribute(ConsentExtensionConstants.ACCOUNT_DATA, addAccList(dataSet)); - } - - request.setAttribute(ConsentExtensionConstants.SELECTED_ACCOUNT, selectedAccount); - request.setAttribute(ConsentExtensionConstants.CONSENT_TYPE, ConsentExtensionConstants.VRP); - - return returnMaps; - - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/model/OBAuthServletInterface.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/model/OBAuthServletInterface.java deleted file mode 100644 index 2c41ffd8..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/model/OBAuthServletInterface.java +++ /dev/null @@ -1,44 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authservlet.model; - -import org.json.JSONObject; - -import java.util.Map; -import java.util.ResourceBundle; - -import javax.servlet.http.HttpServletRequest; - -/** - * The interface to define how the servlet extension should be implemented. - */ -public interface OBAuthServletInterface { - - Map updateRequestAttribute(HttpServletRequest request, - JSONObject dataSet, ResourceBundle resourceBundle); - - Map updateSessionAttribute(HttpServletRequest request, - JSONObject dataSet, ResourceBundle resourceBundle); - - Map updateConsentData(HttpServletRequest request); - - Map updateConsentMetaData(HttpServletRequest request); - - String getJSPPath(); -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/authenticator/CIBAPushAuthenticator.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/authenticator/CIBAPushAuthenticator.java deleted file mode 100644 index 39f665d8..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/authenticator/CIBAPushAuthenticator.java +++ /dev/null @@ -1,365 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.ciba.authenticator; - -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.builder.ConsentStepsBuilder; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentData; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentPersistStep; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentRetrievalStep; -import com.wso2.openbanking.accelerator.consent.extensions.ciba.model.CIBAAuthenticationEndpointErrorResponse; -import com.wso2.openbanking.accelerator.consent.extensions.common.AuthErrorCode; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentCache; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionExporter; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionUtils; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import net.minidev.json.JSONObject; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.application.authentication.framework.cache.AuthenticationContextCache; -import org.wso2.carbon.identity.application.authentication.framework.cache.AuthenticationContextCacheEntry; -import org.wso2.carbon.identity.application.authentication.framework.cache.AuthenticationContextCacheKey; -import org.wso2.carbon.identity.application.authentication.framework.context.AuthenticationContext; -import org.wso2.carbon.identity.application.authentication.framework.exception.AuthenticationFailedException; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedUser; -import org.wso2.carbon.identity.application.authentication.framework.util.FrameworkUtils; -import org.wso2.carbon.identity.application.authenticator.push.PushAuthenticator; -import org.wso2.carbon.identity.application.authenticator.push.common.PushAuthContextManager; -import org.wso2.carbon.identity.application.authenticator.push.common.impl.PushAuthContextManagerImpl; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.oauth.cache.SessionDataCache; -import org.wso2.carbon.identity.oauth.cache.SessionDataCacheEntry; -import org.wso2.carbon.identity.oauth.cache.SessionDataCacheKey; -import org.wso2.carbon.identity.oauth2.model.OAuth2Parameters; - -import java.io.Serializable; -import java.io.UnsupportedEncodingException; -import java.net.URI; -import java.net.URISyntaxException; -import java.net.URLDecoder; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.Optional; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; - - -/** - * CIBA Push Authenticator for sending push notifications to authentication device. - */ -public class CIBAPushAuthenticator extends PushAuthenticator { - - private static final Log log = LogFactory.getLog(CIBAPushAuthenticator.class); - private static final long serialVersionUID = 6106269076155338045L; - - private static List consentRetrievalSteps = null; - private static List consentPersistSteps = null; - - public CIBAPushAuthenticator() { - initializeConsentSteps(); - } - - @Override - public String getFriendlyName() { - - return CIBAPushAuthenticatorConstants.AUTHENTICATOR_FRIENDLY_NAME; - } - - @Override - public String getName() { - - return CIBAPushAuthenticatorConstants.AUTHENTICATOR_NAME; - } - - /** - * Initialize consent builder. - */ - public static synchronized void initializeConsentSteps() { - - if (consentRetrievalSteps == null || consentPersistSteps == null) { - ConsentStepsBuilder consentStepsBuilder = ConsentExtensionExporter.getConsentStepsBuilder(); - - if (consentStepsBuilder != null) { - consentRetrievalSteps = consentStepsBuilder.getConsentRetrievalSteps(); - consentPersistSteps = consentStepsBuilder.getConsentPersistSteps(); - } - - if (consentRetrievalSteps != null && !consentRetrievalSteps.isEmpty()) { - log.info("Consent retrieval steps are not null or empty"); - } else { - log.warn("Consent retrieval steps are null or empty"); - } - if (consentPersistSteps != null && !consentPersistSteps.isEmpty()) { - log.info("Consent persist steps are not null or empty"); - } else { - log.warn("Consent persist steps are null or empty"); - } - } else { - log.debug("Retrieval and persist steps are available"); - } - } - - /** - * Execute consent retrieval steps. - * - * @param consentData Consent Data - * @param jsonObject Json object to store consent data - * @throws ConsentException when an error occurs while executing retrieval steps - */ - protected void executeRetrieval(ConsentData consentData, JSONObject jsonObject) throws ConsentException { - - for (ConsentRetrievalStep step : consentRetrievalSteps) { - if (log.isDebugEnabled()) { - log.debug("Executing retrieval step " + step.getClass().toString()); - } - step.execute(consentData, jsonObject); - } - } - - /** - * Retrieve consent. - * - * @param request HTTP request - * @param response HTTP response - * @param sessionDataKey Session data key - * @return Consent data - * @throws ConsentException when an error occurs while retrieving consent - */ - protected JSONObject retrieveConsent(HttpServletRequest request, HttpServletResponse response, - String sessionDataKey) throws ConsentException { - - String loggedInUser; - String app; - String spQueryParams; - String scopeString; - - SessionDataCacheEntry cacheEntry = ConsentCache.getCacheEntryFromSessionDataKey(sessionDataKey); - OAuth2Parameters oAuth2Parameters = cacheEntry.getoAuth2Parameters(); - URI redirectURI; - try { - redirectURI = new URI(oAuth2Parameters.getRedirectURI()); - } catch (URISyntaxException e) { - //Unlikely to happen. In case it happens, error response is sent - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Invalid redirect URI"); - } - //Extracting client ID for regulatory identification and redirect URI for error redirects - String clientId = oAuth2Parameters.getClientId(); - String state = oAuth2Parameters.getState(); - - Map sensitiveDataMap = - ConsentExtensionUtils.getSensitiveDataWithConsentKey(sessionDataKey); - - if ("false".equals(sensitiveDataMap.get(CIBAPushAuthenticatorConstants.IS_ERROR))) { - loggedInUser = (String) sensitiveDataMap.get(CIBAPushAuthenticatorConstants.LOGGED_IN_USER); - app = (String) sensitiveDataMap.get(CIBAPushAuthenticatorConstants.APPLICATION); - spQueryParams = (String) sensitiveDataMap.get(CIBAPushAuthenticatorConstants.SP_QUERY_PARAMS); - scopeString = (String) sensitiveDataMap.get(CIBAPushAuthenticatorConstants.SCOPE); - } else { - String isError = (String) sensitiveDataMap.get(CIBAPushAuthenticatorConstants.IS_ERROR); - //Have to throw standard error because cannot access redirect URI with this error - log.error("Error while getting endpoint parameters. " + isError); - throw new ConsentException(redirectURI, AuthErrorCode.SERVER_ERROR, - CIBAPushAuthenticatorConstants.ERROR_SERVER_ERROR, state); - } - - JSONObject jsonObject = new JSONObject(); - ConsentData consentData = createConsentData(sessionDataKey, loggedInUser, spQueryParams, scopeString, app, - request); - consentData.setSensitiveDataMap(sensitiveDataMap); - consentData.setRedirectURI(redirectURI); - - if (clientId == null) { - log.error("Client Id not available"); - //Unlikely error. Included just in case. - throw new ConsentException(redirectURI, AuthErrorCode.SERVER_ERROR, - CIBAPushAuthenticatorConstants.ERROR_SERVER_ERROR, state); - } - consentData.setClientId(clientId); - consentData.setState(state); - - try { - consentData.setRegulatory(IdentityCommonUtil.getRegulatoryFromSPMetaData(clientId)); - } catch (OpenBankingException e) { - log.error("Error while getting regulatory data", e); - throw new ConsentException(redirectURI, AuthErrorCode.SERVER_ERROR, "Error while obtaining regulatory data", - state); - } - - executeRetrieval(consentData, jsonObject); - if (consentData.getType() == null || consentData.getApplication() == null) { - log.error(CIBAPushAuthenticatorConstants.ERROR_NO_TYPE_AND_APP_DATA); - throw new ConsentException(consentData.getRedirectURI(), AuthErrorCode.SERVER_ERROR, - CIBAPushAuthenticatorConstants.ERROR_SERVER_ERROR, state); - } - ConsentExtensionUtils.setCommonDataToResponse(consentData, jsonObject); - try { - ConsentCache.addConsentDataToCache(sessionDataKey, consentData); - } catch (ConsentManagementException e) { - log.error("Error while adding consent data to cache", e); - throw new ConsentException(consentData.getRedirectURI(), AuthErrorCode.SERVER_ERROR, - CIBAPushAuthenticatorConstants.ERROR_SERVER_ERROR, state); - } - return jsonObject; - } - - @Generated(message = "This method is separated for unit testing purposes") - protected ConsentData createConsentData(String sessionDataKey, String loggedInUser, String spQueryParams, - String scopeString, String app, HttpServletRequest request) { - return new ConsentData(sessionDataKey, loggedInUser, spQueryParams, scopeString, app, - ConsentExtensionUtils.getHeaders(request)); - } - - /** - * Get the authenticated user. - * - * @param request Push authenticator HTTP request - * @return Authenticated User - */ - @Override - protected AuthenticatedUser getAuthenticatedUser(HttpServletRequest request) { - - // In OB CIBA, only this Push Authenticator IDP is expected to be executed during the CIBA auth flow - // Hence, the login_hint attribute in the CIBA request object is used to identify the user - return AuthenticatedUser.createLocalAuthenticatedUserFromSubjectIdentifier(request. - getParameter(CIBAPushAuthenticatorConstants.LOGIN_HINT)); - } - - @Generated(message = "This method is separated for unit testing purposes") - protected AuthenticationContext getAutenticationContext(String sessionDataKey) { - PushAuthContextManager contextManager = new PushAuthContextManagerImpl(); - - return contextManager.getContext(sessionDataKey); - } - - /** - * OB specific implementation to retrieve consent data. - * @param sessionDataKey Session data key - * @return consent data - * @throws AuthenticationFailedException Authentication failed exception - */ - @Override - protected Optional getAdditionalInfo(HttpServletRequest request, HttpServletResponse response, - String sessionDataKey) throws AuthenticationFailedException { - - AuthenticationContext context = getAutenticationContext(sessionDataKey); - - // update the authentication context with required values for OB specific requirements - try { - String queryParams = FrameworkUtils - .getQueryStringWithFrameworkContextId(context.getQueryParams(), context.getCallerSessionKey(), - context.getContextIdentifier()); - Map params = splitQuery(queryParams); - handlePreConsent(context, params); - } catch (UnsupportedEncodingException e) { - throw new AuthenticationFailedException("Error occurred when processing the request object", e); - } - - SessionDataCacheKey cacheKey = ConsentCache.getCacheKey(sessionDataKey); - SessionDataCacheEntry cacheEntry = ConsentCache.getCacheEntryFromCacheKey(cacheKey); - - cacheEntry.setLoggedInUser(context.getSubject()); - SessionDataCache.getInstance().addToCache(cacheKey, cacheEntry); - - // Authentication context is added to cache as it is obtained from the cache in a later step by the Parameter - // Resolver object - AuthenticationContextCache.getInstance().addToCache( - new AuthenticationContextCacheKey(sessionDataKey), new AuthenticationContextCacheEntry(context)); - - JSONObject additionalInfo = retrieveConsent(request, response, sessionDataKey); - String bindingMessage = request.getParameter(CIBAPushAuthenticatorConstants.BINDING_MESSAGE); - if (StringUtils.isNotEmpty(bindingMessage)) { - additionalInfo.put(CIBAPushAuthenticatorConstants.BINDING_MESSAGE, bindingMessage); - } - return Optional.ofNullable(additionalInfo.toJSONString()); - } - - /** - * set attributes to context which will be required to prompt the consent page. - * - * @param context authentication context - * @param params query params - */ - protected void handlePreConsent(AuthenticationContext context, Map params) { - ServiceProvider serviceProvider = context.getSequenceConfig().getApplicationConfig().getServiceProvider(); - - context.addEndpointParam(CIBAPushAuthenticatorConstants.LOGGED_IN_USER, - params.get(CIBAPushAuthenticatorConstants.LOGIN_HINT)); - context.addEndpointParam(CIBAPushAuthenticatorConstants.USER_TENANT_DOMAIN, - "@carbon.super"); - context.addEndpointParam(CIBAPushAuthenticatorConstants.REQUEST, - params.get(CIBAPushAuthenticatorConstants.REQUEST_OBJECT)); - context.addEndpointParam(CIBAPushAuthenticatorConstants.SCOPE, - params.get(CIBAPushAuthenticatorConstants.SCOPE)); - context.addEndpointParam(CIBAPushAuthenticatorConstants.APPLICATION, serviceProvider.getApplicationName()); - context.addEndpointParam(CIBAPushAuthenticatorConstants.CONSENT_PROMPTED, true); - context.addEndpointParam(CIBAPushAuthenticatorConstants.AUTH_REQ_ID, - context.getAuthenticationRequest().getRequestQueryParams() - .get(CIBAPushAuthenticatorConstants.NONCE)[0]); - } - - /** - * Returns a map of query parameters from the given query param string. - * @param queryParamsString HTTP request query parameters - * @return Query parameter map - * @throws UnsupportedEncodingException Unsupported encoding exception - */ - protected Map splitQuery(String queryParamsString) throws UnsupportedEncodingException { - final Map queryParams = new HashMap<>(); - final String[] pairs = queryParamsString.split("&"); - for (String pair : pairs) { - final int idx = pair.indexOf("="); - final String key = idx > 0 ? URLDecoder.decode(pair.substring(0, idx), "UTF-8") : pair; - final String value = - idx > 0 && pair.length() > idx + 1 ? URLDecoder.decode(pair.substring(idx + 1), "UTF-8") : null; - queryParams.put(key, value); - } - return queryParams; - } - - /** - * Extend this method to create error response on toolkits. Set necessary status codes and error payloads to - * CIBAAuthenticationEndpointErrorResponse. - * - * @param httpStatusCode Http status code - * @param errorCode Error code - * @param errorDescription Error description - * @return CIBAAuthenticationEndpointErrorResponse CIBA Authentication Endpoint Error Response - */ - public static CIBAAuthenticationEndpointErrorResponse createErrorResponse(int httpStatusCode, String errorCode, - String errorDescription) { - - CIBAAuthenticationEndpointErrorResponse cibaPushServletErrorResponse = - new CIBAAuthenticationEndpointErrorResponse(); - JSONObject errorResponse = new JSONObject(); - errorResponse.put(CIBAPushAuthenticatorConstants.ERROR_DESCRIPTION, errorDescription); - errorResponse.put(CIBAPushAuthenticatorConstants.ERROR, errorCode); - cibaPushServletErrorResponse.setPayload(errorResponse); - cibaPushServletErrorResponse.setHttpStatusCode(httpStatusCode); - - return cibaPushServletErrorResponse; - } - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/authenticator/CIBAPushAuthenticatorConstants.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/authenticator/CIBAPushAuthenticatorConstants.java deleted file mode 100644 index 76e4e491..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/authenticator/CIBAPushAuthenticatorConstants.java +++ /dev/null @@ -1,49 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.ciba.authenticator; - -/** - * CIBA Push Authenticator Constants. - */ -public class CIBAPushAuthenticatorConstants { - - public static final String AUTHENTICATOR_NAME = "ciba"; - public static final String AUTHENTICATOR_FRIENDLY_NAME = "CIBA Authenticator"; - public static final String REQUEST = "request"; - public static final String REQUEST_OBJECT = "request_object"; - public static final String BINDING_MESSAGE = "binding_message"; - - //Consent Related constants - public static final String LOGGED_IN_USER = "loggedInUser"; - public static final String USER_TENANT_DOMAIN = "userTenantDomain"; - public static final String SCOPE = "scope"; - public static final String APPLICATION = "application"; - public static final String CONSENT_PROMPTED = "consentPrompted"; - public static final String AUTH_REQ_ID = "auth_req_id"; - public static final String NONCE = "nonce"; - public static final String LOGIN_HINT = "login_hint"; - public static final String SP_QUERY_PARAMS = "spQueryParams"; - - // error constants - public static final String IS_ERROR = "isError"; - public static final String ERROR_SERVER_ERROR = "Internal server error"; - public static final String ERROR_NO_TYPE_AND_APP_DATA = "Type and application data is unavailable"; - public static final String ERROR_DESCRIPTION = "error_description"; - public static final String ERROR = "error"; -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/impl/CIBAAuthenticationEndpointDefaultImpl.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/impl/CIBAAuthenticationEndpointDefaultImpl.java deleted file mode 100644 index cfa00001..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/impl/CIBAAuthenticationEndpointDefaultImpl.java +++ /dev/null @@ -1,34 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.ciba.impl; - -import com.wso2.openbanking.accelerator.consent.extensions.ciba.model.CIBAAuthenticationEndpointInterface; -import net.minidev.json.JSONObject; - -/** - * Implementation to extend CIBA push servlet consent persistence data. - */ -public class CIBAAuthenticationEndpointDefaultImpl implements CIBAAuthenticationEndpointInterface { - - @Override - public JSONObject updateConsentData(JSONObject consentData) { - return consentData; - } - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/model/CIBAAuthenticationEndpointErrorResponse.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/model/CIBAAuthenticationEndpointErrorResponse.java deleted file mode 100644 index 6c640b33..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/model/CIBAAuthenticationEndpointErrorResponse.java +++ /dev/null @@ -1,50 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.ciba.model; - -import net.minidev.json.JSONObject; - -/** - * CIBA authentication endpoint error response. - */ -public class CIBAAuthenticationEndpointErrorResponse { - - private int httpStatusCode = 0; - private JSONObject payload = null; - - public int getHttpStatusCode() { - - return httpStatusCode; - } - public void setHttpStatusCode(int httpStatusCode) { - - this.httpStatusCode = httpStatusCode; - } - - public JSONObject getPayload() { - - return payload; - } - public void setPayload(JSONObject payload) { - - this.payload = payload; - } - -} - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/model/CIBAAuthenticationEndpointInterface.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/model/CIBAAuthenticationEndpointInterface.java deleted file mode 100644 index a84abdc2..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/model/CIBAAuthenticationEndpointInterface.java +++ /dev/null @@ -1,34 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.ciba.model; - -import net.minidev.json.JSONObject; - -/** - * The interface to extend CIBA push servlet consent persistence data. - */ -public interface CIBAAuthenticationEndpointInterface { - - /** - * Set additional data to consent data. - * @param consentData consent data json object - * @return updated consent data json object - */ - JSONObject updateConsentData(JSONObject consentData); -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/AuthErrorCode.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/AuthErrorCode.java deleted file mode 100644 index 67234110..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/AuthErrorCode.java +++ /dev/null @@ -1,112 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.common; - -/** - * Enum of the error redirect codes from both OAuth 2.0 and OIDC Core 1.0 specifications. - */ -public enum AuthErrorCode { - - /** - * invalid_request, see ... - */ - INVALID_REQUEST("invalid_request"), - /** - * unauthorized_client, see ... - */ - UNAUTHORIZED_CLIENT("unauthorized_client"), - /** - * access_denied, see ... - */ - ACCESS_DENIED("access_denied"), - /** - * unsupported_response_type, see ... - */ - UNSUPPORTED_RESPONSE_TYPE("unsupported_response_type"), - /** - * invalid_scope, see ... - */ - INVALID_SCOPE("invalid_scope"), - /** - * server_error, see ... - */ - SERVER_ERROR("server_error"), - /** - * temporarily_unavailable, see ... - */ - TEMPORARILY_UNAVAILABLE("temporarily_unavailable"), - /** - * interaction_required, see - * OpenID Connect Core 1.0}. - */ - INTERACTION_REQUIRED("interaction_required"), - /** - * login_required, see - * OpenID Connect Core 1.0}. - */ - LOGIN_REQUIRED("login_required"), - /** - * account_selection_required, see - * OpenID Connect Core 1.0}. - */ - ACCOUNT_SELECTION_REQUIRED("account_selection_required"), - /** - * consent_required, see - * OpenID Connect Core 1.0}. - */ - CONSENT_REQUIRED("consent_required"), - /** - * invalid_request_uri, see - * OpenID Connect Core 1.0}. - */ - INVALID_REQUEST_URI("invalid_request_uri"), - /** - * invalid_request_object, see - * OpenID Connect Core 1.0}. - */ - INVALID_REQUEST_OBJECT("invalid_request_object"), - /** - * request_not_supported, see - * OpenID Connect Core 1.0}. - */ - REQUEST_NOT_SUPPORTED("request_not_supported"), - /** - * request_uri_not_supported, see - * OpenID Connect Core 1.0}. - */ - REQUEST_URI_NOT_SUPPORTED("request_uri_not_supported"), - /** - * registration_not_supported, see - * OpenID Connect Core 1.0}. - */ - REGISTRATION_NOT_SUPPORTED("registration_not_supported"); - - private final String errorCode; - - AuthErrorCode(String errorCode) { - - this.errorCode = errorCode; - } - - @Override - public String toString() { - - return errorCode; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentCache.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentCache.java deleted file mode 100644 index 7c9cbf35..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentCache.java +++ /dev/null @@ -1,189 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.common; - -import com.google.gson.Gson; -import com.google.gson.JsonObject; -import com.google.gson.JsonParser; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentData; -import com.wso2.openbanking.accelerator.consent.mgt.dao.constants.ConsentMgtDAOConstants; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import com.wso2.openbanking.accelerator.identity.cache.IdentityCache; -import com.wso2.openbanking.accelerator.identity.cache.IdentityCacheKey; -import net.minidev.json.JSONValue; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.oauth.cache.SessionDataCache; -import org.wso2.carbon.identity.oauth.cache.SessionDataCacheEntry; -import org.wso2.carbon.identity.oauth.cache.SessionDataCacheKey; - -import java.net.URISyntaxException; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Map; -import java.util.Set; - - -/** - * Class for maintaining Consent Cache. - */ -public class ConsentCache { - - private static volatile IdentityCache consentCache; - - private static Log log = LogFactory.getLog(ConsentCache.class); - - private static ConsentCoreServiceImpl consentCoreService = new ConsentCoreServiceImpl(); - - private static final String preserveConsent = (String) OpenBankingConfigParser.getInstance().getConfiguration() - .get(ConsentExtensionConstants.PRESERVE_CONSENT); - private static boolean storeConsent = preserveConsent == null ? false : Boolean.parseBoolean(preserveConsent); - - /** - * Get consent cache instance. - * @return consent cache instance - */ - public static IdentityCache getInstance() { - if (consentCache == null) { - synchronized (ConsentCache.class) { - if (consentCache == null) { - consentCache = new IdentityCache(); - } - } - } - - return consentCache; - } - - /** - * Add consent data to consent data cache. - * @param sessionDataKey session data key - * @param consentData consent data - * @throws ConsentManagementException if an error occurs while adding consent data to cache - */ - public static void addConsentDataToCache(String sessionDataKey, ConsentData consentData) - throws ConsentManagementException { - - ConsentCache.getInstance().addToCache(IdentityCacheKey.of(sessionDataKey), - consentData); - - storeConsent(consentData, sessionDataKey); - } - - /** - * Add consent data to database. - * @param sessionDataKey session data key - * @param consentData consent data - * @throws ConsentManagementException if an error occurs while storing consent data - */ - public static void storeConsent(ConsentData consentData, String sessionDataKey) throws ConsentManagementException { - - Gson gson = new Gson(); - if (storeConsent) { - String consent = gson.toJson(consentData); - Map authorizeData = new HashMap<>(); - authorizeData.put(consentData.getSessionDataKey(), consent); - if (consentCoreService.getConsentAttributesByName(sessionDataKey).isEmpty()) { - consentCoreService.storeConsentAttributes(consentData.getConsentId(), authorizeData); - } - } - } - - /** - * Get new session data cache key using session data key. - * @param sessionDataKey Session data key - * @return session data cache key - */ - public static SessionDataCacheKey getCacheKey(String sessionDataKey) { - - return new SessionDataCacheKey(sessionDataKey); - } - - /** - * Get session data cache entry by session data cache key. - * @param cacheKey session data cache key - * @return Session data cache entry - */ - public static SessionDataCacheEntry getCacheEntryFromCacheKey(SessionDataCacheKey cacheKey) { - - return SessionDataCache.getInstance().getValueFromCache(cacheKey); - } - - /** - * Get Consent data from the consent cache. - * @param sessionDataKey Session data key - * @return consent data - */ - public static ConsentData getConsentDataFromCache(String sessionDataKey) { - - ConsentData consentData = (ConsentData) ConsentCache.getInstance() - .getFromCache(IdentityCacheKey.of(sessionDataKey)); - if (consentData == null) { - if (storeConsent) { - Map consentDetailsMap = - null; - try { - consentDetailsMap = consentCoreService.getConsentAttributesByName(sessionDataKey); - if (consentDetailsMap.isEmpty()) { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Unable to get consent data"); - } - Set keys = consentDetailsMap.keySet(); - String consentId = new ArrayList<>(keys).get(0); - JsonObject consentDetails = new JsonParser() - .parse(consentDetailsMap.get(consentId)).getAsJsonObject(); - consentData = ConsentExtensionUtils.getConsentDataFromAttributes(consentDetails, sessionDataKey); - - if (consentDetailsMap.isEmpty()) { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Unable to get consent data"); - } - // remove all session data related to the consent from consent attributes - ArrayList keysToDelete = new ArrayList<>(); - - Map consentAttributes = consentCoreService. - getConsentAttributes(consentData.getConsentId()).getConsentAttributes(); - - consentAttributes.forEach((key, value) -> { - if (JSONValue.isValidJson(value) && value.contains(ConsentMgtDAOConstants.SESSION_DATA_KEY)) { - keysToDelete.add(key); - } - }); - consentCoreService.deleteConsentAttributes(consentData.getConsentId(), keysToDelete); - } catch (ConsentManagementException | URISyntaxException e) { - log.error("Error while retrieving consent data from cache", e); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Unable to get consent data"); - } - } else { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Unable to get consent data"); - } - } - return consentData; - } - - /** - * Get Cache Entry by Session Data Key. - * @param sessionDataKey Session Data Key - * @return Session data cache entry - */ - public static SessionDataCacheEntry getCacheEntryFromSessionDataKey(String sessionDataKey) { - - return ConsentCache.getCacheEntryFromCacheKey(ConsentCache.getCacheKey(sessionDataKey)); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentException.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentException.java deleted file mode 100644 index 8bad616d..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentException.java +++ /dev/null @@ -1,130 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.common; - -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.io.UnsupportedEncodingException; -import java.net.URI; -import java.net.URISyntaxException; -import java.net.URLEncoder; -import java.nio.charset.StandardCharsets; - -/** - * Consent exception class to be used in consent components and extensions. - */ -public class ConsentException extends RuntimeException { - - private static final Log log = LogFactory.getLog(ConsentException.class); - - private JSONObject payload; - private ResponseStatus status; - private URI errorRedirectURI; - - public ConsentException(ResponseStatus status, JSONObject payload, Throwable cause) { - - super(cause); - this.status = status; - this.payload = payload; - } - - public ConsentException(ResponseStatus status, JSONObject payload) { - - this.status = status; - this.payload = payload; - } - - public ConsentException(ResponseStatus status, String description) { - - this.status = status; - this.payload = createDefaultErrorObject(this.status, description); - } - - /** - * This method is created to send error redirects in in the authorization flow. The parameter validations are done - * in compliance with the OAuth2 and OIDC specifications. - * - * @param errorRedirectURI REQUIRED The base URI which the redirect should go to. - * @param error REQUIRED The error code of the error. Should be a supported value in OAuth2/OIDC - * @param errorDescription OPTIONAL The description of the error. - * @param state REQUIRED if a "state" parameter was present in the client authorization request. - */ - public ConsentException(URI errorRedirectURI, AuthErrorCode error, String errorDescription, String state) { - - if (errorRedirectURI != null && error != null) { - try { - //add 302 as error code since this will be a redirect - this.status = ResponseStatus.FOUND; - //set parameters as uri fragments - //https://openid.net/specs/oauth-v2-multiple-response-types-1_0.html#rfc.section.5 - String errorResponse = ConsentExtensionConstants.ERROR_URI_FRAGMENT - .concat(URLEncoder.encode(error.toString(), StandardCharsets.UTF_8.toString())); - if (errorDescription != null) { - errorResponse = errorResponse.concat(ConsentExtensionConstants.ERROR_DESCRIPTION_PARAMETER) - .concat(URLEncoder.encode(errorDescription, StandardCharsets.UTF_8.toString())); - } - if (state != null) { - errorResponse = errorResponse.concat(ConsentExtensionConstants.STATE_PARAMETER) - .concat(URLEncoder.encode(state, StandardCharsets.UTF_8.toString())); - } - this.errorRedirectURI = new URI(errorRedirectURI.toString().concat(errorResponse)); - - } catch (URISyntaxException | UnsupportedEncodingException e) { - log.error("Error while building the uri", e); - } - } - } - - public JSONObject createDefaultErrorObject(ResponseStatus status, String description) { - - JSONObject error = new JSONObject(); - JSONArray errorList = new JSONArray(); - JSONObject errorObj = new JSONObject(); - error.appendField("Code", String.valueOf(status.getStatusCode())); - error.appendField("Message", status.getReasonPhrase()); - errorObj.appendField("ErrorCode", String.valueOf(status.getStatusCode())); - errorObj.appendField("Message", description); - errorList.appendElement(errorObj); - error.appendField("Errors", errorList); - return error; - } - - public JSONObject getPayload() { - - return payload; - } - - public ResponseStatus getStatus() { - - return status; - } - - public URI getErrorRedirectURI() { - - return errorRedirectURI; - } - - public void setErrorRedirectURI(URI errorRedirectURI) { - - this.errorRedirectURI = errorRedirectURI; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentExtensionConstants.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentExtensionConstants.java deleted file mode 100644 index 4062a1bf..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentExtensionConstants.java +++ /dev/null @@ -1,227 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.common; - -/** - * Constant class for consent extension module. - */ -public class ConsentExtensionConstants { - - public static final String ERROR_URI_FRAGMENT = "#error="; - public static final String ERROR_DESCRIPTION_PARAMETER = "&error_description="; - public static final String STATE_PARAMETER = "&state="; - public static final String PRESERVE_CONSENT = "Consent.PreserveConsentLink"; - public static final String SENSITIVE_DATA_MAP = "sensitiveDataMap"; - public static final String LOGGED_IN_USER = "loggedInUser"; - public static final String SP_QUERY_PARAMS = "spQueryParams"; - public static final String SCOPES = "scopeString"; - public static final String APPLICATION = "application"; - public static final String REQUEST_HEADERS = "requestHeaders"; - public static final String REQUEST_URI = "redirectURI"; - public static final String USERID = "userId"; - public static final String CONSENT_ID = "ConsentId"; - public static final String CONSENT_ID_VALIDATION = "ConsentId"; - public static final String CLIENT_ID = "clientId"; - public static final String REGULATORY = "regulatory"; - public static final String CONSENT_RESOURCE = "consentResource"; - public static final String AUTH_RESOURCE = "authResource"; - public static final String META_DATA = "metaDataMap"; - public static final String TYPE = "type"; - public static final String X_IDEMPOTENCY_KEY = "x-idempotency-key"; - public static final String IS_VALID = "isValid"; - public static final String HTTP_CODE = "httpCode"; - public static final String ERRORS = "errors"; - public static final String PAYMENTS = "payments"; - public static final String VRP = "vrp"; - - public static final String DATA = "Data"; - public static final String INITIATION = "Initiation"; - public static final String STATUS = "Status"; - public static final String STATUS_UPDATE_TIME = "StatusUpdateDateTime"; - public static final String CREATION_DATE_TIME = "CreationDateTime"; - public static final String FUNDSCONFIRMATIONS = "fundsconfirmations"; - public static final String SCHEME_NAME = "SchemeName"; - public static final String IDENTIFICATION = "Identification"; - public static final String NAME = "Name"; - public static final String SECONDARY_IDENTIFICATION = "SecondaryIdentification"; - public static final String OB_SORT_CODE_ACCOUNT_NUMBER = "OB.SortCodeAccountNumber"; - public static final String SORT_CODE_ACCOUNT_NUMBER = "SortCodeAccountNumber"; - public static final int ACCOUNT_IDENTIFICATION_LENGTH = 14; - public static final String SORT_CODE_PATTERN = "^[0-9]{6}[0-9]{8}$"; - public static final String CUSTOM_LOCAL_INSTRUMENT_VALUES = "Consent.CustomLocalInstrumentValues"; - public static final String AMOUNT = "Amount"; - public static final String CREDITOR_ACC = "CreditorAccount"; - public static final String DEBTOR_ACC = "DebtorAccount"; - public static final String INSTRUCTED_AMOUNT = "InstructedAmount"; - public static final String LOCAL_INSTRUMENT = "LocalInstrument"; - public static final String ACCOUNT_CONSENT_GET_PATH = "account-access-consents"; - public static final String ACCOUNT_CONSENT_DELETE_PATH = "account-access-consents/"; - public static final String UUID_REGEX = - "[0-9a-fA-F]{8}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{12}"; - public static final String INTERACTION_ID_HEADER = "x-fapi-interaction-id"; - public static final String PERMISSIONS = "Permissions"; - public static final String COF_CONSENT_PATH = "funds-confirmation-consents"; - public static final String PAYMENT_CONSENT_PATH = "payment-consents"; - public static final String CONSENT_KEY = "OauthConsentKey"; - public static final String REQUEST_KEY = "AuthRequestKey"; - public static final String CUTOFF_DATE_ENABLED = "Consent.PaymentRestrictions.CutOffDateTime.Enabled"; - public static final String MAX_INSTRUCTED_AMOUNT = "Consent.PaymentRestrictions" + - ".MaximumInstructedAmount"; - public static final String DAILY_CUTOFF = "Consent.PaymentRestrictions.CutOffDateTime" + - ".DailyCutOffTime"; - public static final String REJECT = "REJECT"; - public static final String CUTOFF_DATE_POLICY = "Consent.PaymentRestrictions.CutOffDateTime" + - ".CutOffDateTimePolicy"; - public static final String ACCEPT = "ACCEPT"; - public static final String ZONE_ID = "ZoneId"; - public static final String IS_ERROR = "isError"; - public static final String ACCOUNTS = "accounts"; - public static final String CONSENT_DATA = "consentData"; - public static final String TITLE = "title"; - public static final String DEBTOR_ACCOUNT_ID = "AccountId"; - public static final String ACCOUNT_ID = "account_id"; - public static final String DATA_REQUESTED = "data_requested"; - public static final String PAYMENT_ACCOUNT = "paymentAccount"; - public static final String COF_ACCOUNT = "cofAccount"; - public static final String AWAITING_AUTH_STATUS = "awaitingAuthorisation"; - public static final String CUT_OFF_DATE_TIME = "CutOffDateTime"; - public static final String IDEMPOTENCY_KEY = "IdempotencyKey"; - public static final int NUMBER_OF_PARTS_IN_JWS = 3; - public static final String CLAIMS = "claims"; - public static final String[] CLAIM_FIELDS = new String[]{"userinfo", "id_token"}; - public static final String OPENBANKING_INTENT_ID = "openbanking_intent_id"; - public static final String VALUE = "value"; - public static final String AUTHORIZED_STATUS = "authorised"; - public static final String EXPIRATION_DATE = "ExpirationDateTime"; - public static final String EXPIRATION_DATE_TITLE = "Expiration Date Time"; - public static final String INSTRUCTED_AMOUNT_TITLE = "Instructed Amount"; - public static final String CURRENCY_TITLE = "Currency"; - public static final String CURRENCY = "Currency"; - public static final String AMOUNT_TITLE = "Amount"; - public static final String END_TO_END_IDENTIFICATION_TITLE = "End to End Identification"; - public static final String END_TO_END_IDENTIFICATION = "EndToEndIdentification"; - public static final String INSTRUCTION_IDENTIFICATION_TITLE = "Instruction Identification"; - public static final String INSTRUCTION_IDENTIFICATION = "InstructionIdentification"; - public static final String REJECTED_STATUS = "rejected"; - public static final String OPEN_ENDED_AUTHORIZATION = "Open Ended Authorization Requested"; - public static final String DEBTOR_ACC_TITLE = "Debtor Account"; - public static final String SCHEME_NAME_TITLE = "Scheme Name"; - public static final String IDENTIFICATION_TITLE = "Identification"; - public static final String NAME_TITLE = "Name"; - public static final String SECONDARY_IDENTIFICATION_TITLE = "Secondary Identification"; - public static final String CREDITOR_ACC_TITLE = "Creditor Account"; - public static final String CONSENT_TYPE = "consent_type"; - public static final String TRANSACTION_FROM_DATE = "TransactionFromDateTime"; - public static final String TRANSACTION_TO_DATE = "TransactionToDateTime"; - public static final String TRANSACTION_FROM_DATE_TITLE = "Transaction From Date Time"; - public static final String TRANSACTION_TO_DATE_TITLE = "Transaction To Date Time"; - public static final String PAYMENT_TYPE_TITLE = "Payment Type"; - public static final String CURRENCY_OF_TRANSFER_TITLE = "Currency of Transfer"; - public static final String CURRENCY_OF_TRANSFER = "CurrencyOfTransfer"; - public static final String INTERNATIONAL_PAYMENTS = "International Payments"; - public static final String DOMESTIC_PAYMENTS = "Domestic Payments"; - public static final String CREATED_STATUS = "created"; - public static final String IS_VALID_PAYLOAD = "isValidPayload"; - public static final String ERROR_CODE = "errorCode"; - public static final String ERROR_MESSAGE = "errorMessage"; - public static final String RISK = "Risk"; - public static final String COF_CONSENT_INITIATION_PATH = "/funds-confirmation-consents"; - public static final String COF_CONSENT_CONSENT_ID_PATH = "/funds-confirmation-consents/{ConsentId}"; - public static final String COF_SUBMISSION_PATH = "/funds-confirmations"; - public static final String ACCOUNT_ID_LIST = "AccountIds"; - public static final String CREATION_TIME = "CreationDateTime"; - public static final String LINKS = "Links"; - public static final String SELF = "Self"; - public static final String META = "Meta"; - public static final String ACCOUNTS_SELF_LINK = "Consent.AccountAPIURL"; - public static final String PAYMENT_SELF_LINK = "Consent.PaymentAPIURL"; - public static final String COF_SELF_LINK = "Consent.FundsConfirmationAPIURL"; - public static final String VRP_SELF_LINK = "Consent.VRPAPIURL"; - public static final String REVOKED_STATUS = "revoked"; - public static final String DISPLAY_NAME = "display_name"; - public static final String ACCOUNT_DATA = "account_data"; - public static final String SELECTED_ACCOUNT = "selectedAccount"; - public static final String PAYMENT_COF_PATH = "funds-confirmation"; - public static final String AWAITING_UPLOAD_STATUS = "awaitingUpload"; - public static final String OB_REVOKED_STATUS = "Revoked"; - public static final String OB_REJECTED_STATUS = "Rejected"; - public static final String OB_AUTHORIZED_STATUS = "Authorised"; - public static final String OB_AWAITING_AUTH_STATUS = "AwaitingAuthorisation"; - public static final String OB_AWAITING_UPLOAD_STATUS = "AwaitingUpload"; - - //VRP Constants - public static final String VRP_CONSENT_PATH = "domestic-vrp-consents"; - public static final String VRP_PAYMENT = "vrp-payment"; - public static final String PAID_AMOUNT = "paid-amount"; - public static final String LAST_PAYMENT_DATE = "last-payment-date"; - public static final String AUTH_TYPE_AUTHORIZATION = "authorization"; - public static final String CONTROL_PARAMETERS = "ControlParameters"; - public static final String MAXIMUM_INDIVIDUAL_AMOUNT = "MaximumIndividualAmount"; - public static final String MAXIMUM_INDIVIDUAL_AMOUNT_CURRENCY = "MaximumIndividualAmount.Amount.Currency"; - public static final String PERIODIC_LIMITS = "PeriodicLimits"; - public static final String PERIODIC_TYPES = "PeriodicTypes"; - public static final String PERIOD_AMOUNT_LIMIT = "Amount"; - public static final String PERIOD_LIMIT_CURRENCY = "PeriodicLimits.Currency"; - public static final String CYCLIC_EXPIRY_TIME = "cyclicExpiryTime"; - public static final String CYCLIC_REMAINING_AMOUNT = "cyclicRemainingAmount"; - - //vrp period alignment - public static final String PERIOD_ALIGNMENT = "PeriodAlignment"; - - // vrp periodic alignment types - public static final String CONSENT = "Consent"; - public static final String CALENDAR = "Calendar"; - - //vrp periodicLimits - public static final String PERIOD_TYPE = "PeriodType"; - - //vrp periodic types - public static final String DAY = "Day"; - public static final String WEEK = "Week"; - public static final String FORTNIGHT = "Fortnight"; - public static final String MONTH = "Month"; - public static final String HALF_YEAR = "Half-year"; - public static final String YEAR = "Year"; - public static final String VALID_TO_DATE_TIME = "ValidToDateTime"; - public static final String VALID_FROM_DATE_TIME = "ValidFromDateTime"; - public static final String VRP_RESPONSE_PROCESS_PATH = "vrp-response-process"; - - // vrp authorization flow constants - public static final String DOMESTIC_VRP = "Domestic VRP"; - public static final String CONTROL_PARAMETER_MAX_INDIVIDUAL_AMOUNT_TITLE = "Maximum amount per payment"; - public static final String CONTROL_PARAMETER_VALID_TO_DATE_TITLE = "Valid to date and time"; - public static final String CONTROL_PARAMETER_PERIOD_ALIGNMENT_TITLE = "Period Alignment"; - public static final String CONTROL_PARAMETER_PERIOD_TYPE_TITLE = "Period Type"; - public static final Object CONTROL_PARAMETER_AMOUNT_TITLE = "Maximum payment amount per "; - public static final String VRP_ACCOUNT = "vrpAccount"; - public static final Object CONTROL_PARAMETER_VALID_FROM_DATE_TITLE = "Valid from date and time"; - - // VRP submission flow - public static final String ACCOUNT_IDS = "accountIds"; - public static final String INSTRUCTION = "Instruction"; - public static final String REMITTANCE_INFO = "RemittanceInformation"; - public static final String REFERENCE = "Reference"; - public static final String UNSTRUCTURED = "Unstructured"; - public static final String CONTEXT_CODE = "PaymentContextCode"; - public static final String PAYMENT_TYPE = "PaymentType"; - public static final String VRP_PATH = "/domestic-vrps"; - public static final String PREVIOUS_PAID_AMOUNT = "prevPaidAmount"; - public static final String PREVIOUS_LAST_PAYMENT_DATE = "prevLastPaymentDate"; - - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentExtensionExporter.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentExtensionExporter.java deleted file mode 100644 index 5988299c..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentExtensionExporter.java +++ /dev/null @@ -1,81 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.common; - -import com.wso2.openbanking.accelerator.consent.extensions.admin.builder.ConsentAdminBuilder; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.builder.ConsentStepsBuilder; -import com.wso2.openbanking.accelerator.consent.extensions.manage.builder.ConsentManageBuilder; -import com.wso2.openbanking.accelerator.consent.extensions.validate.builder.ConsentValidateBuilder; - -/** - * Exporter service to facilitate access to loaded builder classes in data holder from other modules. - */ -public class ConsentExtensionExporter { - - private static volatile ConsentExtensionExporter consentExtExporter; - private static ConsentAdminBuilder consentAdminBuilder; - private static ConsentManageBuilder consentManageBuilder; - private static ConsentStepsBuilder consentStepsBuilder; - private static ConsentValidateBuilder consentValidateBuilder; - - public static ConsentExtensionExporter getInstance() { - if (consentExtExporter == null) { - synchronized (ConsentExtensionExporter.class) { - if (consentExtExporter == null) { - consentExtExporter = new ConsentExtensionExporter(); - } - } - } - - return consentExtExporter; - } - - public static ConsentValidateBuilder getConsentValidateBuilder() { - return consentValidateBuilder; - } - - public static void setConsentValidateBuilder(ConsentValidateBuilder consentValidateBuilder) { - ConsentExtensionExporter.consentValidateBuilder = consentValidateBuilder; - } - - public static ConsentStepsBuilder getConsentStepsBuilder() { - return consentStepsBuilder; - } - - public static void setConsentStepsBuilder(ConsentStepsBuilder consentStepsBuilder) { - ConsentExtensionExporter.consentStepsBuilder = consentStepsBuilder; - } - - public static ConsentManageBuilder getConsentManageBuilder() { - return consentManageBuilder; - } - - public static void setConsentManageBuilder(ConsentManageBuilder consentManageBuilder) { - ConsentExtensionExporter.consentManageBuilder = consentManageBuilder; - } - - public static ConsentAdminBuilder getConsentAdminBuilder() { - return consentAdminBuilder; - } - - public static void setConsentAdminBuilder(ConsentAdminBuilder consentAdminBuilder) { - ConsentExtensionExporter.consentAdminBuilder = consentAdminBuilder; - } - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentExtensionUtils.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentExtensionUtils.java deleted file mode 100644 index 141410ab..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentExtensionUtils.java +++ /dev/null @@ -1,417 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.common; - -import com.google.gson.Gson; -import com.google.gson.JsonObject; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentData; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentMappingResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.context.PrivilegedCarbonContext; -import org.wso2.carbon.identity.local.auth.api.core.ParameterResolverService; - -import java.io.Serializable; -import java.net.URI; -import java.net.URISyntaxException; -import java.nio.charset.StandardCharsets; -import java.text.DateFormat; -import java.text.SimpleDateFormat; -import java.time.LocalDate; -import java.time.LocalTime; -import java.time.OffsetDateTime; -import java.time.OffsetTime; -import java.time.ZoneId; -import java.time.format.DateTimeFormatter; -import java.util.ArrayList; -import java.util.Base64; -import java.util.Collections; -import java.util.Date; -import java.util.Enumeration; -import java.util.HashMap; -import java.util.Map; -import java.util.Set; - - -import javax.servlet.http.HttpServletRequest; - -/** - * Util class for consent extensions. - */ -public class ConsentExtensionUtils { - - private static final Log log = LogFactory.getLog(ConsentExtensionUtils.class); - private static Gson gson = new Gson(); - public static void setCommonDataToResponse(ConsentData consentData, JSONObject jsonObject) throws ConsentException { - - if (!jsonObject.containsKey(ConsentExtensionConstants.TYPE)) { - jsonObject.appendField(ConsentExtensionConstants.TYPE, consentData.getType()); - } - if (!jsonObject.containsKey(ConsentExtensionConstants.APPLICATION)) { - jsonObject.appendField(ConsentExtensionConstants.APPLICATION, consentData.getApplication()); - } - } - - public static JSONObject detailedConsentToJSON(DetailedConsentResource detailedConsentResource) { - JSONObject consentResource = new JSONObject(); - - consentResource.appendField("consentId", detailedConsentResource.getConsentID()); - consentResource.appendField("clientId", detailedConsentResource.getClientID()); - try { - consentResource.appendField("receipt", (new JSONParser(JSONParser.MODE_PERMISSIVE)). - parse(detailedConsentResource.getReceipt())); - } catch (ParseException e) { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Exception occurred while parsing" + - " receipt"); - } - consentResource.appendField("consentType", detailedConsentResource.getConsentType()); - consentResource.appendField("currentStatus", detailedConsentResource.getCurrentStatus()); - consentResource.appendField("consentFrequency", detailedConsentResource.getConsentFrequency()); - consentResource.appendField("validityPeriod", detailedConsentResource.getValidityPeriod()); - consentResource.appendField("createdTimestamp", detailedConsentResource.getCreatedTime()); - consentResource.appendField("updatedTimestamp", detailedConsentResource.getUpdatedTime()); - consentResource.appendField("recurringIndicator", detailedConsentResource.isRecurringIndicator()); - JSONObject attributes = new JSONObject(); - Map attMap = detailedConsentResource.getConsentAttributes(); - for (Map.Entry entry : attMap.entrySet()) { - attributes.appendField(entry.getKey(), entry.getValue()); - } - consentResource.appendField("consentAttributes", attributes); - JSONArray authorizationResources = new JSONArray(); - ArrayList authArray = detailedConsentResource.getAuthorizationResources(); - for (AuthorizationResource resource : authArray) { - JSONObject resourceJSON = new JSONObject(); - resourceJSON.appendField("authorizationId", resource.getAuthorizationID()); - resourceJSON.appendField("consentId", resource.getConsentID()); - resourceJSON.appendField("userId", resource.getUserID()); - resourceJSON.appendField("authorizationStatus", resource.getAuthorizationStatus()); - resourceJSON.appendField("authorizationType", resource.getAuthorizationType()); - resourceJSON.appendField("updatedTime", resource.getUpdatedTime()); - authorizationResources.add(resourceJSON); - } - consentResource.appendField("authorizationResources", authorizationResources); - JSONArray consentMappingResources = new JSONArray(); - ArrayList mappingArray = detailedConsentResource.getConsentMappingResources(); - for (ConsentMappingResource resource : mappingArray) { - JSONObject resourceJSON = new JSONObject(); - resourceJSON.appendField("mappingId", resource.getMappingID()); - resourceJSON.appendField("authorizationId", resource.getAuthorizationID()); - resourceJSON.appendField("accountId", resource.getAccountID()); - resourceJSON.appendField("permission", resource.getPermission()); - resourceJSON.appendField("mappingStatus", resource.getMappingStatus()); - consentMappingResources.add(resourceJSON); - } - consentResource.appendField("consentMappingResources", consentMappingResources); - return consentResource; - } - - public static JSONObject getRequestObjectPayload(String requestObject) { - try { - - // validate request object and get the payload - String requestObjectPayload; - String[] jwtTokenValues = requestObject.split("\\."); - if (jwtTokenValues.length == 3) { - requestObjectPayload = new String(Base64.getUrlDecoder().decode(jwtTokenValues[1]), - StandardCharsets.UTF_8); - } else { - throw new ConsentException(ResponseStatus.BAD_REQUEST, "request object is not signed JWT"); - } - Object payload = new JSONParser(JSONParser.MODE_PERMISSIVE).parse(requestObjectPayload); - if (!(payload instanceof JSONObject)) { - throw new ConsentException(ResponseStatus.BAD_REQUEST, "Payload is not a JSON object"); - } - return (JSONObject) payload; - - } catch (ParseException e) { - log.error("Exception occurred while getting consent data. Caused by : ", e); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, e.getMessage()); - } - } - - - /** - * Extract headers from a request object. - * - * @param request The request object - * @return Map of header key value pairs - */ - public static Map getHeaders(HttpServletRequest request) { - Map headers = new HashMap<>(); - Enumeration headerNames = request.getHeaderNames(); - while (headerNames.hasMoreElements()) { - String headerName = headerNames.nextElement(); - headers.put(headerName, request.getHeader(headerName)); - } - return headers; - } - - /** - * Get the sensitive data corresponding to the session data consent key. - * - * @param sessionDataKeyConsent The session data key corresponding to the data hidden from redirect URLs - * @return The hidden sensitive data as key-value pairs. - */ - public static Map getSensitiveDataWithConsentKey(String sessionDataKeyConsent) { - - return getSensitiveData(sessionDataKeyConsent); - } - - /** - * Get the sensitive data corresponding to the session data key or the session data consent key. - * - * @param key The session data key or session data consent key corresponding to the data hidden from redirect URLs - * @return The hidden sensitive data as key-value pairs. - */ - public static Map getSensitiveData(String key) { - - Map sensitiveDataSet = new HashMap<>(); - - Object serviceObj = PrivilegedCarbonContext.getThreadLocalCarbonContext() - .getOSGiService(ParameterResolverService.class, null); - if (serviceObj instanceof ParameterResolverService) { - ParameterResolverService resolverService = (ParameterResolverService) serviceObj; - - Set filter = Collections.emptySet(); - - sensitiveDataSet.putAll((resolverService) - .resolveParameters(ConsentExtensionConstants.CONSENT_KEY, key, filter)); - - if (sensitiveDataSet.isEmpty()) { - sensitiveDataSet.putAll((resolverService) - .resolveParameters(ConsentExtensionConstants.REQUEST_KEY, key, filter)); - } - - if (sensitiveDataSet.isEmpty()) { - log.error("No available data for key provided"); - sensitiveDataSet.put(ConsentExtensionConstants.IS_ERROR, "No available data for key provided"); - return sensitiveDataSet; - } - - sensitiveDataSet.put(ConsentExtensionConstants.IS_ERROR, "false"); - return sensitiveDataSet; - - } else { - log.error("Could not retrieve ParameterResolverService OSGi service"); - sensitiveDataSet.put(ConsentExtensionConstants.IS_ERROR, "Could not retrieve parameter service"); - return sensitiveDataSet; - } - } - - /** - * @param consentDetails json object of consent data - * @param sessionDataKey session data key - * @return ConsentData object - * @throws URISyntaxException if the URI is invalid - */ - public static ConsentData getConsentDataFromAttributes(JsonObject consentDetails, String sessionDataKey) - throws URISyntaxException { - - JsonObject sensitiveDataMap = consentDetails.get(ConsentExtensionConstants.SENSITIVE_DATA_MAP) - .getAsJsonObject(); - ConsentData consentData = new ConsentData(sessionDataKey, - sensitiveDataMap.get(ConsentExtensionConstants.LOGGED_IN_USER).getAsString(), - sensitiveDataMap.get(ConsentExtensionConstants.SP_QUERY_PARAMS).getAsString(), - consentDetails.get(ConsentExtensionConstants.SCOPES).getAsString(), - sensitiveDataMap.get(ConsentExtensionConstants.APPLICATION).getAsString(), - gson.fromJson(consentDetails.get(ConsentExtensionConstants.REQUEST_HEADERS), Map.class)); - consentData.setSensitiveDataMap(gson.fromJson(sensitiveDataMap, Map.class)); - URI redirectURI = new URI(consentDetails.get(ConsentExtensionConstants.REQUEST_URI).getAsString()); - consentData.setRedirectURI(redirectURI); - consentData.setUserId(consentDetails.get(ConsentExtensionConstants.USERID).getAsString()); - consentData.setConsentId(consentDetails.get(ConsentExtensionConstants.CONSENT_ID).getAsString()); - consentData.setClientId(consentDetails.get(ConsentExtensionConstants.CLIENT_ID).getAsString()); - consentData.setRegulatory(Boolean.parseBoolean(consentDetails.get(ConsentExtensionConstants.REGULATORY) - .getAsString())); - ConsentResource consentResource = gson.fromJson(consentDetails.get(ConsentExtensionConstants.CONSENT_RESOURCE), - ConsentResource.class); - consentData.setConsentResource(consentResource); - AuthorizationResource authorizationResource = - gson.fromJson(consentDetails.get(ConsentExtensionConstants.AUTH_RESOURCE), AuthorizationResource.class); - consentData.setAuthResource(authorizationResource); - consentData.setMetaDataMap(gson.fromJson(consentDetails.get(ConsentExtensionConstants.META_DATA), Map.class)); - consentData.setType(consentDetails.get(ConsentExtensionConstants.TYPE).getAsString()); - return consentData; - } - - /** - * Validates whether Cutoffdatetime is enabled, if the request is arriving past the cut off time and if it - * should be rejected by policy. - * - * @return if the request should be rejected, or not. - */ - public static boolean shouldInitiationRequestBeRejected() { - - return Boolean.parseBoolean((String) OpenBankingConfigParser.getInstance().getConfiguration().get( - ConsentExtensionConstants.CUTOFF_DATE_ENABLED)) && isCutOffTimeElapsed() - && ConsentExtensionConstants.REJECT.equals(OpenBankingConfigParser.getInstance().getConfiguration() - .get(ConsentExtensionConstants.CUTOFF_DATE_POLICY)); - } - - /** - * Validates whether the CutOffTime for the day has elapsed. - * - * @return has elapsed - */ - public static boolean isCutOffTimeElapsed() { - - OffsetTime dailyCutOffTime = OffsetTime.parse((String) OpenBankingConfigParser.getInstance().getConfiguration() - .get(ConsentExtensionConstants.DAILY_CUTOFF)); - OffsetTime currentTime = LocalTime.now().atOffset(dailyCutOffTime.getOffset()); - if (log.isDebugEnabled()) { - log.debug("Request received at" + currentTime + " daily cut off time set to " + dailyCutOffTime); - } - return currentTime.isAfter(dailyCutOffTime); - } - /** - * validate whether Cutoffdatetime is enabled, if the request is arriving past the cut off time - * and if it was accepted by policy.git a. - * - * @return if request is accepted and cut off date time has passed, or not - */ - - public static boolean isRequestAcceptedPastElapsedTime() { - - if (Boolean.parseBoolean((String) OpenBankingConfigParser.getInstance().getConfiguration() - .get(OpenBankingConstants.CUTOFF_DATE_ENABLED)) && - isCutOffTimeElapsed() && ConsentExtensionConstants.ACCEPT - .equals(OpenBankingConfigParser.getInstance().getConfiguration() - .get(OpenBankingConstants.CUTOFF_DATE_POLICY))) { - - log.debug("Request Accepted but CutOffDateTime has elapsed."); - return true; - } - return false; - } - /** - * Returns the DateTime by adding given number of days and the with the given Time. - * - * @param daysToAdd Number of days to add - * @param time Time to add - * @return DateTime value for the day - */ - public static String constructDateTime(long daysToAdd, String time) { - - String configuredZoneId = (String) OpenBankingConfigParser.getInstance().getConfiguration() - .get(OpenBankingConstants.ZONE_ID); - String dateValue = LocalDate.now(ZoneId.of(configuredZoneId)).plusDays(daysToAdd) + "T" + - (OffsetTime.parse(time)); - - OffsetDateTime offSetDateVal = OffsetDateTime.parse(dateValue); - DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ssXXX"); - return dateTimeFormatter.format(offSetDateVal); - } - - /** - * Validates whether Cutoffdatetime is enabled, if the request is arriving past the cut off date and if it - * should be rejected by policy. - * - * @param timeStamp Initiation timestamp - * @return if the request should be rejected, or not. - */ - public static boolean shouldSubmissionRequestBeRejected(String timeStamp) { - - String isCutOffDateEnabled = (String) OpenBankingConfigParser.getInstance().getConfiguration() - .get(OpenBankingConstants.CUTOFF_DATE_ENABLED); - String cutOffDatePolicy = (String) OpenBankingConfigParser.getInstance().getConfiguration() - .get(OpenBankingConstants.CUTOFF_DATE_POLICY); - - if (Boolean.parseBoolean(isCutOffDateEnabled) && ConsentExtensionConstants.REJECT.equals(cutOffDatePolicy)) { - if (isCutOffTimeElapsed()) { - log.debug("Request Rejected as CutOffTime has elapsed."); - return true; - } - - if (hasCutOffDateElapsed(timeStamp)) { - log.debug("Request Rejected as CutOffDate has elapsed."); - return true; - } - } - return false; - } - /** - * Validates whether the cutOffDate and the initiation date are the same. - * - * @return if the request should be rejected, or not. - */ - private static boolean hasCutOffDateElapsed(String initiationTimestamp) { - - OffsetDateTime initiationDateTime = OffsetDateTime.parse(initiationTimestamp); - OffsetDateTime currentDateTime = OffsetDateTime.parse(getCurrentCutOffDateTime()); - return initiationDateTime.getMonth() != currentDateTime.getMonth() || - initiationDateTime.getDayOfMonth() != currentDateTime.getDayOfMonth(); - } - /** - * Returns the CutOffDateTime from the CutOffTime. - * - * @return CutOffDateTime value for the day - */ - public static String getCurrentCutOffDateTime() { - - return LocalDate.now() + "T" + (OffsetTime.parse((String) OpenBankingConfigParser.getInstance() - .getConfiguration() - .get(OpenBankingConstants.DAILY_CUTOFF))); - } - /** - * Convert long date values to ISO 8601 format. - * @param dateValue Date value in long - * @return ISO 8601 formatted date - */ - public static String convertToISO8601(long dateValue) { - - DateFormat simple = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssXXX"); - Date simpleDateVal = new Date(dateValue * 1000); - return simple.format(simpleDateVal); - } - public static ConsentCoreServiceImpl getConsentService() { - return new ConsentCoreServiceImpl(); - } - - /** - * Get the mapping status. - * - * @param defaultStatus Default status returned from the accelerator - * @return Mapping UK status - */ - public static String getConsentStatus(String defaultStatus) { - - switch (defaultStatus) { - case ConsentExtensionConstants.AUTHORIZED_STATUS: - return ConsentExtensionConstants.OB_AUTHORIZED_STATUS; - case ConsentExtensionConstants.REVOKED_STATUS: - return ConsentExtensionConstants.OB_REVOKED_STATUS; - case ConsentExtensionConstants.REJECTED_STATUS: - return ConsentExtensionConstants.OB_REJECTED_STATUS; - case ConsentExtensionConstants.AWAITING_UPLOAD_STATUS: - return ConsentExtensionConstants.OB_AWAITING_UPLOAD_STATUS; - default: - return ConsentExtensionConstants.OB_AWAITING_AUTH_STATUS; - } - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentServiceUtil.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentServiceUtil.java deleted file mode 100644 index 0713e4ff..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ConsentServiceUtil.java +++ /dev/null @@ -1,34 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.common; - -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; - -/** - * Common Util Class for accessing external services. - */ -public class ConsentServiceUtil { - - @Generated(message = "Excluded from coverage since this is used for testing purposes") - public static ConsentCoreServiceImpl getConsentService() { - return new ConsentCoreServiceImpl(); - } -} - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ResponseStatus.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ResponseStatus.java deleted file mode 100644 index 396599a4..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/ResponseStatus.java +++ /dev/null @@ -1,258 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.common; - -/** - * Enum of the supported response status in accelerator. - * HTTP/1.1 documentation - ... - */ -public enum ResponseStatus { - - /** - * 200 OK, see ... - */ - OK(200, "OK"), - /** - * 201 Created, see ... - */ - CREATED(201, "Created"), - /** - * 202 Accepted, see ... - */ - ACCEPTED(202, "Accepted"), - /** - * 204 No Content, see ... - */ - NO_CONTENT(204, "No Content"), - /** - * 205 Reset Content, see ... - * - * @since 2.0 - */ - RESET_CONTENT(205, "Reset Content"), - /** - * 206 Reset Content, see ... - * - * @since 2.0 - */ - PARTIAL_CONTENT(206, "Partial Content"), - /** - * 301 Moved Permanently, see ... - */ - MOVED_PERMANENTLY(301, "Moved Permanently"), - /** - * 302 Found, see ... - * - * @since 2.0 - */ - FOUND(302, "Found"), - /** - * 303 See Other, see ... - */ - SEE_OTHER(303, "See Other"), - /** - * 304 Not Modified, see ... - */ - NOT_MODIFIED(304, "Not Modified"), - /** - * 305 Use Proxy, see link - "..." of - * HTTP/1.1 documentation. - * - * @since 2.0 - */ - USE_PROXY(305, "Use Proxy"), - /** - * 307 Temporary Redirect, see ... - */ - TEMPORARY_REDIRECT(307, "Temporary Redirect"), - /** - * 400 Bad Request, see ... - */ - BAD_REQUEST(400, "Bad Request"), - /** - * 401 Unauthorized, see ... - */ - UNAUTHORIZED(401, "Unauthorized"), - /** - * 402 Payment Required, see ... - * - * @since 2.0 - */ - PAYMENT_REQUIRED(402, "Payment Required"), - /** - * 403 Forbidden, see ... - */ - FORBIDDEN(403, "Forbidden"), - /** - * 404 Not Found, see ... - */ - NOT_FOUND(404, "Not Found"), - /** - * 405 Method Not Allowed, see ... - * - * @since 2.0 - */ - METHOD_NOT_ALLOWED(405, "Method Not Allowed"), - /** - * 406 Not Acceptable, see ... - */ - NOT_ACCEPTABLE(406, "Not Acceptable"), - /** - * 409 Conflict, see ... - */ - CONFLICT(409, "Conflict"), - /** - * 410 Gone, see ... - */ - GONE(410, "Gone"), - /** - * 411 Length Required, see ... - * - * @since 2.0 - */ - LENGTH_REQUIRED(411, "Length Required"), - /** - * 412 Precondition Failed, see ... - */ - PRECONDITION_FAILED(412, "Precondition Failed"), - /** - * 413 Request Entity Too Large, - * see ... - * - * @since 2.0 - */ - REQUEST_ENTITY_TOO_LARGE(413, "Request Entity Too Large"), - /** - * 414 Request-URI Too Long, see ... - * - * @since 2.0 - */ - REQUEST_URI_TOO_LONG(414, "Request-URI Too Long"), - /** - * 415 Unsupported Media Type, - * see ... - */ - UNSUPPORTED_MEDIA_TYPE(415, "Unsupported Media Type"), - /** - * 416 Requested Range Not Satisfiable, - * see ... - * - * @since 2.0 - */ - REQUESTED_RANGE_NOT_SATISFIABLE(416, "Requested Range Not Satisfiable"), - /** - * 417 Expectation Failed, - * see ... - * - * @since 2.0 - */ - EXPECTATION_FAILED(417, "Expectation Failed"), - /** - * 422 Unprocessable Entity. - */ - UNPROCESSABLE_ENTITY(422, "Unprocessable Entity"), - /** - * 429 Too Many Requests. - */ - TOO_MANY_REQUESTS(429, "Too Many Requests"), - /** - * 500 Internal Server Error, see ... - */ - INTERNAL_SERVER_ERROR(500, "Internal Server Error"), - /** - * 501 Not Implemented, see ... - * - * @since 2.0 - */ - NOT_IMPLEMENTED(501, "Not Implemented"), - /** - * 502 Bad Gateway, see ..." - * - * @since 2.0 - */ - BAD_GATEWAY(502, "Bad Gateway"), - /** - * 503 Service Unavailable, see ... - */ - SERVICE_UNAVAILABLE(503, "Service Unavailable"), - /** - * 504 Gateway Timeout, see ... - * - * @since 2.0 - */ - GATEWAY_TIMEOUT(504, "Gateway Timeout"), - /** - * 505 HTTP Version Not Supported, - * see ... - * - * @since 2.0 - */ - HTTP_VERSION_NOT_SUPPORTED(505, "HTTP Version Not Supported"); - - private final int code; - private final String reason; - - ResponseStatus(final int statusCode, final String reasonPhrase) { - this.code = statusCode; - this.reason = reasonPhrase; - } - - /** - * Get the associated status code. - * - * @return the status code. - */ - public int getStatusCode() { - return code; - } - - /** - * Get the reason phrase. - * - * @return the reason phrase. - */ - public String getReasonPhrase() { - return toString(); - } - - /** - * Get the reason phrase. - * - * @return the reason phrase. - */ - @Override - public String toString() { - return reason; - } - - /** - * Convert a numerical status code into the corresponding Status. - * - * @param statusCode the numerical status code. - * @return the matching Status or null is no matching Status is defined. - */ - public static ResponseStatus fromStatusCode(final int statusCode) { - for (ResponseStatus s : ResponseStatus.values()) { - if (s.code == statusCode) { - return s; - } - } - return null; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/factory/AcceleratorConsentExtensionFactory.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/factory/AcceleratorConsentExtensionFactory.java deleted file mode 100644 index 0c15cb9d..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/factory/AcceleratorConsentExtensionFactory.java +++ /dev/null @@ -1,59 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.common.factory; - -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.manage.impl.AccountConsentManageRequestHandler; -import com.wso2.openbanking.accelerator.consent.extensions.manage.impl.CofConsentRequestHandler; -import com.wso2.openbanking.accelerator.consent.extensions.manage.impl.ConsentManageRequestHandler; -import com.wso2.openbanking.accelerator.consent.extensions.manage.impl.PaymentConsentRequestHandler; -import com.wso2.openbanking.accelerator.consent.extensions.manage.impl.VRPConsentRequestHandler; - -/** - * Factory class to get the class based in request type. - */ -public class AcceleratorConsentExtensionFactory { - /** - * Method to get the Consent Manage Request Validator. - * - * @param requestPath Request path of the request - * @return ConsentManageRequestValidator - */ - public static ConsentManageRequestHandler getConsentManageRequestValidator(String requestPath) { - - ConsentManageRequestHandler consentManageRequestHandler = null; - - switch (requestPath) { - case ConsentExtensionConstants.ACCOUNT_CONSENT_GET_PATH: - consentManageRequestHandler = new AccountConsentManageRequestHandler(); - break; - case ConsentExtensionConstants.COF_CONSENT_PATH: - consentManageRequestHandler = new CofConsentRequestHandler(); - break; - case ConsentExtensionConstants.PAYMENT_CONSENT_PATH: - consentManageRequestHandler = new PaymentConsentRequestHandler(); - break; - case ConsentExtensionConstants.VRP_CONSENT_PATH: - consentManageRequestHandler = new VRPConsentRequestHandler(); - break; - default: - return null; - } - return consentManageRequestHandler; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyConstants.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyConstants.java deleted file mode 100644 index e9866c25..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyConstants.java +++ /dev/null @@ -1,42 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.common.idempotency; - -/** - * Constants related to idempotency operations. - */ -public class IdempotencyConstants { - - public static final String CONTENT_TYPE_TAG = "content-type"; - public static final String X_IDEMPOTENCY_KEY = "x-idempotency-key"; - public static final String IDEMPOTENCY_KEY_NAME = "IdempotencyKey"; - public static final String ISO_FORMAT = "yyyy-MM-dd'T'HH:mm:ssXXX"; - public static final String EMPTY_OBJECT = "{}"; - public static final String ERROR_PAYLOAD_NOT_SIMILAR = "Payloads are not similar. Hence this is not a valid" + - " idempotent request"; - public static final String ERROR_AFTER_ALLOWED_TIME = "Request received after the allowed time., Hence this is" + - " not a valid idempotent request"; - public static final String ERROR_MISMATCHING_CLIENT_ID = "Client ID sent in the request does not match with the" + - " client ID in the retrieved consent. Hence this is not a valid idempotent request"; - public static final String ERROR_NO_CONSENT_DETAILS = "No consent details found for the consent ID %s, Hence this" + - " is not a valid idempotent request"; - public static final String JSON_COMPARING_ERROR = "Error occurred while comparing JSON payloads"; - public static final String CONSENT_RETRIEVAL_ERROR = "Error while retrieving detailed consent data"; - public static final String SAME_CONSENT_ID_ERROR = "Cannot use different unique identifier for the same" + - " consent ID when the request does not contain a payload."; -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidationException.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidationException.java deleted file mode 100644 index 821287ca..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidationException.java +++ /dev/null @@ -1,35 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.common.idempotency; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; - -/** - * Used for handling exceptions in Idempotency Validation. - */ -public class IdempotencyValidationException extends OpenBankingException { - - public IdempotencyValidationException(String message) { - super(message); - } - - public IdempotencyValidationException(String message, Throwable e) { - super(message, e); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidationResult.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidationResult.java deleted file mode 100644 index a67f6694..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidationResult.java +++ /dev/null @@ -1,82 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.common.idempotency; - -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; - -/** - * Class to hold idempotency validation result. - */ -public class IdempotencyValidationResult { - - private boolean isIdempotent; - private boolean isValid; - private DetailedConsentResource consent; - private String consentId; - - public IdempotencyValidationResult() { - } - - public IdempotencyValidationResult(boolean isIdempotent, boolean isValid, DetailedConsentResource consent, - String consentId) { - this.isIdempotent = isIdempotent; - this.isValid = isValid; - this.consent = consent; - this.consentId = consentId; - } - - public IdempotencyValidationResult(boolean isIdempotent, boolean isValid) { - this.isIdempotent = isIdempotent; - this.isValid = isValid; - this.consent = null; - this.consentId = null; - } - - public boolean isIdempotent() { - return isIdempotent; - } - - public void setIsIdempotent(boolean isIdempotent) { - this.isIdempotent = isIdempotent; - } - - public boolean isValid() { - return isValid; - } - - public void setValid(boolean isValid) { - this.isValid = isValid; - } - - public DetailedConsentResource getConsent() { - return consent; - } - - public void setConsent(DetailedConsentResource consent) { - this.consent = consent; - } - - public String getConsentId() { - return consentId; - } - - public void setConsentID(String consentId) { - this.consentId = consentId; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidationUtils.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidationUtils.java deleted file mode 100644 index 19639519..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidationUtils.java +++ /dev/null @@ -1,131 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.common.idempotency; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import com.wso2.openbanking.accelerator.consent.mgt.service.ConsentCoreService; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.text.DateFormat; -import java.text.SimpleDateFormat; -import java.time.Duration; -import java.time.OffsetDateTime; -import java.util.ArrayList; -import java.util.Date; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -/** - * Class to hold idempotency validation utils. - */ -public class IdempotencyValidationUtils { - - private static final Log log = LogFactory.getLog(IdempotencyValidationUtils.class); - private static final ConsentCoreService consentCoreService = ConsentExtensionsDataHolder.getInstance() - .getConsentCoreService(); - - /** - * Method to retrieve the consent ids that have the idempotency key name and value as attribute. - * - * @param idempotencyKeyName Idempotency Key Name - * @param idempotencyKeyValue Idempotency Key Value - * @return List of consent ids if available, else an empty list will be returned - */ - static List getConsentIdsFromIdempotencyKey(String idempotencyKeyName, - String idempotencyKeyValue) { - try { - return consentCoreService.getConsentIdByConsentAttributeNameAndValue( - idempotencyKeyName, idempotencyKeyValue); - } catch (ConsentManagementException e) { - log.debug("No consent ids found for the idempotency key value"); - return new ArrayList<>(); - } - } - - /** - * Method to retrieve the consent ids and idempotency key value using the idempotency key. - * - * @param idempotencyKeyName Idempotency Key Name - * @return Map of consent ids and idempotency key vallue if available, else an empty map will be returned - */ - static Map getAttributesFromIdempotencyKey(String idempotencyKeyName) { - try { - return consentCoreService.getConsentAttributesByName(idempotencyKeyName); - } catch (ConsentManagementException e) { - log.debug("No consent ids found for the idempotency key value"); - return new HashMap<>(); - } - } - - /** - * Method to compare the client ID sent in the request and client id retrieved from the database. - * - * @param requestClientID Client ID sent in the request - * @param dbClientID client ID retrieved from the database - * @return true if the client ID sent in the request and client id retrieved from the database are equal - */ - static boolean isClientIDEqual(String requestClientID, String dbClientID) { - if (requestClientID == null) { - return false; - } - return requestClientID.equals(dbClientID); - } - - /** - * Method to check whether difference between two dates is less than the configured time. - * - * @param createdTime Created Time of the request - * @return true if the request is received within allowed time - */ - static boolean isRequestReceivedWithinAllowedTime(long createdTime) { - - if (createdTime == 0L) { - log.debug("Created time is of the previous request is not correctly set. Hence returning false"); - return false; - } - String allowedTimeDuration = OpenBankingConfigParser.getInstance().getIdempotencyAllowedTime(); - if (StringUtils.isNotBlank(allowedTimeDuration)) { - OffsetDateTime createdDate = OffsetDateTime.parse(toISO8601DateTime(createdTime)); - OffsetDateTime currDate = OffsetDateTime.now(createdDate.getOffset()); - - long diffInMinutes = Duration.between(createdDate, currDate).toMinutes(); - return diffInMinutes <= Long.parseLong(allowedTimeDuration); - } else { - log.error("Idempotency allowed duration is not configured in the system. Hence returning false"); - return false; - } - } - - /** - * Convert long date values to ISO 8601 format. ISO 8601 format - "yyyy-MM-dd'T'HH:mm:ssXXX" - * @param epochDate Date value in epoch format - * @return ISO 8601 formatted date - */ - private static String toISO8601DateTime(long epochDate) { - - DateFormat simple = new SimpleDateFormat(IdempotencyConstants.ISO_FORMAT); - Date simpleDateVal = new Date(epochDate * 1000); - return simple.format(simpleDateVal); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidator.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidator.java deleted file mode 100644 index 0e6ed237..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidator.java +++ /dev/null @@ -1,301 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.common.idempotency; - -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.ObjectMapper; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import com.wso2.openbanking.accelerator.consent.extensions.manage.model.ConsentManageData; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.ConsentCoreService; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.io.IOException; -import java.util.List; -import java.util.Map; - -/** - * Class to handle idempotency related operations. - */ -public class IdempotencyValidator { - - private static final Log log = LogFactory.getLog(IdempotencyValidator.class); - private static final ConsentCoreService consentCoreService = ConsentExtensionsDataHolder.getInstance() - .getConsentCoreService(); - - /** - * Method to check whether the request is idempotent. - * This method will first check whether idempotency validation is enabled. After that it will validate whether - * required parameters for validation is present. - * For validation, need to check whether the idempotency key values is present as a consent attribute, if present - * the consent will be retrieved. Finally following conditions will be validated. - * - Whether the client id sent in the request and client id retrieved from the database are equal - * - Whether the difference between two dates is less than the configured time - * - Whether payloads are equal - * - * @param consentManageData Consent Manage Data - * @return IdempotencyValidationResult - * @throws IdempotencyValidationException If an error occurs while validating idempotency - */ - public IdempotencyValidationResult validateIdempotency(ConsentManageData consentManageData) - throws IdempotencyValidationException { - - if (!OpenBankingConfigParser.getInstance().isIdempotencyValidationEnabled()) { - return new IdempotencyValidationResult(false, false); - } - // If request is empty then cannot proceed with idempotency validation - if (consentManageData.getPayload() == null) { - log.error("Request payload is empty. Hence cannot proceed with idempotency validation"); - return new IdempotencyValidationResult(false, false); - } - // If client id is empty then cannot proceed with idempotency validation - if (StringUtils.isBlank(consentManageData.getClientId())) { - log.error("Client ID is empty. Hence cannot proceed with idempotency validation"); - return new IdempotencyValidationResult(false, false); - } - String idempotencyKeyValue = consentManageData.getHeaders().get(getIdempotencyHeaderName()); - // If idempotency key value is empty then cannot proceed with idempotency validation - if (StringUtils.isBlank(idempotencyKeyValue)) { - log.error("Idempotency Key Valueis empty. Hence cannot proceed with idempotency validation"); - return new IdempotencyValidationResult(false, false); - } - try { - String idempotencyKeyName = getIdempotencyAttributeName(consentManageData.getRequestPath()); - if (!IdempotencyConstants.EMPTY_OBJECT.equals(consentManageData.getPayload().toString())) { - // Retrieve consent ids that have the idempotency key name and value as attribute - List consentIds = IdempotencyValidationUtils - .getConsentIdsFromIdempotencyKey(idempotencyKeyName, idempotencyKeyValue); - // Check whether the consent id list is not empty. If idempotency key exists in the database then - // the consent Id list will be not empty. - if (!consentIds.isEmpty()) { - if (log.isDebugEnabled()) { - log.debug(String.format("Idempotency Key %s exists in the database. Hence this is an" + - " idempotent request", idempotencyKeyValue)); - } - for (String consentId : consentIds) { - DetailedConsentResource consentResource = consentCoreService.getDetailedConsent(consentId); - if (consentResource != null) { - return validateIdempotencyConditions(consentManageData, consentResource); - } else { - String errorMsg = String.format(IdempotencyConstants.ERROR_NO_CONSENT_DETAILS, consentId); - log.error(errorMsg); - throw new IdempotencyValidationException(errorMsg); - } - } - } - } else { - return validateIdempotencyWithoutPayload(consentManageData, idempotencyKeyName, idempotencyKeyValue); - } - } catch (IOException e) { - log.error(IdempotencyConstants.JSON_COMPARING_ERROR, e); - throw new IdempotencyValidationException(IdempotencyConstants.JSON_COMPARING_ERROR); - } catch (ConsentManagementException e) { - log.error(IdempotencyConstants.CONSENT_RETRIEVAL_ERROR, e); - return new IdempotencyValidationResult(true, false); - } - return new IdempotencyValidationResult(false, false); - } - - /** - * Method to check whether the idempotency conditions are met for requests without payload. - * This method will validate the following conditions. - * - Whether the idempotency key value is different for the same consent id - * - Whether the client id sent in the request and client id retrieved from the database are equal - * - Whether the difference between two dates is less than the configured time - * - Whether payloads are equal - * - * @param consentManageData Consent Manage Data - * @param idempotencyKeyName Idempotency Key Name - * @param idempotencyKeyValue Idempotency Key value - * @return IdempotencyValidationResult - */ - private IdempotencyValidationResult validateIdempotencyWithoutPayload(ConsentManageData consentManageData, - String idempotencyKeyName, - String idempotencyKeyValue) - throws IdempotencyValidationException, IOException, ConsentManagementException { - - // Retrieve consent ids and idempotency key values that have the idempotency key name - Map attributes = IdempotencyValidationUtils.getAttributesFromIdempotencyKey(idempotencyKeyName); - // Check whether the attributes map is not empty. If idempotency key exists in the database then - // the consent Id list will be not empty. - if (!attributes.isEmpty()) { - if (log.isDebugEnabled()) { - log.debug(String.format("Idempotency Key %s exists in the database. Hence this is an" + - " idempotent request", idempotencyKeyValue)); - } - for (Map.Entry entry : attributes.entrySet()) { - // If the idempotency key value is different for the same consent id then it is not a valid idempotent - if (consentManageData.getRequestPath().contains(entry.getKey()) && - !idempotencyKeyValue.equals(entry.getValue())) { - throw new IdempotencyValidationException(IdempotencyConstants.SAME_CONSENT_ID_ERROR); - } - DetailedConsentResource consentRequest = consentCoreService.getDetailedConsent(entry.getKey()); - if (consentRequest != null) { - return validateIdempotencyConditions(consentManageData, consentRequest); - } else { - String errorMsg = String.format(IdempotencyConstants.ERROR_NO_CONSENT_DETAILS, entry.getKey()); - log.error(errorMsg); - throw new IdempotencyValidationException(errorMsg); - } - } - - } - return new IdempotencyValidationResult(false, false); - } - - /** - * Method to check whether the idempotency conditions are met. - * This method will validate the following conditions. - * - Whether the client id sent in the request and client id retrieved from the database are equal - * - Whether the difference between two dates is less than the configured time - * - Whether payloads are equal - * - * @param consentManageData Consent Manage Data - * @param consentResource Detailed Consent Resource - * @return IdempotencyValidationResult - */ - private IdempotencyValidationResult validateIdempotencyConditions(ConsentManageData consentManageData, - DetailedConsentResource consentResource) - throws IdempotencyValidationException, IOException { - // Compare the client ID sent in the request and client id retrieved from the database - // to validate whether the request is received from the same client - if (IdempotencyValidationUtils.isClientIDEqual(consentResource.getClientID(), - consentManageData.getClientId())) { - // Check whether difference between two dates is less than the configured time - if (IdempotencyValidationUtils.isRequestReceivedWithinAllowedTime(getCreatedTimeOfPreviousRequest( - consentManageData.getRequestPath(), consentResource.getConsentID()))) { - // Compare whether JSON payloads are equal - if (isPayloadSimilar(consentManageData, getPayloadOfPreviousRequest( - consentManageData.getRequestPath(), consentResource.getConsentID()))) { - log.debug("Payloads are similar and request received within allowed" + - " time. Hence this is a valid idempotent request"); - return new IdempotencyValidationResult(true, true, - consentResource, consentResource.getConsentID()); - } else { - log.error(IdempotencyConstants.ERROR_PAYLOAD_NOT_SIMILAR); - throw new IdempotencyValidationException(IdempotencyConstants - .ERROR_PAYLOAD_NOT_SIMILAR); - } - } else { - log.error(IdempotencyConstants.ERROR_AFTER_ALLOWED_TIME); - throw new IdempotencyValidationException(IdempotencyConstants - .ERROR_AFTER_ALLOWED_TIME); - } - } else { - log.error(IdempotencyConstants.ERROR_MISMATCHING_CLIENT_ID); - throw new IdempotencyValidationException(IdempotencyConstants.ERROR_MISMATCHING_CLIENT_ID); - } - } - - /** - * Method to get the Idempotency Attribute Name store in consent Attributes. - * - * @param resourcePath Resource Path - * @return idempotency Attribute Name. - */ - public String getIdempotencyAttributeName(String resourcePath) { - return IdempotencyConstants.IDEMPOTENCY_KEY_NAME; - } - - /** - * Method to get the Idempotency Header Name according to the request. - * - * @return idempotency Header Name. - */ - public String getIdempotencyHeaderName() { - return IdempotencyConstants.X_IDEMPOTENCY_KEY; - } - - /** - * Method to get created time from the Detailed Consent Resource. - * - * @param resourcePath Resource Path - * @param consentId ConsentId - * @return Created Time. - */ - public long getCreatedTimeOfPreviousRequest(String resourcePath, String consentId) { - DetailedConsentResource consentRequest = null; - try { - consentRequest = consentCoreService.getDetailedConsent(consentId); - } catch (ConsentManagementException e) { - log.error(IdempotencyConstants.CONSENT_RETRIEVAL_ERROR, e); - return 0L; - } - if (consentRequest == null) { - return 0L; - } - return consentRequest.getCreatedTime(); - } - - /** - * Method to get payload from previous request. - * - * @param resourcePath Resource Path - * @param consentId ConsentId - * @return Map containing the payload. - */ - public String getPayloadOfPreviousRequest(String resourcePath, String consentId) { - DetailedConsentResource consentRequest = null; - try { - consentRequest = consentCoreService.getDetailedConsent(consentId); - } catch (ConsentManagementException e) { - log.error(IdempotencyConstants.CONSENT_RETRIEVAL_ERROR, e); - return null; - } - if (consentRequest == null) { - return null; - } - return consentRequest.getReceipt(); - } - - /** - * Method to compare whether payloads are equal. - * - * @param consentManageData Consent Manage Data Object - * @param consentReceipt Payload received from database - * @return Whether payloads are equal - */ - public boolean isPayloadSimilar(ConsentManageData consentManageData, String consentReceipt) { - - if (consentManageData.getPayload() == null || consentReceipt == null) { - return false; - } - - JsonNode expectedNode = null; - JsonNode actualNode = null; - try { - ObjectMapper mapper = new ObjectMapper(); - expectedNode = mapper.readTree(consentManageData.getPayload().toString()); - actualNode = mapper.readTree(consentReceipt); - if (log.isDebugEnabled()) { - log.debug(String.format("Expected payload for idempotent request is: %s. But actual payload " + - "received is %s", expectedNode.toString(), actualNode.toString())); - } - } catch (JsonProcessingException e) { - log.error(IdempotencyConstants.JSON_COMPARING_ERROR, e); - return false; - } - return expectedNode.equals(actualNode); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/event/executors/ConsentAmendmentHistoryEventExecutor.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/event/executors/ConsentAmendmentHistoryEventExecutor.java deleted file mode 100644 index 178fc3d4..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/event/executors/ConsentAmendmentHistoryEventExecutor.java +++ /dev/null @@ -1,92 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.event.executors; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.event.executor.OBEventExecutor; -import com.wso2.openbanking.accelerator.common.event.executor.model.OBEvent; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentHistoryResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.ConsentCoreService; -import com.wso2.openbanking.accelerator.consent.mgt.service.constants.ConsentCoreServiceConstants; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.Map; - -/** - * Open banking event executor for Consent Amendment History Asynchronous Persistence. - */ -public class ConsentAmendmentHistoryEventExecutor implements OBEventExecutor { - - private static final Log log = LogFactory.getLog(ConsentAmendmentHistoryEventExecutor.class); - - @Override - public void processEvent(OBEvent obEvent) { - - String eventType = obEvent.getEventType(); - if (OpenBankingConfigParser.getInstance().isConsentAmendmentHistoryEnabled() && - (ConsentCoreServiceConstants.CONSENT_AMENDED_STATUS.equalsIgnoreCase(eventType) || - OpenBankingConstants.DEFAULT_STATUS_FOR_REVOKED_CONSENTS.equalsIgnoreCase(eventType))) { - - ConsentCoreService consentCoreService = ConsentExtensionsDataHolder.getInstance().getConsentCoreService(); - try { - Map eventData = obEvent.getEventData(); - String consentID = eventData.get("ConsentId").toString(); - - Map consentDataMap = (Map) eventData.get("ConsentDataMap"); - DetailedConsentResource detailedCurrentConsent = (DetailedConsentResource) - consentDataMap.get(ConsentCoreServiceConstants.DETAILED_CONSENT_RESOURCE); - DetailedConsentResource detailedHistoryConsent = (DetailedConsentResource) - consentDataMap.get(ConsentCoreServiceConstants.CONSENT_AMENDMENT_HISTORY_RESOURCE); - long amendedTimestamp = (long) - consentDataMap.get(ConsentCoreServiceConstants.CONSENT_AMENDMENT_TIME) / 1000; - - String amendmentReason; - if (ConsentCoreServiceConstants.CONSENT_AMENDED_STATUS.equalsIgnoreCase(eventType)) { - amendmentReason = ConsentCoreServiceConstants.AMENDMENT_REASON_CONSENT_AMENDMENT_FLOW; - } else { - amendmentReason = ConsentCoreServiceConstants.AMENDMENT_REASON_CONSENT_REVOCATION; - } - ConsentHistoryResource consentHistoryResource = new ConsentHistoryResource(); - consentHistoryResource.setDetailedConsentResource(detailedHistoryConsent); - consentHistoryResource.setReason(amendmentReason); - consentHistoryResource.setTimestamp(amendedTimestamp); - - boolean result = consentCoreService.storeConsentAmendmentHistory(consentID, consentHistoryResource, - detailedCurrentConsent); - - if (result) { - if (log.isDebugEnabled()) { - log.debug(String.format("Consent Amendment History of consentID: %s persisted successfully.", - consentID)); - } - } else { - log.error(String.format("Failed to persist Consent Amendment History of consentID : %s. " + - consentID)); - } - } catch (ConsentManagementException e) { - log.error("An error occurred while persisting consent amendment history data.", e); - } - } - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/event/executors/VRPEventExecutor.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/event/executors/VRPEventExecutor.java deleted file mode 100644 index 126ad48a..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/event/executors/VRPEventExecutor.java +++ /dev/null @@ -1,119 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.event.executors; - -import com.wso2.openbanking.accelerator.common.event.executor.OBEventExecutor; -import com.wso2.openbanking.accelerator.common.event.executor.model.OBEvent; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.manage.model.PeriodicLimit; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; - -import java.math.BigDecimal; -import java.util.ArrayList; -import java.util.List; - -/** - * This class is responsible for executing Variable Recurring Payments (VRP) events. - * It implements the OBEventExecutor interface and overrides its methods to provide - * specific implementations for VRP events. - */ -public class VRPEventExecutor implements OBEventExecutor { - - public static List validateInstructedAmountWithControlParameters(BigDecimal instructedAmount, - JSONObject controlParameters) { - - /** - * Validates the instructed amount with control parameters and returns a list of PeriodicLimit objects. - * If the instructed amount is greater than the maximum individual amount or the cyclic remaining amount, - * an empty list is returned. If the JSON parsing fails, an empty list is also returned. - * - * @param instructedAmount The instructed amount to be validated - * @param controlParameters The control parameters to be used for validation - * @return A list of PeriodicLimit objects - */ - List periodicLimitsList = new ArrayList<>(); - - BigDecimal maxIndividualAmount = BigDecimal.valueOf(Double.parseDouble(controlParameters. - getAsString(ConsentExtensionConstants.MAXIMUM_INDIVIDUAL_AMOUNT))); - - if (instructedAmount.compareTo(maxIndividualAmount) > 0) { - return periodicLimitsList; - } - - JSONParser parser = new JSONParser(JSONParser.MODE_JSON_SIMPLE); - JSONArray periodicLimits; - - try { - periodicLimits = (JSONArray) parser.parse(controlParameters. - getAsString(ConsentExtensionConstants.PERIODIC_LIMITS)); - } catch (ParseException e) { - // Log the error or handle it as needed - return periodicLimitsList; - } - - long currentMoment = System.currentTimeMillis() / 1000; - - for (Object obj : periodicLimits) { - JSONObject limit = (JSONObject) obj; - BigDecimal amount = BigDecimal. - valueOf(Double.parseDouble(limit.getAsString(ConsentExtensionConstants.AMOUNT))); - long cyclicExpiryTime = Long.parseLong(limit.getAsString(ConsentExtensionConstants.CYCLIC_EXPIRY_TIME)); - BigDecimal cyclicRemainingAmount = BigDecimal. - valueOf(Double.parseDouble(limit.getAsString(ConsentExtensionConstants.CYCLIC_REMAINING_AMOUNT))); - - String periodType = limit.getAsString(ConsentExtensionConstants.PERIOD_TYPE); - String periodAlignment = limit.getAsString(ConsentExtensionConstants.PERIOD_ALIGNMENT); - - PeriodicLimit periodicLimit = new PeriodicLimit(periodType, amount, periodAlignment); - - if (currentMoment <= cyclicExpiryTime) { - if (instructedAmount.compareTo(cyclicRemainingAmount) > 0) { - return periodicLimitsList; - } else { - cyclicRemainingAmount = cyclicRemainingAmount.subtract(instructedAmount); - } - } else { - while (currentMoment > periodicLimit.getCyclicExpiryTime()) { - periodicLimit.setCyclicExpiryTime(); - } - cyclicRemainingAmount = amount; - if (instructedAmount.compareTo(cyclicRemainingAmount) > 0) { - return periodicLimitsList; - } else { - cyclicRemainingAmount = cyclicRemainingAmount.subtract(instructedAmount); - } - } - periodicLimitsList.add(periodicLimit); - } - - return periodicLimitsList; - } - - /** - * Processes the given OBEvent. This method is part of the OBEventExecutor interface and needs to be implemented. - * - * @param obEvent The OBEvent to be processed - */ - @Override - public void processEvent(OBEvent obEvent) { - - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/internal/ConsentExtensionsComponent.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/internal/ConsentExtensionsComponent.java deleted file mode 100644 index 0c9e6e95..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/internal/ConsentExtensionsComponent.java +++ /dev/null @@ -1,108 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.internal; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.consent.extensions.ciba.authenticator.CIBAPushAuthenticator; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionExporter; -import com.wso2.openbanking.accelerator.consent.extensions.util.PeriodicalConsentJobActivator; -import com.wso2.openbanking.accelerator.consent.mgt.service.ConsentCoreService; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.osgi.service.component.ComponentContext; -import org.osgi.service.component.annotations.Activate; -import org.osgi.service.component.annotations.Component; -import org.osgi.service.component.annotations.Deactivate; -import org.osgi.service.component.annotations.Reference; -import org.osgi.service.component.annotations.ReferenceCardinality; -import org.osgi.service.component.annotations.ReferencePolicy; -import org.wso2.carbon.identity.application.authentication.framework.ApplicationAuthenticator; - -/** - * The Component class for activating consent extensions osgi service. - */ -@Component( - name = "com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsComponent", - immediate = true) -public class ConsentExtensionsComponent { - private static Log log = LogFactory.getLog(ConsentExtensionsComponent.class); - - @Activate - protected void activate(ComponentContext context) { - - context.getBundleContext().registerService(ConsentExtensionExporter.class.getName(), - ConsentExtensionExporter.getInstance(), null); - if (log.isDebugEnabled()) { - log.debug("Consent extensions are registered successfully."); - } - new PeriodicalConsentJobActivator().activate(); - if (log.isDebugEnabled()) { - log.debug("Periodical Consent Status Updater Started"); - } - CIBAPushAuthenticator authenticator = new CIBAPushAuthenticator(); - context.getBundleContext().registerService(ApplicationAuthenticator.class.getName(), - authenticator, null); - if (log.isDebugEnabled()) { - log.debug("CIBA Push authenticator bundle is activated"); - } - - } - - @Reference( - service = OpenBankingConfigurationService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetConfigService" - ) - public void setConfigService(OpenBankingConfigurationService openBankingConfigurationService) { - - ConsentExtensionsDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - } - - public void unsetConfigService(OpenBankingConfigurationService openBankingConfigurationService) { - - ConsentExtensionsDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - - } - - @Deactivate - protected void deactivate(ComponentContext context) { - - log.debug("Open banking Consent Extensions component is deactivated"); - } - - @Reference( - service = com.wso2.openbanking.accelerator.consent.mgt.service.ConsentCoreService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetConsentCoreService" - ) - public void setConsentCoreService(ConsentCoreService consentCoreService) { - - log.debug("Setting the Consent Core Service"); - ConsentExtensionsDataHolder.getInstance().setConsentCoreService(consentCoreService); - } - - public void unsetConsentCoreService(ConsentCoreService consentCoreService) { - - log.debug("UnSetting the Consent Core Service"); - ConsentExtensionsDataHolder.getInstance().setConsentCoreService(null); - - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/internal/ConsentExtensionsDataHolder.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/internal/ConsentExtensionsDataHolder.java deleted file mode 100644 index 553d2d0d..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/internal/ConsentExtensionsDataHolder.java +++ /dev/null @@ -1,135 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.internal; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.consent.extensions.admin.builder.ConsentAdminBuilder; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.builder.ConsentStepsBuilder; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionExporter; -import com.wso2.openbanking.accelerator.consent.extensions.manage.builder.ConsentManageBuilder; -import com.wso2.openbanking.accelerator.consent.extensions.validate.builder.ConsentValidateBuilder; -import com.wso2.openbanking.accelerator.consent.mgt.service.ConsentCoreService; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -/** - * Contains Data holder class for consent extensions. - */ -public class ConsentExtensionsDataHolder { - - private static Log log = LogFactory.getLog(ConsentExtensionsDataHolder.class); - private static volatile ConsentExtensionsDataHolder instance; - private OpenBankingConfigurationService openBankingConfigurationService; - private ConsentCoreService consentCoreService; - - private ConsentStepsBuilder consentStepsBuilder; - private ConsentAdminBuilder consentAdminBuilder; - private ConsentManageBuilder consentManageBuilder; - private ConsentValidateBuilder consentValidateBuilder; - - // Prevent instantiation - private ConsentExtensionsDataHolder() {} - - /** - * Return a singleton instance of the data holder. - * - * @return A singleton instance of the data holder - */ - public static synchronized ConsentExtensionsDataHolder getInstance() { - if (instance == null) { - synchronized (ConsentExtensionsDataHolder.class) { - if (instance == null) { - instance = new ConsentExtensionsDataHolder(); - } - } - } - return instance; - } - - public OpenBankingConfigurationService getOpenBankingConfigurationService() { - - return openBankingConfigurationService; - } - - public void setOpenBankingConfigurationService( - OpenBankingConfigurationService openBankingConfigurationService) { - - this.openBankingConfigurationService = openBankingConfigurationService; - - ConsentStepsBuilder consentStepsBuilder = new ConsentStepsBuilder(); - consentStepsBuilder.build(); - this.setConsentStepsBuilder(consentStepsBuilder); - ConsentExtensionExporter.setConsentStepsBuilder(consentStepsBuilder); - - ConsentAdminBuilder consentAdminBuilder = new ConsentAdminBuilder(); - consentAdminBuilder.build(); - this.setConsentAdminBuilder(consentAdminBuilder); - ConsentExtensionExporter.setConsentAdminBuilder(consentAdminBuilder); - - ConsentManageBuilder consentManageBuilder = new ConsentManageBuilder(); - consentManageBuilder.build(); - this.setConsentManageBuilder(consentManageBuilder); - ConsentExtensionExporter.setConsentManageBuilder(consentManageBuilder); - - ConsentValidateBuilder consentValidateBuilder = new ConsentValidateBuilder(); - consentValidateBuilder.build(); - this.setConsentValidateBuilder(consentValidateBuilder); - ConsentExtensionExporter.setConsentValidateBuilder(consentValidateBuilder); - } - - public ConsentStepsBuilder getConsentStepsBuilder() { - return consentStepsBuilder; - } - - public void setConsentStepsBuilder(ConsentStepsBuilder consentStepsBuilder) { - this.consentStepsBuilder = consentStepsBuilder; - } - - public ConsentAdminBuilder getConsentAdminBuilder() { - return consentAdminBuilder; - } - - public void setConsentAdminBuilder(ConsentAdminBuilder consentAdminBuilder) { - this.consentAdminBuilder = consentAdminBuilder; - } - - public ConsentManageBuilder getConsentManageBuilder() { - return consentManageBuilder; - } - - public void setConsentManageBuilder(ConsentManageBuilder consentManageBuilder) { - this.consentManageBuilder = consentManageBuilder; - } - - public ConsentValidateBuilder getConsentValidateBuilder() { - return consentValidateBuilder; - } - - public void setConsentValidateBuilder(ConsentValidateBuilder consentValidateBuilder) { - this.consentValidateBuilder = consentValidateBuilder; - } - - public ConsentCoreService getConsentCoreService() { - return consentCoreService; - } - - public void setConsentCoreService(ConsentCoreService consentCoreService) { - this.consentCoreService = consentCoreService; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/builder/ConsentManageBuilder.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/builder/ConsentManageBuilder.java deleted file mode 100644 index 9e2295a5..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/builder/ConsentManageBuilder.java +++ /dev/null @@ -1,48 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.manage.builder; - -import com.wso2.openbanking.accelerator.common.util.OpenBankingUtils; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import com.wso2.openbanking.accelerator.consent.extensions.manage.model.ConsentManageHandler; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -/** - * Builder class for consent manage handler. - */ -public class ConsentManageBuilder { - - private static final Log log = LogFactory.getLog(ConsentManageBuilder.class); - private ConsentManageHandler consentManageHandler = null; - private static String manageHandlerConfigPath = "Consent.ManageHandler"; - - public void build() { - - String handlerConfig = (String) ConsentExtensionsDataHolder.getInstance().getOpenBankingConfigurationService(). - getConfigurations().get(manageHandlerConfigPath); - consentManageHandler = (ConsentManageHandler) OpenBankingUtils.getClassInstanceFromFQN(handlerConfig); - - log.debug("Manage handler loaded successfully"); - } - - public ConsentManageHandler getConsentManageHandler() { - return consentManageHandler; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/AccountConsentManageRequestHandler.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/AccountConsentManageRequestHandler.java deleted file mode 100644 index 83d7d0f2..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/AccountConsentManageRequestHandler.java +++ /dev/null @@ -1,343 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.manage.impl; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import com.wso2.openbanking.accelerator.consent.extensions.manage.model.ConsentManageData; -import com.wso2.openbanking.accelerator.consent.extensions.util.ConsentManageUtil; -import com.wso2.openbanking.accelerator.consent.mgt.dao.constants.ConsentMgtDAOConstants; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.constants.ConsentCoreServiceConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.handler.EventNotificationPersistenceServiceHandler; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.time.Instant; -import java.time.LocalDateTime; -import java.time.OffsetDateTime; -import java.time.ZoneOffset; -import java.time.ZonedDateTime; -import java.time.format.DateTimeFormatter; -import java.time.format.DateTimeParseException; -import java.util.Arrays; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -/** - * Consent Manage request handler class for Account Request Validation. - */ -public class AccountConsentManageRequestHandler implements ConsentManageRequestHandler { - - private static final Log log = LogFactory.getLog(AccountConsentManageRequestHandler.class); - private static final String ACCOUNT_CONSENT_GET_PATH = "account-access-consents"; - private static final String UUID_REGEX = - "[0-9a-fA-F]{8}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{12}"; - private static final String REVOKED_STATUS = "revoked"; - private static final List validPermissions = Arrays.asList( - "ReadAccountsDetail", - "ReadTransactionsDetail", - "ReadBalances"); - private static final String ACCOUNT_CONSENT_CREATE_PATH = "account-access-consents"; - private static final String CREATED_STATUS = "created"; - private static final String AUTH_TYPE_AUTHORIZATION = "authorization"; - - - /** - * Method to handle Account Consent Manage Post Request. - * - * @param consentManageData Object containing request details - */ - @Override - public void handleConsentManagePost(ConsentManageData consentManageData) { - - //Get the request payload from the ConsentManageData - Object request = consentManageData.getPayload(); - - if (request == null || request instanceof JSONArray) { - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.NOT_JSON_OBJECT_ERROR); - } - - JSONObject requestObject; - if (consentManageData.getRequestPath().equals(ACCOUNT_CONSENT_CREATE_PATH)) { - //Validate Account Initiation request - requestObject = (JSONObject) request; - if (!validateInitiation(requestObject)) { - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.PAYLOAD_INVALID); - } - } else { - throw new ConsentException(ResponseStatus.BAD_REQUEST, "Request path invalid"); - } - - ConsentResource requestedConsent = new ConsentResource(consentManageData.getClientId(), - requestObject.toJSONString(), ConsentExtensionConstants.ACCOUNTS, - ConsentExtensionConstants.AWAITING_AUTH_STATUS); - - //Set request object to the response - JSONObject response = requestObject; - - DetailedConsentResource createdConsent; - - appendConsentExpirationTimestampAttribute(requestedConsent); - - //create consent - try { - createdConsent = ConsentExtensionsDataHolder.getInstance().getConsentCoreService() - .createAuthorizableConsent(requestedConsent, null, - CREATED_STATUS, AUTH_TYPE_AUTHORIZATION, true); - consentManageData.setResponsePayload(ConsentManageUtil.getInitiationResponse(response, createdConsent, - consentManageData, ConsentExtensionConstants.ACCOUNTS)); - consentManageData.setResponseStatus(ResponseStatus.CREATED); - } catch (ConsentManagementException e) { - log.error(e.getMessage()); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, e.getMessage()); - } - } - - @Override - public void handleConsentManageGet(ConsentManageData consentManageData) { - - if (consentManageData.getRequestPath().startsWith(ACCOUNT_CONSENT_GET_PATH)) { - String consentId = consentManageData.getRequestPath().split("/")[1]; - if (ConsentManageUtil.isConsentIdValid(consentId)) { - try { - ConsentResource consent = ConsentExtensionsDataHolder.getInstance().getConsentCoreService() - .getConsent(consentId, false); - if (consent == null) { - log.error("No valid consent found for given information"); - throw new ConsentException(ResponseStatus.BAD_REQUEST, - ErrorConstants.NO_CONSENT_FOR_CLIENT_ERROR); - } - if (!consent.getClientID().equals(consentManageData.getClientId())) { - //Throwing same error as null scenario since client will not be able to identify if consent - // exists if consent does not belong to them - throw new ConsentException(ResponseStatus.BAD_REQUEST, - ErrorConstants.NO_CONSENT_FOR_CLIENT_ERROR); - } - JSONObject receiptJSON = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE). - parse(consent.getReceipt()); - JSONObject data = (JSONObject) receiptJSON.get("Data"); - data.appendField("ConsentId", consent.getConsentID()); - data.appendField("CreationDateTime", convertEpochDateTime(consent.getCreatedTime())); - data.appendField("StatusUpdateDateTime", convertEpochDateTime(consent.getUpdatedTime())); - receiptJSON.put("Data", data); - consentManageData.setResponsePayload(receiptJSON); - consentManageData.setResponseStatus(ResponseStatus.OK); - } catch (ConsentManagementException | ParseException e) { - log.error(e.getMessage()); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, e.getMessage()); - } - } else { - throw new ConsentException(ResponseStatus.BAD_REQUEST, "Consent ID invalid"); - } - } else { - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.PATH_INVALID); - } - - } - - @Override - public void handleConsentManageDelete(ConsentManageData consentManageData) { - - if (consentManageData.getRequestPath().startsWith(ConsentExtensionConstants.ACCOUNT_CONSENT_DELETE_PATH)) { - String consentId = consentManageData.getRequestPath().split( - ConsentExtensionConstants.ACCOUNT_CONSENT_DELETE_PATH)[1]; - if (ConsentManageUtil.isConsentIdValid(consentId)) { - try { - ConsentResource consentResource = ConsentExtensionsDataHolder.getInstance().getConsentCoreService() - .getConsent(consentId, false); - - if (!consentResource.getClientID().equals(consentManageData.getClientId())) { - //Throwing this error in a generic manner since client will not be able to identify if consent - // exists if consent does not belong to them - throw new ConsentException(ResponseStatus.BAD_REQUEST, - ErrorConstants.NO_CONSENT_FOR_CLIENT_ERROR); - } - - if (REVOKED_STATUS.equals(consentResource.getCurrentStatus())) { - throw new ConsentException(ResponseStatus.BAD_REQUEST, - "Consent already in revoked state"); - } - - boolean success = ConsentExtensionsDataHolder.getInstance().getConsentCoreService() - .revokeConsentWithReason(consentId, REVOKED_STATUS, - ConsentCoreServiceConstants.CONSENT_REVOKE_REASON); - if (!success) { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - "Token revocation unsuccessful"); - } - consentManageData.setResponseStatus(ResponseStatus.NO_CONTENT); - if (success) { - // persist a new notification to the DB - // This is a sample event notification persisting. This can be modified in the Toolkit level - if (OpenBankingConfigParser.getInstance().isRealtimeEventNotificationEnabled()) { - JSONObject notificationInfo = new JSONObject(); - notificationInfo.put("consentID", consentId); - notificationInfo.put("status", "Consent Revocation"); - notificationInfo.put("timeStamp", System.currentTimeMillis()); - EventNotificationPersistenceServiceHandler.getInstance().persistRevokeEvent( - consentResource.getClientID(), consentId, - "Consent Revocation", notificationInfo); - } - } - } catch (ConsentManagementException e) { - log.error(e.getMessage()); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, e.getMessage()); - } - } else { - throw new ConsentException(ResponseStatus.BAD_REQUEST, "Consent ID invalid"); - } - } else { - throw new ConsentException(ResponseStatus.BAD_REQUEST, "Request path invalid"); - } - } - - private boolean validateInitiation(JSONObject initiation) { - - if (!initiation.containsKey("Data") || !(initiation.get("Data") instanceof JSONObject)) { - return false; - } - - JSONObject data = (JSONObject) initiation.get("Data"); - - if (!data.containsKey("Permissions") || !(data.get("Permissions") instanceof JSONArray)) { - return false; - } - - JSONArray permissions = (JSONArray) data.get("Permissions"); - for (Object permission : permissions) { - if (!(permission instanceof String)) { - return false; - } - String permissionString = (String) permission; - if (!validPermissions.contains(permissionString)) { - return false; - } - } - - if (!data.containsKey("ExpirationDateTime") || !(data.get("ExpirationDateTime") instanceof String)) { - return false; - } - - if (!isConsentExpirationTimeValid(data.getAsString("ExpirationDateTime"))) { - return false; - } - - if (!data.containsKey("TransactionFromDateTime") || !(data.get("TransactionFromDateTime") instanceof String)) { - return false; - } - - if (!data.containsKey("TransactionToDateTime") || !(data.get("TransactionToDateTime") instanceof String)) { - return false; - } - - if (!isTransactionFromToTimeValid(data.getAsString("TransactionFromDateTime"), - data.getAsString("TransactionToDateTime"))) { - return false; - } - - return true; - } - - private static boolean isConsentExpirationTimeValid(String expDateVal) { - - if (expDateVal == null) { - return true; - } - try { - OffsetDateTime expDate = OffsetDateTime.parse(expDateVal); - OffsetDateTime currDate = OffsetDateTime.now(expDate.getOffset()); - - if (log.isDebugEnabled()) { - log.debug("Provided expiry date is: " + expDate + " current date is: " + currDate); - } - - return expDate.compareTo(currDate) > 0; - } catch (DateTimeParseException e) { - return false; - } - } - - private static boolean isTransactionFromToTimeValid(String fromDateVal, String toDateVal) { - - if (fromDateVal == null || toDateVal == null) { - return true; - } - try { - OffsetDateTime fromDate = OffsetDateTime.parse(fromDateVal); - OffsetDateTime toDate = OffsetDateTime.parse(toDateVal); - - // From date is earlier than To date - return (fromDate.compareTo(toDate) <= 0); - } catch (DateTimeParseException e) { - return false; - } - } - - /** - * Method to append the consent expiration time (UTC) as a consent attribute. - * @param requestedConsent Consent Resource - */ - public static void appendConsentExpirationTimestampAttribute(ConsentResource requestedConsent) { - - Map consentAttributes = requestedConsent.getConsentAttributes(); - JSONObject receiptJSON = null; - try { - receiptJSON = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE). - parse(requestedConsent.getReceipt()); - JSONObject data = null; - if (receiptJSON.containsKey(ConsentExtensionConstants.DATA)) { - data = (JSONObject) receiptJSON.get(ConsentExtensionConstants.DATA); - } - if (data != null && data.containsKey(ConsentMgtDAOConstants.CONSENT_EXPIRY_TIME_ATTRIBUTE)) { - String expireTime = data.get(ConsentMgtDAOConstants.CONSENT_EXPIRY_TIME_ATTRIBUTE).toString(); - ZonedDateTime zonedDateTime = ZonedDateTime.parse(expireTime); - // Retrieve the UTC timestamp in long from expiry time. - long expireTimestamp = Instant.from(zonedDateTime).getEpochSecond(); - if (consentAttributes == null) { - consentAttributes = new HashMap(); - } - consentAttributes.put(ConsentMgtDAOConstants.CONSENT_EXPIRY_TIME_ATTRIBUTE, - Long.toString(expireTimestamp)); - requestedConsent.setConsentAttributes(consentAttributes); - } - } catch (ParseException e) { - log.error("Invalid consent receipt received to append expiration time. : " - + requestedConsent.getConsentID()); - } - } - - private static String convertEpochDateTime(long epochTime) { - - int nanoOfSecond = 0; - ZoneOffset offset = ZoneOffset.UTC; - LocalDateTime ldt = LocalDateTime.ofEpochSecond(epochTime, nanoOfSecond, offset); - return DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ss'Z'").format(ldt); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/CofConsentRequestHandler.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/CofConsentRequestHandler.java deleted file mode 100644 index 7599b2d5..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/CofConsentRequestHandler.java +++ /dev/null @@ -1,127 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.manage.impl; - -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentServiceUtil; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import com.wso2.openbanking.accelerator.consent.extensions.manage.model.ConsentManageData; -import com.wso2.openbanking.accelerator.consent.extensions.manage.validator.CofConsentRequestValidator; -import com.wso2.openbanking.accelerator.consent.extensions.util.ConsentManageUtil; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - - -/** - * Consent Manage request handler class for Confirmation of Funds Request Validation. - */ -public class CofConsentRequestHandler implements ConsentManageRequestHandler { - - private static final Log log = LogFactory.getLog(CofConsentRequestHandler.class); - private static final String CREATED_STATUS = "created"; - private static final String AUTH_TYPE_AUTHORIZATION = "authorization"; - @Override - public void handleConsentManagePost(ConsentManageData consentManageData) { - - //Get the request payload from the ConsentManageData - Object request = consentManageData.getPayload(); - if (!(request instanceof JSONObject)) { - log.error(ErrorConstants.PAYLOAD_FORMAT_ERROR); - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.INVALID_REQ_PAYLOAD); - } - - JSONObject requestObject = (JSONObject) request; - - //Validate COF Initiation request - JSONObject validationResponse = CofConsentRequestValidator.validateCOFInitiation(requestObject); - if (validationResponse.containsKey(ConsentExtensionConstants.IS_VALID) && - !((boolean) validationResponse.get(ConsentExtensionConstants.IS_VALID))) { - log.error(ErrorConstants.PAYLOAD_INVALID); - throw new ConsentException((ResponseStatus) validationResponse.get(ConsentExtensionConstants.HTTP_CODE), - (JSONObject) validationResponse.get(ConsentExtensionConstants.ERRORS)); - } - ConsentResource requestedConsent = new ConsentResource(consentManageData.getClientId(), - requestObject.toJSONString(), ConsentExtensionConstants.FUNDSCONFIRMATIONS, - ConsentExtensionConstants.AWAITING_AUTH_STATUS); - - //Set request object to the response - JSONObject response = requestObject; - - DetailedConsentResource createdConsent; - try { - createdConsent = ConsentExtensionsDataHolder.getInstance().getConsentCoreService() - .createAuthorizableConsent(requestedConsent, null, - CREATED_STATUS, AUTH_TYPE_AUTHORIZATION, true); - consentManageData.setResponsePayload(ConsentManageUtil.getInitiationResponse(response, createdConsent, - consentManageData, ConsentExtensionConstants.FUNDSCONFIRMATIONS)); - consentManageData.setResponseStatus(ResponseStatus.CREATED); - } catch (ConsentManagementException e) { - log.error(e.getMessage()); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, e.getMessage()); - } - - } - - @Override - public void handleConsentManageGet(ConsentManageData consentManageData) { - - String consentId = consentManageData.getRequestPath().split("/")[1]; - if (ConsentManageUtil.isConsentIdValid(consentId)) { - try { - ConsentResource consent = ConsentServiceUtil.getConsentService().getConsent(consentId, - false); - if (consent == null) { - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.RESOURCE_CONSENT_MISMATCH); - } - // Check whether the client id is matching - if (!consent.getClientID().equals(consentManageData.getClientId())) { - //Throwing same error as null scenario since client will not be able to identify if consent - // exists if consent does not belong to them - throw new ConsentException(ResponseStatus.BAD_REQUEST, - ErrorConstants.NO_CONSENT_FOR_CLIENT_ERROR); - } - JSONObject receiptJSON = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE). - parse(consent.getReceipt()); - consentManageData.setResponsePayload(ConsentManageUtil - .getInitiationRetrievalResponse(receiptJSON, consent, consentManageData, - ConsentExtensionConstants.FUNDSCONFIRMATIONS)); - consentManageData.setResponseStatus(ResponseStatus.OK); - } catch (ConsentManagementException | ParseException e) { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - ErrorConstants.ACC_INITIATION_RETRIEVAL_ERROR); - } - } else { - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.INVALID_CONSENT_ID); - } - } - - @Override - public void handleConsentManageDelete(ConsentManageData consentManageData) { - ConsentManageUtil.handleConsentManageDelete(consentManageData); - - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/ConsentManageRequestHandler.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/ConsentManageRequestHandler.java deleted file mode 100644 index c09419dc..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/ConsentManageRequestHandler.java +++ /dev/null @@ -1,47 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.manage.impl; - -import com.wso2.openbanking.accelerator.consent.extensions.manage.model.ConsentManageData; - -/** - * Abstract Consent Manage Request Handler class. - */ -public interface ConsentManageRequestHandler { - - /** - * Method to handle Account Consent Manage Post Request. - * - * @param consentManageData Object containing request details - */ - void handleConsentManagePost(ConsentManageData consentManageData); - - /** - * Method to handle Consent Manage GET Request. - * - * @param consentManageData Object containing request details - */ - void handleConsentManageGet(ConsentManageData consentManageData); - - /** - * Method to handle Account Consent Manage Post Request. - * - * @param consentManageData Object containing request details - */ - void handleConsentManageDelete(ConsentManageData consentManageData); -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/DefaultConsentManageHandler.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/DefaultConsentManageHandler.java deleted file mode 100644 index d3f26464..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/DefaultConsentManageHandler.java +++ /dev/null @@ -1,162 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.manage.impl; - -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.consent.extensions.common.factory.AcceleratorConsentExtensionFactory; -import com.wso2.openbanking.accelerator.consent.extensions.manage.model.ConsentManageData; -import com.wso2.openbanking.accelerator.consent.extensions.manage.model.ConsentManageHandler; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.UUID; - -/** - * Consent manage handler default implementation. - */ - -public class DefaultConsentManageHandler implements ConsentManageHandler { - - private static final Log log = LogFactory.getLog(DefaultConsentManageHandler.class); - private static final String INTERACTION_ID_HEADER = "x-fapi-interaction-id"; - private ConsentManageRequestHandler consentManageRequestHandler; - - @Override - public void handleGet(ConsentManageData consentManageData) throws ConsentException { - - if (consentManageData.getHeaders().containsKey(INTERACTION_ID_HEADER)) { - consentManageData.setResponseHeader(INTERACTION_ID_HEADER, - consentManageData.getHeaders().get(INTERACTION_ID_HEADER)); - } else { - consentManageData.setResponseHeader(INTERACTION_ID_HEADER, - UUID.randomUUID().toString()); - } - - //Check whether client ID exists - if (StringUtils.isEmpty(consentManageData.getClientId())) { - log.error(ErrorConstants.MSG_MISSING_CLIENT_ID); - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.MSG_MISSING_CLIENT_ID); - } - - String[] requestPathArray; - - if (consentManageData.getRequestPath() == null) { - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.RESOURCE_NOT_FOUND); - } else { - requestPathArray = consentManageData.getRequestPath().split("/"); - } - - if (requestPathArray != null && !requestPathArray[0].isEmpty()) { - //Get consent manage request validator according to the request path - consentManageRequestHandler = AcceleratorConsentExtensionFactory - .getConsentManageRequestValidator(requestPathArray[0]); - - if (consentManageRequestHandler != null) { - consentManageRequestHandler.handleConsentManageGet(consentManageData); - } else { - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.PATH_INVALID); - } - } else { - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.PATH_INVALID); - } - } - - @Override - public void handlePost(ConsentManageData consentManageData) throws ConsentException { - - //set consent id aa response header - String consentID = UUID.randomUUID().toString(); - if (consentManageData.getHeaders().containsKey(INTERACTION_ID_HEADER)) { - consentManageData.setResponseHeader(INTERACTION_ID_HEADER, - consentManageData.getHeaders().get(INTERACTION_ID_HEADER)); - } else { - consentManageData.setResponseHeader(INTERACTION_ID_HEADER, consentID); - } - - //Get consent manage request validator according to the request path - consentManageRequestHandler = AcceleratorConsentExtensionFactory - .getConsentManageRequestValidator(consentManageData.getRequestPath()); - - if (consentManageRequestHandler != null) { - consentManageRequestHandler.handleConsentManagePost(consentManageData); - } else { - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.PATH_INVALID); - } - } - - @Override - public void handleDelete(ConsentManageData consentManageData) throws ConsentException { - - if (consentManageData.getHeaders().containsKey(ConsentExtensionConstants.INTERACTION_ID_HEADER)) { - consentManageData.setResponseHeader(ConsentExtensionConstants.INTERACTION_ID_HEADER, - consentManageData.getHeaders().get(ConsentExtensionConstants.INTERACTION_ID_HEADER)); - } else { - consentManageData.setResponseHeader(ConsentExtensionConstants.INTERACTION_ID_HEADER, - UUID.randomUUID().toString()); - } - - String[] requestPathArray; - if (consentManageData.getRequestPath() == null) { - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.RESOURCE_NOT_FOUND); - } else { - requestPathArray = consentManageData.getRequestPath().split("/"); - } - - if (requestPathArray != null && !requestPathArray[0].isEmpty()) { - - //Get consent manage request validator according to the request path - consentManageRequestHandler = AcceleratorConsentExtensionFactory - .getConsentManageRequestValidator(requestPathArray[0]); - - if (consentManageRequestHandler != null) { - consentManageRequestHandler.handleConsentManageDelete(consentManageData); - } else { - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.PATH_INVALID); - } - } - } - - @Override - public void handlePut(ConsentManageData consentManageData) throws ConsentException { - - throw new ConsentException(ResponseStatus.METHOD_NOT_ALLOWED, "Method PUT is not supported"); - } - - @Override - public void handlePatch(ConsentManageData consentManageData) throws ConsentException { - - throw new ConsentException(ResponseStatus.METHOD_NOT_ALLOWED, "Method PATCH is not supported"); - } - - @Override - public void handleFileUploadPost(ConsentManageData consentManageData) throws ConsentException { - - throw new ConsentException(ResponseStatus.METHOD_NOT_ALLOWED, "File upload is not supported"); - } - - @Override - public void handleFileGet(ConsentManageData consentManageData) throws ConsentException { - - throw new ConsentException(ResponseStatus.METHOD_NOT_ALLOWED, "File retrieval is not supported"); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/PaymentConsentRequestHandler.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/PaymentConsentRequestHandler.java deleted file mode 100644 index b8040a50..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/PaymentConsentRequestHandler.java +++ /dev/null @@ -1,167 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.manage.impl; - -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionUtils; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentServiceUtil; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import com.wso2.openbanking.accelerator.consent.extensions.manage.model.ConsentManageData; -import com.wso2.openbanking.accelerator.consent.extensions.manage.validator.PaymentsConsentRequestValidator; -import com.wso2.openbanking.accelerator.consent.extensions.util.ConsentManageUtil; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.HashMap; -import java.util.Map; - - -/** - * Consent Manage request handler class for Payment Request Validation. - */ -public class PaymentConsentRequestHandler implements ConsentManageRequestHandler { - private static final Log log = LogFactory.getLog(PaymentConsentRequestHandler.class); - private static final String CREATED_STATUS = "created"; - private static final String AUTH_TYPE_AUTHORIZATION = "authorization"; - - @Override - public void handleConsentManagePost(ConsentManageData consentManageData) { - - try { - //Validate cutoff datetime - if (ConsentExtensionUtils.shouldInitiationRequestBeRejected()) { - log.error(ErrorConstants.MSG_ELAPSED_CUT_OFF_DATE_TIME); - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.PAYMENT_INITIATION_HANDLE_ERROR); - } - - //Get the request payload from the ConsentManageData - Object request = consentManageData.getPayload(); - if (!(request instanceof JSONObject)) { - log.error(ErrorConstants.PAYLOAD_FORMAT_ERROR); - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.INVALID_REQ_PAYLOAD); - } - - JSONObject requestObject = (JSONObject) request; - - //Set request object to the response - JSONObject response = requestObject; - - //Check Idempotency key exists - if (StringUtils.isEmpty(consentManageData.getHeaders() - .get(ConsentExtensionConstants.X_IDEMPOTENCY_KEY))) { - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.IDEMPOTENCY_KEY_NOT_FOUND); - } - - //Handle payment initiation flows - handlePaymentPost(consentManageData, requestObject, response); - - } catch (ConsentManagementException e) { - log.error(e.getMessage()); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - ErrorConstants.PAYMENT_INITIATION_HANDLE_ERROR); - } - - } - - @Override - public void handleConsentManageGet(ConsentManageData consentManageData) { - String consentId = consentManageData.getRequestPath().split("/")[1]; - if (ConsentManageUtil.isConsentIdValid(consentId)) { - try { - ConsentResource consent = ConsentServiceUtil.getConsentService().getConsent(consentId, - false); - if (consent == null) { - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.RESOURCE_CONSENT_MISMATCH); - } - // Check whether the client id is matching - if (!consent.getClientID().equals(consentManageData.getClientId())) { - //Throwing same error as null scenario since client will not be able to identify if consent - // exists if consent does not belong to them - throw new ConsentException(ResponseStatus.BAD_REQUEST, - ErrorConstants.NO_CONSENT_FOR_CLIENT_ERROR); - } - JSONObject receiptJSON = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE). - parse(consent.getReceipt()); - consentManageData.setResponsePayload(ConsentManageUtil - .getInitiationRetrievalResponse(receiptJSON, consent, consentManageData, - ConsentExtensionConstants.PAYMENTS)); - consentManageData.setResponseStatus(ResponseStatus.OK); - } catch (ConsentManagementException | ParseException e) { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - ErrorConstants.ACC_INITIATION_RETRIEVAL_ERROR); - } - } else { - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.INVALID_CONSENT_ID); - } - } - - @Override - public void handleConsentManageDelete(ConsentManageData consentManageData) { - - } - private void handlePaymentPost(ConsentManageData consentManageData, JSONObject requestObject, JSONObject response) - throws ConsentManagementException { - - DetailedConsentResource createdConsent; - - //Validate Payment Initiation request - JSONObject validationResponse = PaymentsConsentRequestValidator - .validatePaymentInitiation(consentManageData.getRequestPath(), requestObject); - if (validationResponse.containsKey(ConsentExtensionConstants.IS_VALID) && - !((boolean) validationResponse.get(ConsentExtensionConstants.IS_VALID))) { - log.error(ErrorConstants.PAYLOAD_INVALID); - throw new ConsentException((ResponseStatus) validationResponse - .get(ConsentExtensionConstants.HTTP_CODE), - (JSONObject) validationResponse.get(ConsentExtensionConstants.ERRORS)); - } - - ConsentResource requestedConsent = new ConsentResource(consentManageData.getClientId(), - requestObject.toJSONString(), ConsentExtensionConstants.PAYMENTS, - ConsentExtensionConstants.AWAITING_AUTH_STATUS); - - createdConsent = ConsentExtensionsDataHolder.getInstance().getConsentCoreService() - .createAuthorizableConsent(requestedConsent, null, - CREATED_STATUS, AUTH_TYPE_AUTHORIZATION, true); - - Map consentAttributes = new HashMap(); - consentAttributes.put(ConsentExtensionConstants.IDEMPOTENCY_KEY, consentManageData.getHeaders() - .get(ConsentExtensionConstants.X_IDEMPOTENCY_KEY)); - ConsentServiceUtil.getConsentService().storeConsentAttributes(createdConsent.getConsentID(), - consentAttributes); - consentManageData.setResponsePayload(ConsentManageUtil.getInitiationResponse(response, createdConsent, - consentManageData, ConsentExtensionConstants.PAYMENTS)); - - Map headers = consentManageData.getHeaders(); - //Setting response headers - //Setting created time and idempotency to headers to handle idempotency in Gateway - consentManageData.setResponseHeader(ConsentExtensionConstants.X_IDEMPOTENCY_KEY, - headers.get(ConsentExtensionConstants.X_IDEMPOTENCY_KEY)); - consentManageData.setResponseStatus(ResponseStatus.CREATED); - } - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/VRPConsentRequestHandler.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/VRPConsentRequestHandler.java deleted file mode 100644 index 9ff83ac1..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/impl/VRPConsentRequestHandler.java +++ /dev/null @@ -1,342 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.manage.impl; - -import com.google.gson.Gson; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentServiceUtil; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.consent.extensions.manage.model.ConsentManageData; -import com.wso2.openbanking.accelerator.consent.extensions.manage.model.PeriodicLimit; -import com.wso2.openbanking.accelerator.consent.extensions.manage.validator.VRPConsentRequestValidator; -import com.wso2.openbanking.accelerator.consent.extensions.util.ConsentManageUtil; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.math.BigDecimal; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import static com.wso2.openbanking.accelerator.consent.extensions.common. - ConsentExtensionConstants.AUTH_TYPE_AUTHORIZATION; -import static com.wso2.openbanking.accelerator.consent.extensions. - common.ConsentExtensionConstants.CREATED_STATUS; - -/** - * Consent Manage request handler class for VRP Payment Request Validation. - */ -public class VRPConsentRequestHandler implements ConsentManageRequestHandler { - - private static final Log log = LogFactory.getLog(VRPConsentRequestHandler.class); - - - /** - * This method is responsible for processing a Variable Recurring Payment Consent Manage POST request. - * It validates the payment request, checks for the existence of an idempotency key. - * - * @param consentManageData Object - */ - @Override - public void handleConsentManagePost(ConsentManageData consentManageData) { - - try { - Object request = consentManageData.getPayload(); - - JSONObject validationResponse = VRPConsentRequestValidator.validateVRPPayload(request); - - if (!(Boolean.parseBoolean(validationResponse.getAsString(ConsentExtensionConstants.IS_VALID)))) { - log.error(validationResponse.get(ConsentExtensionConstants.ERRORS)); - throw new ConsentException((ResponseStatus) validationResponse - .get(ConsentExtensionConstants.HTTP_CODE), - String.valueOf(validationResponse.get(ConsentExtensionConstants.ERRORS))); - } - - if (StringUtils.isEmpty(consentManageData.getHeaders() - .get(ConsentExtensionConstants.X_IDEMPOTENCY_KEY))) { - log.error(ErrorConstants.IDEMPOTENCY_KEY_NOT_FOUND); - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.IDEMPOTENCY_KEY_NOT_FOUND); - } - - //Handle payment initiation flows - handlePaymentPost(consentManageData, request); - - } catch (ConsentManagementException e) { - log.error("Error occurred while handling the initiation request", e); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - ErrorConstants.VRP_INITIATION_HANDLE_ERROR); - } - } - - /** - * This method is responsible for handling the GET request for retrieving consent initiation details. - * It validates the consent ID, checks if the consent exists,verifies if the consent belongs to the - * client making the request. - * - * @param consentManageData Object - */ - @Override - public void handleConsentManageGet(ConsentManageData consentManageData) { - - String consentId = consentManageData.getRequestPath().split("/")[1]; - if (ConsentManageUtil.isConsentIdValid(consentId)) { - try { - ConsentResource consent = ConsentServiceUtil.getConsentService().getConsent(consentId, - false); - // Check whether the client id is matching - if (!consent.getClientID().equals(consentManageData.getClientId())) { - // Throws the error if the client Ids mismatch - if (log.isDebugEnabled()) { - log.debug(String.format("ClientIds missmatch. " + - "Retrieved client id: %s, ConsentmanageData client id: %s", - consent.getClientID(), consentManageData.getClientId())); - } - - throw new ConsentException(ResponseStatus.BAD_REQUEST, - "Invalid client id passed"); - } - - JSONObject receiptJSON = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE). - parse(consent.getReceipt()); - consentManageData.setResponsePayload(ConsentManageUtil - .getInitiationRetrievalResponse(receiptJSON, consent, consentManageData, - ConsentExtensionConstants.VRP)); - consentManageData.setResponseStatus(ResponseStatus.OK); - } catch (ConsentManagementException | ParseException e) { - log.error(ErrorConstants.INVALID_CLIENT_ID_MATCH, e); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - ErrorConstants.VRP_INITIATION_RETRIEVAL_ERROR); - } - } else { - log.error(ErrorConstants.INVALID_CONSENT_ID); - throw new ConsentException(ResponseStatus.BAD_REQUEST, ErrorConstants.INVALID_CONSENT_ID); - } - } - - /** - * Handles the DELETE request for revoking or deleting a consent. - * - * @param consentManageData Object containing request details - */ - @Override - public void handleConsentManageDelete(ConsentManageData consentManageData) { - - ConsentManageUtil.handleConsentManageDelete(consentManageData); - } - - /** - * Method to handle Variable Recurring Payment POST requests. - * This private method processes Variable Recurring Payment POST requests, creating a new consent - * based on the provided request payload. It performs the following actions: - * - Creates a DetailedConsentResource representing the consent initiation. - * - Stores consent attributes, including the idempotency key. - * - Constructs the response payload containing initiation details and sets appropriate headers. - * - Sets the response status to Created. - * - * @param consentManageData Object containing request details, including client ID, request payload, headers. - * @param request Object - * @throws ConsentManagementException if an error occurs while creating the consent or storing consent attributes. - */ - public void handlePaymentPost(ConsentManageData consentManageData, Object request) - throws ConsentManagementException { - - // Check if the request is a JSONObject - if (!(request instanceof JSONObject)) { - log.error("Invalid request type. Expected JSONObject."); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - ErrorConstants.PAYLOAD_FORMAT_ERROR); - } - - JSONObject requestObject = (JSONObject) request; - - // Create a ConsentResource representing the requested consent - ConsentResource requestedConsent = createRequestedConsent(consentManageData, requestObject); - - // Create the consent - DetailedConsentResource createdConsent = createConsent(requestedConsent); - - // Set consent attributes for storing - Map consentAttributes = createConsentAttributes(consentManageData); - - // Store consent attributes - ConsentServiceUtil.getConsentService().storeConsentAttributes(createdConsent.getConsentID(), - consentAttributes); - - // Set response payload and headers - setResponse(consentManageData, requestObject, createdConsent); - } - - /** - * Method to Create a ConsentResource object using the provided ConsentManageData and requestObject. - * - * @param consentManageData Object containing request details - * @param requestObject JSON object representing the request - * @return ConsentResource object - */ - private ConsentResource createRequestedConsent(ConsentManageData consentManageData, JSONObject requestObject) { - return new ConsentResource(consentManageData.getClientId(), - requestObject.toJSONString(), ConsentExtensionConstants.VRP, - ConsentExtensionConstants.AWAITING_AUTH_STATUS); - } - - /** - * Method to create a DetailedConsentResource object using the provided ConsentResource. - * - * @param requestedConsent ConsentResource object - * @return DetailedConsentResource object - * @throws ConsentManagementException if an error occurs while creating the consent - */ - private DetailedConsentResource createConsent(ConsentResource requestedConsent) throws ConsentManagementException { - return ConsentServiceUtil.getConsentService() - .createAuthorizableConsent(requestedConsent, null, - CREATED_STATUS, AUTH_TYPE_AUTHORIZATION, true); - } - - /** - * Method to Create a map of consent attributes using the provided ConsentManageData. - * - * @param consentManageData Object containing request details - * @return Map of consent attributes - */ - private Map createConsentAttributes(ConsentManageData consentManageData) { - Map consentAttributes = new HashMap<>(); - consentAttributes.put(ConsentExtensionConstants.IDEMPOTENCY_KEY, consentManageData.getHeaders() - .get(ConsentExtensionConstants.X_IDEMPOTENCY_KEY)); - - JSONObject controlParameters = getControlParameters(consentManageData); - JSONArray periodicLimitsArray = (JSONArray) controlParameters.get(ConsentExtensionConstants.PERIODIC_LIMITS); - - List periodicLimitsList = createPeriodicLimitsList(periodicLimitsArray); - - JSONObject controlParams = createControlParameters(controlParameters, periodicLimitsList); - - // Convert the JSONObject to a string - String consentAttributesJson = controlParams.toJSONString(); - - // Add the consentAttributesJson to the consentAttributes - consentAttributes.put(ConsentExtensionConstants.CONTROL_PARAMETERS, consentAttributesJson); - - return consentAttributes; - } - - /** - * Method to retrieve control parameters from the provided ConsentManageData. - * - * @param consentManageData Object containing request details - * @return JSONObject of control parameters - */ - private JSONObject getControlParameters(ConsentManageData consentManageData) { - return (JSONObject) ((JSONObject) ((JSONObject) consentManageData.getPayload()) - .get(ConsentExtensionConstants.DATA)).get(ConsentExtensionConstants.CONTROL_PARAMETERS); - } - - /** - * Method to create a list of PeriodicLimit objects from the provided JSONArray. - * - * @param periodicLimitsArray JSONArray of periodic limits - * @return List of PeriodicLimit objects - */ - private List createPeriodicLimitsList(JSONArray periodicLimitsArray) { - List periodicLimitsList = new ArrayList<>(); - - for (Object periodicLimit : periodicLimitsArray) { - JSONObject jsonObject = (JSONObject) periodicLimit; - String periodType = (String) jsonObject.get(ConsentExtensionConstants.PERIOD_TYPE); - BigDecimal amount = BigDecimal.valueOf(Double.parseDouble((String) jsonObject.get(ConsentExtensionConstants. - AMOUNT))); - String periodAlignment = (String) jsonObject.get(ConsentExtensionConstants.PERIOD_ALIGNMENT); - - PeriodicLimit periodicLimits = new PeriodicLimit(periodType, amount, periodAlignment); - periodicLimitsList.add(periodicLimits); - } - - return periodicLimitsList; - } - - /** - * Method to create JSONObject of control parameters using the provided JSONObject and - * list of PeriodicLimit objects. - * - * @param controlParameters JSONObject of control parameters - * @param periodicLimitsList List of PeriodicLimit objects - * @return JSONObject of control parameters - */ - private JSONObject createControlParameters(JSONObject controlParameters, List periodicLimitsList) { - Gson gson = new Gson(); - - // Get MaximumIndividualAmount from controlParameters - JSONObject maximumIndividualAmountObject = (JSONObject) controlParameters. - get(ConsentExtensionConstants.MAXIMUM_INDIVIDUAL_AMOUNT); - double maximumIndividualAmount = Double.parseDouble(maximumIndividualAmountObject - .get(ConsentExtensionConstants.AMOUNT).toString()); - - // Create a new JSONObject - JSONObject jsonObject = new JSONObject(); - - // Add MaximumIndividualAmount to the JSONObject - jsonObject.put(ConsentExtensionConstants.MAXIMUM_INDIVIDUAL_AMOUNT, maximumIndividualAmount); - - // Convert the periodicLimitsList to a JSON string - String periodicLimitsJson = gson.toJson(periodicLimitsList); - - // Parse the JSON string back to a JSONArray - JSONArray newPeriodicLimitsArray; - try { - newPeriodicLimitsArray = (JSONArray) new JSONParser(JSONParser.MODE_PERMISSIVE).parse(periodicLimitsJson); - } catch (ParseException e) { - throw new RuntimeException("Error parsing JSON", e); - } - - // Add the PeriodicLimits array to the JSONObject - jsonObject.put(ConsentExtensionConstants.PERIODIC_LIMITS, newPeriodicLimitsArray); - - return jsonObject; - } - - /** - * Method to set the response payload, headers, and status for the provided ConsentManageData using the - * provided requestObject and createdConsent. - * - * @param consentManageData Object containing request details - * @param requestObject JSON object representing the request - * @param createdConsent DetailedConsentResource object representing the created consent - */ - private void setResponse(ConsentManageData consentManageData, - JSONObject requestObject, DetailedConsentResource createdConsent) { - consentManageData.setResponsePayload(ConsentManageUtil.getInitiationResponse(requestObject, createdConsent, - consentManageData, ConsentExtensionConstants.VRP)); - - // Get request headers - Map headers = consentManageData.getHeaders(); - - consentManageData.setResponseHeader(ConsentExtensionConstants.X_IDEMPOTENCY_KEY, - headers.get(ConsentExtensionConstants.X_IDEMPOTENCY_KEY)); - consentManageData.setResponseStatus(ResponseStatus.CREATED); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/model/ConsentManageData.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/model/ConsentManageData.java deleted file mode 100644 index d6df4a6a..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/model/ConsentManageData.java +++ /dev/null @@ -1,125 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.manage.model; - -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; - -import java.util.Map; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; - -/** - * Data wrapper for consent manage data. - */ -public class ConsentManageData { - - private Map headers; - //Payload can either be a JSONObject or a JSONArray - private Object payload; - private Map queryParams; - private String requestPath; - private String clientId; - private HttpServletRequest request; - private HttpServletResponse response; - private ResponseStatus responseStatus; - private Object responsePayload; - - public ConsentManageData(Map headers, Object payload, Map queryParams, - String requestPath, HttpServletRequest request, HttpServletResponse response) { - this.headers = headers; - this.payload = payload; - this.queryParams = queryParams; - this.requestPath = requestPath; - this.request = request; - this.response = response; - } - - public ConsentManageData(Map headers, Map queryParams, - String requestPath, HttpServletRequest request, HttpServletResponse response) { - this.headers = headers; - this.requestPath = requestPath; - payload = null; - this.queryParams = queryParams; - this.request = request; - this.response = response; - } - - public HttpServletRequest getRequest() { - return request; - } - - public HttpServletResponse getResponse() { - return response; - } - - public Map getQueryParams() { - return queryParams; - } - - public Object getPayload() { - return payload; - } - - public Map getHeaders() { - return headers; - } - - public void setResponseStatus(ResponseStatus responseStatus) { - this.responseStatus = responseStatus; - } - - public void setResponsePayload(Object responsePayload) { - this.responsePayload = responsePayload; - } - - public Object getResponsePayload() { - return responsePayload; - } - - public ResponseStatus getResponseStatus() { - return responseStatus; - } - - public void setResponseHeader(String key, String value) { - response.setHeader(key, value); - } - - public void setResponseHeaders(Map headers) { - for (Map.Entry header : headers.entrySet()) { - setResponseHeader(header.getKey(), header.getValue()); - } - } - - public String getRequestPath() { - return requestPath; - } - - public void setRequestPath(String requestPath) { - this.requestPath = requestPath; - } - - public String getClientId() { - return clientId; - } - - public void setClientId(String clientId) { - this.clientId = clientId; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/model/ConsentManageHandler.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/model/ConsentManageHandler.java deleted file mode 100644 index a3850e99..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/model/ConsentManageHandler.java +++ /dev/null @@ -1,93 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.manage.model; - -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; - -/** - * Consent manage handler interface. - */ -public interface ConsentManageHandler { - - /** - * Function to handle GET requests received to the consent manage endpoint. - * - * @param consentManageData Object containing data regarding the request - * @throws ConsentException Error object with data required for the error response - */ - public void handleGet(ConsentManageData consentManageData) throws ConsentException; - - /** - * Function to handle POST requests received to the consent manage endpoint. - * - * @param consentManageData Object containing data regarding the request - * @throws ConsentException Error object with data required for the error response - */ - public void handlePost(ConsentManageData consentManageData) throws ConsentException; - - /** - * Function to handle DELETE requests received to the consent manage endpoint. - * - * @param consentManageData Object containing data regarding the request - * @throws ConsentException Error object with data required for the error response - */ - public void handleDelete(ConsentManageData consentManageData) throws ConsentException; - - /** - * Function to handle PUT requests received to the consent manage endpoint. - * - * @param consentManageData Object containing data regarding the request - * @throws ConsentException Error object with data required for the error response - */ - public void handlePut(ConsentManageData consentManageData) throws ConsentException; - - /** - * Function to handle PATCH requests received to the consent manage endpoint. - * - * @param consentManageData Object containing data regarding the request - * @throws ConsentException Error object with data required for the error response - */ - public void handlePatch(ConsentManageData consentManageData) throws ConsentException; - - /** - * Function to handle file upload POST requests received to the consent manage endpoint. - * Added as a default method to overcome the issues of existing customers since this was added as an update. - * - * @param consentManageData Object containing data regarding the request - * @throws ConsentException Error object with data required for the error response - */ - default void handleFileUploadPost(ConsentManageData consentManageData) throws ConsentException { - - throw new ConsentException(ResponseStatus.METHOD_NOT_ALLOWED, "File upload is not supported"); - } - - /** - * Function to handle file GET requests received to the consent manage endpoint. - * Added as a default method to overcome the issues of existing customers since this was added as an update. - * - * @param consentManageData Object containing data regarding the request - * @throws ConsentException Error object with data required for the error response - */ - default void handleFileGet(ConsentManageData consentManageData) throws ConsentException { - - throw new ConsentException(ResponseStatus.METHOD_NOT_ALLOWED, "File retrieval is not supported"); - } - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/model/PeriodicLimit.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/model/PeriodicLimit.java deleted file mode 100644 index 6bce52bb..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/model/PeriodicLimit.java +++ /dev/null @@ -1,202 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - *

- * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.manage.model; - -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.util.PeriodicTypesEnum; - -import java.math.BigDecimal; -import java.math.RoundingMode; -import java.time.DayOfWeek; -import java.time.Duration; -import java.time.Instant; -import java.time.LocalDate; -import java.time.Month; -import java.time.Period; -import java.time.ZoneId; -import java.time.temporal.ChronoUnit; -import java.time.temporal.TemporalAdjusters; - -/** - * This class represents a periodic limit for a VRP consent. - * It includes the period type, amount, period alignment, cyclic expiry time, and cyclic paid amount. - */ -public class PeriodicLimit { - - private final String periodType; - private BigDecimal amount; - private String periodAlignment; - private long cyclicExpiryTime; - private BigDecimal cyclicRemainingAmount; - - /** - * Constructs a new PeriodicLimit with the specified period type, amount, and period alignment. - * It also calculates and sets the cyclic expiry time and cyclic paid amount. - * - * @param periodType the period type - * @param amount the amount - * @param periodAlignment the period alignment - */ - public PeriodicLimit(String periodType, BigDecimal amount, String periodAlignment) { - this.periodType = periodType; - this.amount = amount; - this.periodAlignment = periodAlignment; - setCyclicExpiryTime(); - calculateCyclicPaidAmount(); - } - - public BigDecimal getAmount() { - return amount; - } - - public void setAmount(BigDecimal amount) { - this.amount = amount; - } - - public String getPeriodAlignment() { - return periodAlignment; - } - - public void setPeriodAlignment(String periodAlignment) { - this.periodAlignment = periodAlignment; - } - - public long getCyclicExpiryTime() { - return cyclicExpiryTime; - } - - public BigDecimal getCyclicRemainingAmount() { - return cyclicRemainingAmount; - } - - public void setCyclicRemainingAmount(BigDecimal cyclicRemainingAmount) { - this.cyclicRemainingAmount = cyclicRemainingAmount; - } - - /** - * Calculates and sets the cyclic expiry time based on the period type and period alignment. - */ - public void setCyclicExpiryTime() { - Instant now = Instant.now(); - Instant expiryTime; - - if (periodAlignment.equals(ConsentExtensionConstants.CONSENT)) { - expiryTime = calculateExpiryTimeForConsent(now); - } else if (periodAlignment.equals(ConsentExtensionConstants.CALENDAR)) { - expiryTime = calculateExpiryTimeForCalendar(now); - } else { - throw new IllegalArgumentException("Invalid PeriodAlignment"); - } - - cyclicExpiryTime = expiryTime.getEpochSecond(); - } - - - /** - * Calculates the expiry time for a consent based on the period type. - * - * @param now the current time - * @return the expiry time for a consent - */ - private Instant calculateExpiryTimeForConsent(Instant now) { - PeriodicTypesEnum periodType = PeriodicTypesEnum.valueOf(this.periodType.toUpperCase()); - switch (periodType) { - case DAY: - return now.plus(Duration.ofDays(1)); - case WEEK: - return now.plus(Duration.ofDays(7)); - case FORTNIGHT: - return now.plus(Duration.ofDays(14)); - case MONTH: - return now.plus(Period.ofMonths(1)); - case HALF_YEAR: - return now.plus(Period.ofMonths(6)); - case YEAR: - return now.plus(Period.ofYears(1)); - default: - throw new IllegalArgumentException("Invalid PeriodType"); - } - } - - /** - * Calculates the expiry time for a calendar based on the period type. - * - * @param now the current time - * @return the expiry time for a calendar - */ - private Instant calculateExpiryTimeForCalendar(Instant now) { - LocalDate localDate = now.atZone(ZoneId.systemDefault()).toLocalDate(); - PeriodicTypesEnum periodType = PeriodicTypesEnum.valueOf(this.periodType.toUpperCase()); - switch (periodType) { - case DAY: - return localDate.plusDays(1).atStartOfDay(ZoneId.systemDefault()).toInstant(); - case WEEK: - return localDate.with(TemporalAdjusters.next(DayOfWeek.MONDAY)).atStartOfDay(ZoneId.systemDefault()) - .toInstant(); - case FORTNIGHT: - return now.plus(Duration.ofDays(14)); - case MONTH: - return localDate.with(TemporalAdjusters.firstDayOfNextMonth()). - atStartOfDay(ZoneId.systemDefault()).toInstant(); - case HALF_YEAR: - return calculateHalfYearExpiry(localDate); - case YEAR: - return localDate.with(TemporalAdjusters.firstDayOfNextYear()).atStartOfDay(ZoneId.systemDefault()) - .toInstant(); - default: - throw new IllegalArgumentException("Invalid PeriodType"); - } - } - - /** - * Calculates the expiry time for a half year. - * - * @param localDate the current date - * @return the expiry time for a half year - */ - private Instant calculateHalfYearExpiry(LocalDate localDate) { - Month currentMonth = localDate.getMonth(); - if (currentMonth.getValue() < 7) { - return localDate.withMonth(6).with(TemporalAdjusters.lastDayOfMonth()).atStartOfDay(ZoneId.systemDefault()) - .toInstant(); - } else { - return localDate.withMonth(12).with(TemporalAdjusters.lastDayOfMonth()) - .atStartOfDay(ZoneId.systemDefault()) - .toInstant(); - } - } - - /** - * Calculates and sets the cyclic paid amount based on the period alignment. - */ - private void calculateCyclicPaidAmount() { - - if (periodAlignment.equalsIgnoreCase(ConsentExtensionConstants.CONSENT)) { - cyclicRemainingAmount = BigDecimal.valueOf(0); - } else if (periodAlignment.equalsIgnoreCase(ConsentExtensionConstants.CALENDAR)) { - LocalDate now = LocalDate.now(); - LocalDate expiryDate = Instant.ofEpochSecond(cyclicExpiryTime).atZone(ZoneId.systemDefault()) - .toLocalDate(); - BigDecimal divisor = BigDecimal.valueOf(PeriodicTypesEnum.valueOf(this.periodType.toUpperCase()) - .getDivisor()); - BigDecimal days = BigDecimal.valueOf(ChronoUnit.DAYS.between(now, expiryDate)); - cyclicRemainingAmount = amount.divide(divisor, RoundingMode.HALF_UP).multiply(days) - .setScale(2, RoundingMode.HALF_UP); - } - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/validator/CofConsentRequestValidator.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/validator/CofConsentRequestValidator.java deleted file mode 100644 index 9007ace6..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/validator/CofConsentRequestValidator.java +++ /dev/null @@ -1,152 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.manage.validator; - -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.util.ConsentManageUtil; -import net.minidev.json.JSONObject; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -/** - * Consent Manage validator class for Confirmation of Funds Request Validation. - */ -public class CofConsentRequestValidator { - - private static final Log log = LogFactory.getLog(CofConsentRequestValidator.class); - - /** - * Method to validate COF initiation request. - * @param initiation Initiation Object - * @return JSONObject Validation Response - */ - public static JSONObject validateCOFInitiation(JSONObject initiation) { - - JSONObject validationResponse = new JSONObject(); - - //Check request body is valid and not empty - if (!initiation.containsKey(ConsentExtensionConstants.DATA) || - !(initiation.get(ConsentExtensionConstants.DATA) instanceof JSONObject)) { - log.error(ErrorConstants.PAYLOAD_FORMAT_ERROR); - return ConsentManageUtil.getValidationResponse(ErrorConstants.RESOURCE_INVALID_FORMAT, - ErrorConstants.PAYLOAD_FORMAT_ERROR, ErrorConstants.PATH_REQUEST_BODY); - } - - JSONObject data = (JSONObject) initiation.get(ConsentExtensionConstants.DATA); - - //Validate json payload expirationDateTime is a future date - if (data.containsKey(ConsentExtensionConstants.EXPIRATION_DATE) && !ConsentManageUtil - .isConsentExpirationTimeValid(data.getAsString(ConsentExtensionConstants.EXPIRATION_DATE))) { - log.error(ErrorConstants.EXPIRED_DATE_ERROR); - return ConsentManageUtil.getValidationResponse(ErrorConstants.FIELD_INVALID_DATE, - ErrorConstants.EXPIRED_DATE_ERROR, ErrorConstants.PATH_EXPIRATION_DATE); - } - - - if (data.containsKey(ConsentExtensionConstants.DEBTOR_ACC)) { - - Object debtorAccountObj = data.get(ConsentExtensionConstants.DEBTOR_ACC); - //Check whether debtor account is a JsonObject - if (!(debtorAccountObj instanceof JSONObject)) { - log.error(ErrorConstants.MSG_INVALID_DEBTOR_ACC); - return ConsentManageUtil.getValidationResponse(ErrorConstants.FIELD_MISSING, - ErrorConstants.MSG_INVALID_DEBTOR_ACC, ErrorConstants.PATH_DEBTOR_ACCOUNT); - } - - JSONObject debtorAccount = (JSONObject) data.get(ConsentExtensionConstants.DEBTOR_ACC); - //Check whether debtor account is not empty - if (debtorAccount.isEmpty()) { - log.error(ErrorConstants.MSG_INVALID_DEBTOR_ACC); - return ConsentManageUtil.getValidationResponse(ErrorConstants.FIELD_MISSING, - ErrorConstants.MSG_INVALID_DEBTOR_ACC, ErrorConstants.PATH_DEBTOR_ACCOUNT); - } - - //Check Debtor Account Scheme name exists - if (!debtorAccount.containsKey(ConsentExtensionConstants.SCHEME_NAME) || - StringUtils.isEmpty(debtorAccount.getAsString(ConsentExtensionConstants.SCHEME_NAME))) { - log.error(ErrorConstants.MISSING_DEBTOR_ACC_SCHEME_NAME); - return ConsentManageUtil.getValidationResponse(ErrorConstants.FIELD_MISSING, - ErrorConstants.MISSING_DEBTOR_ACC_SCHEME_NAME, ErrorConstants.COF_PATH_DEBTOR_ACCOUNT_SCHEME); - } - - //Validate Debtor Account Scheme name - if (debtorAccount.containsKey(ConsentExtensionConstants.SCHEME_NAME) && - (!(debtorAccount.getAsString(ConsentExtensionConstants.SCHEME_NAME) instanceof String) || - !ConsentManageUtil.isDebtorAccSchemeNameValid(debtorAccount - .getAsString(ConsentExtensionConstants.SCHEME_NAME)))) { - log.error(ErrorConstants.INVALID_DEBTOR_ACC_SCHEME_NAME); - return ConsentManageUtil.getValidationResponse(ErrorConstants.FIELD_INVALID, - ErrorConstants.INVALID_DEBTOR_ACC_SCHEME_NAME, ErrorConstants.COF_PATH_DEBTOR_ACCOUNT_SCHEME); - } - - //Check Debtor Account Identification existing - if (!debtorAccount.containsKey(ConsentExtensionConstants.IDENTIFICATION) || - StringUtils.isEmpty(debtorAccount.getAsString(ConsentExtensionConstants.IDENTIFICATION))) { - log.error(ErrorConstants.MISSING_DEBTOR_ACC_IDENTIFICATION); - return ConsentManageUtil.getValidationResponse(ErrorConstants.FIELD_MISSING, - ErrorConstants.MISSING_DEBTOR_ACC_IDENTIFICATION, - ErrorConstants.COF_PATH_DEBTOR_ACCOUNT_IDENTIFICATION); - } - - //Validate Debtor Account Identification - if (debtorAccount.containsKey(ConsentExtensionConstants.IDENTIFICATION) && - (!(debtorAccount.getAsString(ConsentExtensionConstants.IDENTIFICATION) instanceof String) || - !ConsentManageUtil.isDebtorAccIdentificationValid(debtorAccount - .getAsString(ConsentExtensionConstants.IDENTIFICATION)))) { - log.error(ErrorConstants.INVALID_DEBTOR_ACC_IDENTIFICATION); - return ConsentManageUtil.getValidationResponse(ErrorConstants.FIELD_INVALID, - ErrorConstants.INVALID_DEBTOR_ACC_IDENTIFICATION, - ErrorConstants.COF_PATH_DEBTOR_ACCOUNT_IDENTIFICATION); - } - - //Validate Debtor Account Name - if (debtorAccount.containsKey(ConsentExtensionConstants.NAME) && - (!(debtorAccount.getAsString(ConsentExtensionConstants.NAME) instanceof String) || - !ConsentManageUtil.isDebtorAccNameValid(debtorAccount - .getAsString(ConsentExtensionConstants.NAME)))) { - log.error(ErrorConstants.INVALID_DEBTOR_ACC_NAME); - return ConsentManageUtil.getValidationResponse(ErrorConstants.FIELD_INVALID, - ErrorConstants.INVALID_DEBTOR_ACC_NAME, ErrorConstants.COF_PATH_DEBTOR_ACCOUNT_NAME); - } - - //Validate Debtor Account Secondary Identification - if (debtorAccount.containsKey(ConsentExtensionConstants.SECONDARY_IDENTIFICATION) && - (!(debtorAccount.getAsString(ConsentExtensionConstants.SECONDARY_IDENTIFICATION) - instanceof String) || - !ConsentManageUtil.isDebtorAccSecondaryIdentificationValid(debtorAccount - .getAsString(ConsentExtensionConstants.SECONDARY_IDENTIFICATION)))) { - log.error(ErrorConstants.INVALID_DEBTOR_ACC_SEC_IDENTIFICATION); - return ConsentManageUtil.getValidationResponse(ErrorConstants.FIELD_INVALID, - ErrorConstants.INVALID_DEBTOR_ACC_SEC_IDENTIFICATION, - ErrorConstants.COF_PATH_DEBTOR_ACCOUNT_SECOND_IDENTIFICATION); - } - } else { - log.error(ErrorConstants.MSG_MISSING_DEBTOR_ACC); - return ConsentManageUtil.getValidationResponse(ErrorConstants.FIELD_MISSING, - ErrorConstants.MSG_MISSING_DEBTOR_ACC, ErrorConstants.PATH_DEBTOR_ACCOUNT); - } - - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - - } -} - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/validator/PaymentsConsentRequestValidator.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/validator/PaymentsConsentRequestValidator.java deleted file mode 100644 index 2653591e..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/validator/PaymentsConsentRequestValidator.java +++ /dev/null @@ -1,61 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.manage.validator; - -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.util.ConsentManageUtil; -import com.wso2.openbanking.accelerator.consent.extensions.util.PaymentPayloadValidator; -import net.minidev.json.JSONObject; - -/** - * Consent Manage validator class for Payment Request Validation. - */ -public class PaymentsConsentRequestValidator { - - /** - * Method to validate payment initiation request. - * - * @param requestPath Request Path of the request - * @param initiation Initiation Object - * @return JSONObject Validation Response - */ - public static JSONObject validatePaymentInitiation(String requestPath, JSONObject initiation) { - - JSONObject validationResponse = new JSONObject(); - - //Check request body is valid and not empty - JSONObject dataValidationResult = ConsentManageUtil.validateInitiationDataBody(initiation); - if (!(boolean) dataValidationResult.get(ConsentExtensionConstants.IS_VALID)) { - return dataValidationResult; - } - - JSONObject data = (JSONObject) initiation.get(ConsentExtensionConstants.DATA); - - if (data.containsKey(ConsentExtensionConstants.INITIATION)) { - JSONObject initiationValidationResult = PaymentPayloadValidator - .validatePaymentInitiationPayload(requestPath, - (JSONObject) data.get(ConsentExtensionConstants.INITIATION)); - if (!(boolean) initiationValidationResult.get(ConsentExtensionConstants.IS_VALID)) { - return initiationValidationResult; - } - } - - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/validator/VRPConsentRequestValidator.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/validator/VRPConsentRequestValidator.java deleted file mode 100644 index 0c4f1847..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/manage/validator/VRPConsentRequestValidator.java +++ /dev/null @@ -1,815 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - *

- * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.manage.validator; - -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.util.ConsentManageUtil; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.time.OffsetDateTime; -import java.time.format.DateTimeFormatter; -import java.time.format.DateTimeParseException; -import java.util.HashSet; -import java.util.Iterator; -import java.util.Set; - - -/** - * Consent Manage validator class for Variable Recurring Payment Request Validation. - */ -public class VRPConsentRequestValidator { - - private static final Log log = LogFactory.getLog(VRPConsentRequestValidator.class); - - /** - * Method to validate a variable recurring payment request. - * This method performs validation on the variable recurring payment request. - * It checks the validity of the data body, the initiation payload, control parameters, - * and ensures that the risk information is present. If any validation fails, the method returns a detailed - * validation response indicating the error. If all validations pass, the returned validation response - * indicates that the initiation request is valid. - * - * @param request The initiation object containing the variable recurring payment initiation request. - * @return A validation response object indicating whether the initiation request is valid. - */ - public static JSONObject validateVRPPayload(Object request) { - - JSONObject validationResponse = new JSONObject(); - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - - //Get the request payload from the ConsentManageData - if (!(request instanceof JSONObject)) { - log.error(ErrorConstants.PAYLOAD_FORMAT_ERROR); - return ConsentManageUtil.getValidationResponse(ErrorConstants.PAYLOAD_FORMAT_ERROR); - } - - JSONObject requestBody = (JSONObject) request; - //Check request body is valid and not empty - JSONObject dataValidationResult = ConsentManageUtil.validateInitiationDataBody(requestBody); - - if (!(Boolean.parseBoolean(dataValidationResult.getAsString(ConsentExtensionConstants.IS_VALID)))) { - return dataValidationResult; - } - - //Check consent initiation is valid and not empty - JSONObject initiationValidationResult = VRPConsentRequestValidator.validateConsentInitiation(requestBody); - - if (!(Boolean.parseBoolean(initiationValidationResult.getAsString(ConsentExtensionConstants.IS_VALID)))) { - return initiationValidationResult; - } - - JSONObject controlParameterValidationResult = VRPConsentRequestValidator. - validateConsentControlParameters(requestBody); - - if (!(Boolean.parseBoolean(controlParameterValidationResult. - getAsString(ConsentExtensionConstants.IS_VALID)))) { - return controlParameterValidationResult; - } - - JSONObject riskValidationResult = VRPConsentRequestValidator.validateConsentRisk(requestBody); - - if (!(Boolean.parseBoolean(riskValidationResult.getAsString(ConsentExtensionConstants.IS_VALID)))) { - return riskValidationResult; - } - - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } - - /** - * Method to validate control parameters for variable recurring payments. - * This method performs validation on the control parameters for variable recurring payments. - * It checks the validity of maximum individual amount, requested execution date-time, and periodic limits. - * If any validation fails, the method returns a detailed validation response indicating the error. - * If all validations pass, the returned validation response indicates that the control parameters are valid. - * - * @param controlParameters The initiation object containing control parameters for variable recurring payments. - * @return A validation response object indicating whether the control parameters are valid. - */ - public static JSONObject validateControlParameters(JSONObject controlParameters) { - JSONObject validationResponse = new JSONObject(); - - JSONObject maximumIndividualAmountResult = validateMaximumIndividualAmount(controlParameters); - if (!(Boolean.parseBoolean(maximumIndividualAmountResult.getAsString(ConsentExtensionConstants.IS_VALID)))) { - return maximumIndividualAmountResult; - } - - JSONObject maximumIndividualAmountCurrencyValidationResult = validateMaximumIndividualAmountCurrency - (controlParameters); - if (!(Boolean.parseBoolean(maximumIndividualAmountCurrencyValidationResult. - getAsString(ConsentExtensionConstants.IS_VALID)))) { - return maximumIndividualAmountCurrencyValidationResult; - } - - JSONObject parameterDateTimeValidationResult = validateParameterDateTime(controlParameters); - if (!(Boolean.parseBoolean(parameterDateTimeValidationResult. - getAsString(ConsentExtensionConstants.IS_VALID)))) { - return parameterDateTimeValidationResult; - } - - // Validate Periodic Limits - JSONObject periodicLimitsValidationResult = validatePeriodicLimits(controlParameters); - if (!(Boolean.parseBoolean(periodicLimitsValidationResult.getAsString(ConsentExtensionConstants.IS_VALID)))) { - return periodicLimitsValidationResult; - } - - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } - - /** - * Checks whether the given object is a valid JSONArray. - * This method verifies if the provided object is not null and is an instance of JSONArray. - * It is commonly used to validate whether a given object represents a valid JSON array. - * - * @param value The object to be checked for being a valid JSONArray. - * @return true if the object is a valid JSONArray, false otherwise. - */ - public static boolean isValidJSONArray(Object value) { - String errorMessage = String.format(ErrorConstants.INVALID_PARAMETER_MESSAGE, "periodic limit", - "JSONObject"); - return value instanceof JSONArray; - } - - /** - * Validates the Maximum Individual Amount in the control parameters of a consent request. - * - * @param controlParameters The JSON object representing the control parameters of the consent request. - * @return A JSON object containing the validation response. - */ - public static JSONObject validateMaximumIndividualAmount(JSONObject controlParameters) { - - JSONObject validationResponse = new JSONObject(); - - //Validate Maximum individual amount in control parameters - if (controlParameters.containsKey(ConsentExtensionConstants.MAXIMUM_INDIVIDUAL_AMOUNT)) { - - Object maximumIndividualAmount = controlParameters. - get(ConsentExtensionConstants.MAXIMUM_INDIVIDUAL_AMOUNT); - - String errorMessage = String.format(ErrorConstants.INVALID_PARAMETER_MESSAGE, - "maximum individual amount", "JSONObject"); - - // Check if the control parameter is valid - if (!isValidJSONObject(maximumIndividualAmount)) { - return ConsentManageUtil.getValidationResponse(errorMessage); - } - - JSONObject maximumIndividualAmountResult = validateJsonObjectKey((JSONObject) maximumIndividualAmount, - ConsentExtensionConstants.AMOUNT, String.class); - if (!(Boolean.parseBoolean(maximumIndividualAmountResult. - getAsString(ConsentExtensionConstants.IS_VALID)))) { - return maximumIndividualAmountResult; - } - } else { - log.error(ErrorConstants.MISSING_MAXIMUM_INDIVIDUAL_AMOUNT); - return ConsentManageUtil.getValidationResponse(ErrorConstants.MISSING_MAXIMUM_INDIVIDUAL_AMOUNT); - - } - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } - - /** - * Validates the Currency in Maximum Individual Amount in the control parameters of a consent request. - * - * @param controlParameters The JSON object representing the control parameters of the consent request. - * @return A JSON object containing the validation response. - */ - public static JSONObject validateMaximumIndividualAmountCurrency(JSONObject controlParameters) { - // Retrieve the maximum individual amount from the control parameters - Object maximumIndividualAmount = controlParameters.get(ConsentExtensionConstants.MAXIMUM_INDIVIDUAL_AMOUNT); - - // Validate the currency of the maximum individual amount - JSONObject maximumIndividualAmountValidationResult = validateJsonObjectKey((JSONObject) maximumIndividualAmount, - ConsentExtensionConstants.CURRENCY, String.class); - if (!(Boolean.parseBoolean(maximumIndividualAmountValidationResult. - getAsString(ConsentExtensionConstants.IS_VALID)))) { - return maximumIndividualAmountValidationResult; - } - String maximumIndividualAmountCurrency; - maximumIndividualAmountCurrency = ((JSONObject) maximumIndividualAmount). - getAsString(ConsentExtensionConstants.CURRENCY); - - // Retrieve the periodic limits from the control parameters - JSONArray periodicLimits = (JSONArray) controlParameters.get(ConsentExtensionConstants.PERIODIC_LIMITS); - - // Iterate over the periodic limits and check if the currency of the limit is the same as the currency - // of maximum individual amount - for (Object limitObj : periodicLimits) { - if (limitObj instanceof JSONObject) { - JSONObject limit = (JSONObject) limitObj; - String limitCurrency = limit.getAsString(ConsentExtensionConstants.CURRENCY); - if (!maximumIndividualAmountCurrency.equals(limitCurrency)) { - log.error(ErrorConstants.CURRENCY_MISMATCH); - return ConsentManageUtil.getValidationResponse(ErrorConstants.FIELD_INVALID, - ErrorConstants.CURRENCY_MISMATCH, - ErrorConstants.PATH_PERIOD_LIMIT_CURRENCY); - } - } - } - - // If all validations pass, return a valid response - JSONObject validationResponse = new JSONObject(); - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } - - /** - * Method to validate variable recurring payment periodic limits. - * This method validates the periodic limits specified in the control parameters for variable recurring payments. - * It checks if the provided JSON array of periodic limits is valid and then iterates through each limit - * to ensure that required fields such as amount, currency, period alignment, and period type are present and - * meet the specified criteria. - * - * @param controlParameters Initiation Object containing periodic limits - * @return validation response object indicating whether the provided periodic limits are valid - */ - public static JSONObject validatePeriodicLimits(JSONObject controlParameters) { - JSONObject validationResponse = new JSONObject(); - - // Check if the periodic limits key is present - if (controlParameters.containsKey(ConsentExtensionConstants.PERIODIC_LIMITS)) { - - // Retrieve the periodic limits from the control parameters - Object periodicLimit = controlParameters.get(ConsentExtensionConstants.PERIODIC_LIMITS); - - // Check if the control parameter is a valid JSON array - if (!isValidJSONArray(periodicLimit)) { - return ConsentManageUtil.getValidationResponse(ErrorConstants.INVALID_PARAMETER_PERIODIC_LIMITS); - } - JSONArray periodicLimits = (JSONArray) controlParameters.get(ConsentExtensionConstants.PERIODIC_LIMITS); - // Create a set to store the periodTypes - Set periodTypes = new HashSet<>(); - Iterator parameters = periodicLimits.iterator(); - - while (parameters.hasNext()) { - JSONObject limit = (JSONObject) parameters.next(); - JSONObject amountValidationResult = validateAmountPeriodicLimit(controlParameters); - if (!(Boolean.parseBoolean(amountValidationResult. - getAsString(ConsentExtensionConstants.IS_VALID)))) { - return amountValidationResult; - } - - JSONObject currencyValidationResult = validateCurrencyPeriodicLimit(controlParameters); - if (!(Boolean.parseBoolean(currencyValidationResult. - getAsString(ConsentExtensionConstants.IS_VALID)))) { - return currencyValidationResult; - } - - JSONObject periodAlignmentValidationResult = validatePeriodAlignment(limit); - if (!(Boolean.parseBoolean(periodAlignmentValidationResult. - getAsString(ConsentExtensionConstants.IS_VALID)))) { - return periodAlignmentValidationResult; - } - - JSONObject periodTypeValidationResult = validatePeriodType(limit); - if (!(Boolean.parseBoolean(periodTypeValidationResult. - getAsString(ConsentExtensionConstants.IS_VALID)))) { - return periodTypeValidationResult; - } - - //Check if periodicLimits size exceeds 6 - if (periodicLimits.size() > ErrorConstants.MAXIMUM_PERIODIC_LIMITS) { - log.error(ErrorConstants.INVALID_PERIODIC_LIMIT_SIZE); - return ConsentManageUtil.getValidationResponse(ErrorConstants.FIELD_INVALID, - ErrorConstants.INVALID_PERIODIC_LIMIT_SIZE, - ErrorConstants.PATH_PERIOD_LIMIT); - } - - // Get the periodType from the limit - String periodType = limit.getAsString(ConsentExtensionConstants.PERIOD_TYPE); - - // If the periodType is already in the set, log an error and return a validation response - if (!periodTypes.add(periodType)) { - log.error(ErrorConstants.DUPLICATE_PERIOD_TYPE); - return ConsentManageUtil.getValidationResponse(ErrorConstants.FIELD_INVALID, - ErrorConstants.DUPLICATE_PERIOD_TYPE, - ErrorConstants.PATH_PERIOD_TYPE); - } - } - } else { - // If periodic limits key is missing, return an error - log.error(ErrorConstants.MISSING_PERIOD_LIMITS); - return ConsentManageUtil.getValidationResponse(ErrorConstants.MISSING_PERIOD_LIMITS); - } - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } - - /** - * Validates the Currency in periodic limits in the control parameters of a consent request. - * - * @param controlParameters The JSON object representing the control parameters of the consent request. - * @return A JSON object containing the validation response. - */ - public static JSONObject validateCurrencyPeriodicLimit(JSONObject controlParameters) { - - JSONObject validationResponse = new JSONObject(); - - JSONArray periodicLimits = (JSONArray) controlParameters.get(ConsentExtensionConstants.PERIODIC_LIMITS); - JSONObject currencyValidationResponse = validateAmountCurrencyPeriodicLimits((JSONArray) periodicLimits, - ConsentExtensionConstants.CURRENCY, String.class); - if (!(Boolean.parseBoolean(currencyValidationResponse. - getAsString(ConsentExtensionConstants.IS_VALID)))) { - return currencyValidationResponse; - } - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } - - /** - * Validates the Amount in periodic limits in the control parameters of a consent request. - * - * @param controlParameters The JSON object representing the control parameters of the consent request. - * @return A JSON object containing the validation response. - */ - public static JSONObject validateAmountPeriodicLimit(JSONObject controlParameters) { - - JSONObject validationResponse = new JSONObject(); - - JSONArray periodicLimits = (JSONArray) controlParameters.get(ConsentExtensionConstants.PERIODIC_LIMITS); - - JSONObject amountValidationResponse = validateAmountCurrencyPeriodicLimits((JSONArray) periodicLimits, - ConsentExtensionConstants.AMOUNT, String.class); - if (!(Boolean.parseBoolean(amountValidationResponse. - getAsString(ConsentExtensionConstants.IS_VALID)))) { - return amountValidationResponse; - } - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } - - /** - * Validates the date-time parameters in the control parameters of a consent request. - * - * @param controlParameters The JSON object representing the control parameters of the consent request. - * @return A JSON object containing the validation response. If the date-time parameters are valid, - * it sets the "IS_VALID" field to true; otherwise, it contains an error response. - */ - public static JSONObject validateParameterDateTime(JSONObject controlParameters) { - JSONObject validationResponse = new JSONObject(); - - if (controlParameters.containsKey(ConsentExtensionConstants.VALID_TO_DATE_TIME)) { - - if (!ConsentManageUtil.isValid8601(controlParameters - .getAsString(ConsentExtensionConstants.VALID_TO_DATE_TIME))) { - log.error(" Date and Time is not in valid ISO 8601 format"); - return ConsentManageUtil.getValidationResponse(ErrorConstants.INVALID_VALID_TO_DATE_TIME); - } - - Object validToDateTimeRetrieval = controlParameters.get(ConsentExtensionConstants.VALID_TO_DATE_TIME); - JSONObject validToDateTimeValidationResponse = isValidDateTimeObject(validToDateTimeRetrieval); - if (!(Boolean.parseBoolean(validToDateTimeValidationResponse. - getAsString(ConsentExtensionConstants.IS_VALID)))) { - return validToDateTimeValidationResponse; - } - - String validToDateTimeString = controlParameters.getAsString(ConsentExtensionConstants.VALID_TO_DATE_TIME); - OffsetDateTime validToDateTime = OffsetDateTime.parse(validToDateTimeString); - - if (controlParameters.containsKey(ConsentExtensionConstants.VALID_FROM_DATE_TIME)) { - - if (!ConsentManageUtil.isValid8601(controlParameters - .getAsString(ConsentExtensionConstants.VALID_FROM_DATE_TIME))) { - log.error("Date and Time is not in valid ISO 8601 format"); - return ConsentManageUtil.getValidationResponse(ErrorConstants.INVALID_VALID_FROM_DATE_TIME); - } - - - Object validFromDateTimeRetrieval = controlParameters.get - (ConsentExtensionConstants.VALID_FROM_DATE_TIME); - JSONObject validFromDateTimeValidationResponse = isValidDateTimeObject(validFromDateTimeRetrieval); - if (!(Boolean.parseBoolean(validFromDateTimeValidationResponse. - getAsString(ConsentExtensionConstants.IS_VALID)))) { - return validFromDateTimeValidationResponse; - } - - String validFromoDateTimeString = controlParameters.getAsString - (ConsentExtensionConstants.VALID_FROM_DATE_TIME); - - OffsetDateTime validFromDateTime = OffsetDateTime.parse(validFromoDateTimeString); - OffsetDateTime currentDateTime = OffsetDateTime.now(validToDateTime.getOffset()); - - // If ValidToDateTime is older than current date OR ValidToDateTime is older than ValidFromDateTime, - // return error - if (!validFromDateTime.isBefore(currentDateTime) || !currentDateTime.isBefore(validToDateTime)) { - log.error(String.format("Invalid date-time range, " + - "validToDateTime: %s, validFromDateTime: %s, currentDateTime: %s", - validToDateTime, validFromDateTime, currentDateTime)); - - String errorMessage = String.format(ErrorConstants.DATE_INVALID_PARAMETER_MESSAGE); - - return ConsentManageUtil.getValidationResponse(errorMessage); - } - } else { - log.error("validFromDateTime parameter is missing in the payload"); - return ConsentManageUtil.getValidationResponse(ErrorConstants.MISSING_VALID_FROM_DATE_TIME); - } - } else { - log.error("Missing validToDateTime parameter is missing in the payload"); - return ConsentManageUtil.getValidationResponse(ErrorConstants.MISSING_VALID_TO_DATE_TIME); - } - - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } - - /** - * Validator class to validate the payload of a variable recurring payment initiation. - * This method performs validation on the initiation payload for a variable recurring payment. - * It checks and validates the debtor account and creditor account information if present in the payload. - * If any validation fails, it returns a JSON object with details about the validation error. - * If the initiation payload passes all validations, the returned JSON object indicates a valid initiation. - * - * @param initiation The JSON object representing the variable recurring payment initiation payload. - * @return validationResponse - */ - public static JSONObject validateVRPInitiationPayload(JSONObject initiation) { - - JSONObject validationResponse = new JSONObject(); - - //Validate DebtorAccount - if (initiation.containsKey(ConsentExtensionConstants.DEBTOR_ACC)) { - - Object debtorAccount = initiation.get(ConsentExtensionConstants.DEBTOR_ACC); - - if (!isValidJSONObject(debtorAccount)) { - String errorMessage = String.format(ErrorConstants.INVALID_PARAMETER_MESSAGE, - "debtor account", "JSONObject"); - - return ConsentManageUtil.getValidationResponse(errorMessage); - } - - JSONObject validationResult = ConsentManageUtil.validateDebtorAccount((JSONObject) debtorAccount); - - if (!(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID)))) { - log.error(validationResult.get(ConsentExtensionConstants.ERRORS)); - return validationResult; - } - - } else { - log.error(ErrorConstants.PAYLOAD_FORMAT_ERROR_DEBTOR_ACC); - return ConsentManageUtil.getValidationResponse(ErrorConstants.PAYLOAD_FORMAT_ERROR_DEBTOR_ACC); - } - - //Validate CreditorAccount - if (initiation.containsKey(ConsentExtensionConstants.CREDITOR_ACC)) { - - Object creditorAccount = initiation.get(ConsentExtensionConstants.CREDITOR_ACC); - - if (!isValidJSONObject(creditorAccount)) { - String errorMessage = String.format(ErrorConstants.INVALID_PARAMETER_MESSAGE, - "creditor account", "JSONObject"); - - return ConsentManageUtil.getValidationResponse(errorMessage); - } - - JSONObject validationResult = ConsentManageUtil.validateCreditorAccount((JSONObject) creditorAccount); - - if (!(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID)))) { - log.error(validationResult.get(ConsentExtensionConstants.ERRORS)); - return validationResult; - } - - } else { - log.error(ErrorConstants.PAYLOAD_FORMAT_ERROR_CREDITOR_ACC); - return ConsentManageUtil.getValidationResponse(ErrorConstants.PAYLOAD_FORMAT_ERROR); - } - - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } - - - /** - * Validates the presence of a specified key in a JSONObject (either the amount or the currency) - * and checks if the associated value is a non-empty string. - * - * @param parentObj The JSONObject to be validated. - * @param key The key to be checked for presence in the parentObj. - * @param expectedType The expected type of the value associated with the key. - * @param The expected type of the value associated with the key. - * @return true if the specified key is present in the parentObj and the associated value is a - * non-empty string. - */ - public static JSONObject validateJsonObjectKey(JSONObject parentObj, String key, Class expectedType) { - JSONObject validationResponse = new JSONObject(); - //Refractor removing passing class - if (parentObj != null) { - if (parentObj.containsKey(key)) { - Object value = parentObj.get(key); - - if (expectedType.isInstance(value)) { - if (value instanceof String && !((String) value).isEmpty()) { - if ("Amount".equals(key)) { - // For the "amount" key, try parsing as Double allowing letters - if (isDouble((String) value)) { - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } else { - String errorMessage = "The value of '" + key + "' is not a valid number"; - return ConsentManageUtil.getValidationResponse(errorMessage); - } - } else { - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } - } else { - String errorMessage = "The value of '" + key + "' is not a " + expectedType.getSimpleName() - + " or the value is empty"; - return ConsentManageUtil.getValidationResponse(errorMessage); - } - } else { - String errorMessage = "The value of '" + key + "' is not of type " + expectedType.getSimpleName(); - return ConsentManageUtil.getValidationResponse(errorMessage); - } - } else { - String errorMessage = "Mandatory parameter '" + key + "' is not present in payload"; - return ConsentManageUtil.getValidationResponse(errorMessage); - } - } else { - String errorMessage = "parameter passed in is null"; - return ConsentManageUtil.getValidationResponse(errorMessage); - } - } - - /** - * Validates the presence of a specified key in a JSONArray (either the amount or the currency) - * in periodiclimits and checks if the associated value is a non-empty string. - * - * @param parentArray The JSONObject to be validated. - * @param key The key to be checked for presence in the parentObj. - * @param expectedType The expected type of the value associated with the key. - * @param The expected type of the value associated with the key. - * @return A JSONObject containing validation results for the entire array. - */ - public static JSONObject validateAmountCurrencyPeriodicLimits(JSONArray parentArray, String key, - Class expectedType) { - JSONObject validationResponse = new JSONObject(); - - if (parentArray != null) { - for (Object obj : parentArray) { - if (obj instanceof JSONObject) { - - JSONObject jsonObject = (JSONObject) obj; - JSONObject elementValidationResult = validateJsonObjectKey(jsonObject, key, expectedType); - - if (!(Boolean.parseBoolean(elementValidationResult.getAsString - (ConsentExtensionConstants.IS_VALID)))) { - return elementValidationResult; - } - } - } - } else { - String errorMessage = "parameter passed in is null"; - return ConsentManageUtil.getValidationResponse(errorMessage); - } - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } - - - /** - Checks if a given string can be parsed into a double value. - @param str The string to be checked. - @return True if the string can be parsed into a double value, false otherwise. - */ - private static boolean isDouble(String str) { - try { - Double.parseDouble(str); - return true; - } catch (NumberFormatException e) { - return false; - } - } - - /** - * Validates the consent initiation payload in the VRP request. - * - * @param request The JSONObject representing the VRP request. - * @return A JSONObject containing the validation response. - */ - public static JSONObject validateConsentInitiation(JSONObject request) { - - JSONObject validationResponse = new JSONObject(); - - JSONObject requestBody = (JSONObject) request; - JSONObject data = (JSONObject) requestBody.get(ConsentExtensionConstants.DATA); - - //Validate initiation in the VRP payload - if (data.containsKey(ConsentExtensionConstants.INITIATION)) { - - Object initiation = data.get(ConsentExtensionConstants.INITIATION); - - if (!isValidJSONObject(initiation)) { - String errorMessage = String.format(ErrorConstants.INVALID_PARAMETER_MESSAGE, - "initiation", "JSONObject"); - - return ConsentManageUtil.getValidationResponse(errorMessage); - } - - JSONObject initiationValidationResult = VRPConsentRequestValidator - .validateVRPInitiationPayload((JSONObject) initiation); - - if (!(Boolean.parseBoolean(initiationValidationResult.getAsString(ConsentExtensionConstants.IS_VALID)))) { - return initiationValidationResult; - } - } else { - log.error(ErrorConstants.PAYLOAD_FORMAT_ERROR_INITIATION); - return ConsentManageUtil.getValidationResponse(ErrorConstants.PAYLOAD_FORMAT_ERROR_INITIATION); - } - - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } - - /** - * Validates the consent control parameters in the VRP request payload. - * - * @param request The JSONObject representing the VRP request. - * @return A JSONObject containing the validation response. - */ - public static JSONObject validateConsentControlParameters(JSONObject request) { - - JSONObject validationResponse = new JSONObject(); - - JSONObject requestBody = (JSONObject) request; - JSONObject data = (JSONObject) requestBody.get(ConsentExtensionConstants.DATA); - - //Validate the ControlParameter in the payload - if (data.containsKey(ConsentExtensionConstants.CONTROL_PARAMETERS)) { - - Object controlParameters = data.get(ConsentExtensionConstants.CONTROL_PARAMETERS); - - if (!isValidJSONObject(controlParameters)) { - String errorMessage = String.format(ErrorConstants.INVALID_PARAMETER_MESSAGE, - "control parameters", "JSONObject"); - - return ConsentManageUtil.getValidationResponse(errorMessage); - } - - JSONObject controlParameterValidationResult = - VRPConsentRequestValidator.validateControlParameters((JSONObject) - data.get(ConsentExtensionConstants.CONTROL_PARAMETERS)); - - if (!(Boolean.parseBoolean(controlParameterValidationResult. - getAsString(ConsentExtensionConstants.IS_VALID)))) { - return controlParameterValidationResult; - } - } else { - log.error(ErrorConstants.PAYLOAD_FORMAT_ERROR_CONTROL_PARAMETER); - return ConsentManageUtil.getValidationResponse(ErrorConstants.PAYLOAD_FORMAT_ERROR_CONTROL_PARAMETER); - } - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } - - /** - * Validates the risk information in the VRP request payload. - * - * @param request The JSONObject representing the VRP request. - * @return A JSONObject containing the validation response. - */ - public static JSONObject validateConsentRisk(JSONObject request) { - - JSONObject validationResponse = new JSONObject(); - - JSONObject requestBody = (JSONObject) request; - JSONObject data = (JSONObject) requestBody.get(ConsentExtensionConstants.DATA); - - // Check Risk key is mandatory - if (!requestBody.containsKey(ConsentExtensionConstants.RISK) || - !(requestBody.get(ConsentExtensionConstants.RISK) instanceof JSONObject - || ((JSONObject) requestBody.get(ConsentExtensionConstants.DATA)).isEmpty())) { - log.error(ErrorConstants.PAYLOAD_FORMAT_ERROR_RISK); - return ConsentManageUtil.getValidationResponse(ErrorConstants.PAYLOAD_FORMAT_ERROR_RISK); - } - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } - - /** - * Validates the periodic alignment in the VRP request payload. - * - * @param limit The JSONObject representing the VRP request. - * @return A JSONObject containing the validation response. - */ - public static JSONObject validatePeriodAlignment(JSONObject limit) { - JSONObject validationResponse = new JSONObject(); - - if (limit.containsKey(ConsentExtensionConstants.PERIOD_ALIGNMENT)) { - Object periodAlignmentObj = limit.get(ConsentExtensionConstants.PERIOD_ALIGNMENT); - - if (periodAlignmentObj instanceof String && !((String) periodAlignmentObj).isEmpty()) { - String periodAlignment = (String) periodAlignmentObj; - - if (ConsentExtensionConstants.CONSENT.equals(periodAlignment) || - ConsentExtensionConstants.CALENDAR.equals(periodAlignment)) { - - validationResponse.put("isValid", true); - validationResponse.put("periodAlignment", periodAlignment); - } else { - return ConsentManageUtil.getValidationResponse(ErrorConstants.INVALID_PERIOD_ALIGNMENT); - } - } else { - return ConsentManageUtil.getValidationResponse(ErrorConstants. - PAYLOAD_FORMAT_ERROR_PERIODIC_LIMITS_ALIGNMENT); - } - } else { - return ConsentManageUtil.getValidationResponse(ErrorConstants.MISSING_PERIOD_ALIGNMENT); - } - return validationResponse; - } - - /** - * Validates the periodic type in the VRP request payload. - * - * @param limit The JSONObject representing the VRP request. - * @return A JSONObject containing the validation response. - */ - public static JSONObject validatePeriodType(JSONObject limit) { - JSONObject validationResponse = new JSONObject(); - - if (limit.containsKey(ConsentExtensionConstants.PERIOD_TYPE)) { - Object periodTypeObj = limit.get(ConsentExtensionConstants.PERIOD_TYPE); - - if (periodTypeObj instanceof String && !((String) periodTypeObj).isEmpty()) { - String periodType = (String) periodTypeObj; - - if (ConsentExtensionConstants.DAY.equals(periodType) || - ConsentExtensionConstants.WEEK.equals(periodType) || - ConsentExtensionConstants.FORTNIGHT.equals(periodType) || - ConsentExtensionConstants.MONTH.equals(periodType) || - ConsentExtensionConstants.HALF_YEAR.equals(periodType) || - ConsentExtensionConstants.YEAR.equals(periodType)) { - - validationResponse.put("isValid", true); - validationResponse.put("periodAlignment", periodType); - } else { - return ConsentManageUtil.getValidationResponse(ErrorConstants.INVALID_PERIOD_TYPE); - } - } else { - return ConsentManageUtil.getValidationResponse(ErrorConstants. - PAYLOAD_FORMAT_ERROR_PERIODIC_LIMITS_PERIOD_TYPE); - } - } else { - return ConsentManageUtil.getValidationResponse(ErrorConstants.MISSING_PERIOD_TYPE); - } - return validationResponse; - } - - /** - * Checks if the given Object is a JSONObject and the JSONObject is non-empty . - * - * @param value The Object to be validated. - * @return true if the object is a non-null and non-empty JSONObject. - */ - public static boolean isValidJSONObject(Object value) { - return value instanceof JSONObject && !((JSONObject) value).isEmpty(); - } - - /** - * Checks if the given object is a valid date-time string and it is non empty. - * - * @param value The object to be checked for a valid date-time format. - * @return True if the object is a non-empty string in ISO date-time format, false otherwise. - */ - private static final DateTimeFormatter dateTimeFormat = DateTimeFormatter.ISO_DATE_TIME; - - public static JSONObject isValidDateTimeObject(Object value) { - JSONObject validationResponse = new JSONObject(); - - if (value instanceof String && !((String) value).isEmpty()) { - try { - String dateTimeString = (String) value; - dateTimeFormat.parse(dateTimeString); - } catch (DateTimeParseException e) { - return ConsentManageUtil.getValidationResponse(ErrorConstants.INVALID_DATE_TIME_FORMAT); - } - } else { - return ConsentManageUtil.getValidationResponse(ErrorConstants.MISSING_DATE_TIME_FORMAT); - } - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/ConsentManageUtil.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/ConsentManageUtil.java deleted file mode 100644 index f4297901..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/ConsentManageUtil.java +++ /dev/null @@ -1,664 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.util; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionUtils; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentServiceUtil; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import com.wso2.openbanking.accelerator.consent.extensions.manage.model.ConsentManageData; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import net.minidev.json.JSONObject; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.time.LocalDateTime; -import java.time.OffsetDateTime; -import java.time.ZoneOffset; -import java.time.format.DateTimeFormatter; -import java.time.format.DateTimeParseException; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.EnumSet; -import java.util.regex.Pattern; - -/** - * Consent manage util class for accelerator. - */ -public class ConsentManageUtil { - private static final Log log = LogFactory.getLog(ConsentManageUtil.class); - private static final OpenBankingConfigParser parser = OpenBankingConfigParser.getInstance(); - - /** - * Check whether valid Data object is provided. - * - * @param initiationRequestbody Data object in initiation payload - * @return whether the Data object is valid - */ - public static JSONObject validateInitiationDataBody(JSONObject initiationRequestbody) { - JSONObject validationResponse = new JSONObject(); - - if (!initiationRequestbody.containsKey(ConsentExtensionConstants.DATA) || !(initiationRequestbody. - get(ConsentExtensionConstants.DATA) - instanceof JSONObject) || ((JSONObject) initiationRequestbody.get(ConsentExtensionConstants.DATA)) - .isEmpty()) { - log.error(ErrorConstants.PAYLOAD_FORMAT_ERROR); - return ConsentManageUtil.getValidationResponse(ErrorConstants.RESOURCE_INVALID_FORMAT, - ErrorConstants.PAYLOAD_FORMAT_ERROR, ErrorConstants.PATH_REQUEST_BODY); - } - - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } - - /** - * Method to construct the consent manage validation response. - * - * @param errorCode Error Code - * @param errorMessage Error Message - * @param errorPath Error Path - * @return JSONObject Validation response - */ - public static JSONObject getValidationResponse(String errorCode, String errorMessage, String errorPath) { - JSONObject validationResponse = new JSONObject(); - - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - validationResponse.put(ConsentExtensionConstants.HTTP_CODE, ResponseStatus.BAD_REQUEST); - validationResponse.put(ConsentExtensionConstants.ERRORS, errorMessage); - return validationResponse; - } - - /** - * Method to construct the consent manage validation response for vrp. - * - * @param errorMessage Error Message - * - * @return JSONObject Validation response - */ - public static JSONObject getValidationResponse(String errorMessage) { - JSONObject validationResponse = new JSONObject(); - - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - validationResponse.put(ConsentExtensionConstants.HTTP_CODE, ResponseStatus.BAD_REQUEST); - validationResponse.put(ConsentExtensionConstants.ERRORS, errorMessage); - return validationResponse; - } - - /** - * Method to validate debtor account. - * - * @param debtorAccount Debtor Account object - * @return JSONObject Validation response - */ - public static JSONObject validateDebtorAccount(JSONObject debtorAccount) { - - JSONObject validationResponse = new JSONObject(); - //Check Debtor Account Scheme name exists - if (!debtorAccount.containsKey(ConsentExtensionConstants.SCHEME_NAME) || - StringUtils.isEmpty(debtorAccount.getAsString(ConsentExtensionConstants.SCHEME_NAME))) { - log.error(ErrorConstants.MISSING_DEBTOR_ACC_SCHEME_NAME); - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - validationResponse.put(ConsentExtensionConstants.HTTP_CODE, ResponseStatus.BAD_REQUEST); - validationResponse.put(ConsentExtensionConstants.ERRORS, ErrorConstants.MISSING_DEBTOR_ACC_SCHEME_NAME); - - return validationResponse; - } - - //Validate Debtor Account Scheme name Length - if (debtorAccount.containsKey(ConsentExtensionConstants.SCHEME_NAME) && - !ConsentManageUtil.validateDebtorAccSchemeNameLength(debtorAccount - .getAsString(ConsentExtensionConstants.SCHEME_NAME))) { - log.error(ErrorConstants.INVALID_DEBTOR_ACC_SCHEME_NAME_LENGTH); - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - validationResponse.put(ConsentExtensionConstants.HTTP_CODE, ResponseStatus.BAD_REQUEST); - validationResponse.put(ConsentExtensionConstants.ERRORS, - ErrorConstants.INVALID_DEBTOR_ACC_SCHEME_NAME_LENGTH); - - return validationResponse; - } - - //Validate Debtor Account Scheme name - if (debtorAccount.containsKey(ConsentExtensionConstants.SCHEME_NAME) && - (!(debtorAccount.get(ConsentExtensionConstants.SCHEME_NAME) instanceof String) || - !ConsentManageUtil.isDebtorAccSchemeNameValid(debtorAccount - .getAsString(ConsentExtensionConstants.SCHEME_NAME)))) { - log.error(ErrorConstants.INVALID_DEBTOR_ACC_SCHEME_NAME); - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - validationResponse.put(ConsentExtensionConstants.HTTP_CODE, ResponseStatus.BAD_REQUEST); - validationResponse.put(ConsentExtensionConstants.ERRORS, ErrorConstants.INVALID_DEBTOR_ACC_SCHEME_NAME); - - return validationResponse; - } - - //Check Debtor Account Identification existing - if (!debtorAccount.containsKey(ConsentExtensionConstants.IDENTIFICATION) || - StringUtils.isEmpty(debtorAccount.getAsString(ConsentExtensionConstants.IDENTIFICATION))) { - log.error(ErrorConstants.MISSING_DEBTOR_ACC_IDENTIFICATION); - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - validationResponse.put(ConsentExtensionConstants.HTTP_CODE, ResponseStatus.BAD_REQUEST); - validationResponse.put(ConsentExtensionConstants.ERRORS, ErrorConstants.MISSING_DEBTOR_ACC_IDENTIFICATION); - - return validationResponse; - } - - //Validate Debtor Account Identification - if (debtorAccount.containsKey(ConsentExtensionConstants.IDENTIFICATION) && - (!(debtorAccount.get(ConsentExtensionConstants.IDENTIFICATION) instanceof String) || - !ConsentManageUtil.isDebtorAccIdentificationValid(debtorAccount - .getAsString(ConsentExtensionConstants.IDENTIFICATION)))) { - log.error(ErrorConstants.INVALID_DEBTOR_ACC_IDENTIFICATION); - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - validationResponse.put(ConsentExtensionConstants.HTTP_CODE, ResponseStatus.BAD_REQUEST); - validationResponse.put(ConsentExtensionConstants.ERRORS, ErrorConstants.INVALID_DEBTOR_ACC_IDENTIFICATION); - - return validationResponse; - } - - //Validate Debtor Account Name - if (debtorAccount.containsKey(ConsentExtensionConstants.NAME) && - (!(debtorAccount.get(ConsentExtensionConstants.NAME) instanceof String) || - !ConsentManageUtil.isDebtorAccNameValid(debtorAccount - .getAsString(ConsentExtensionConstants.NAME)))) { - log.error(ErrorConstants.INVALID_DEBTOR_ACC_NAME); - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - validationResponse.put(ConsentExtensionConstants.HTTP_CODE, ResponseStatus.BAD_REQUEST); - validationResponse.put(ConsentExtensionConstants.ERRORS, ErrorConstants.INVALID_DEBTOR_ACC_NAME); - - return validationResponse; - } - - //Validate Debtor Account Secondary Identification - if (debtorAccount.containsKey(ConsentExtensionConstants.SECONDARY_IDENTIFICATION) && - (!(debtorAccount.get(ConsentExtensionConstants.SECONDARY_IDENTIFICATION) instanceof String) || - !ConsentManageUtil.isDebtorAccSecondaryIdentificationValid(debtorAccount - .getAsString(ConsentExtensionConstants.SECONDARY_IDENTIFICATION)))) { - log.error(ErrorConstants.INVALID_DEBTOR_ACC_SEC_IDENTIFICATION); - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - validationResponse.put(ConsentExtensionConstants.HTTP_CODE, ResponseStatus.BAD_REQUEST); - validationResponse.put(ConsentExtensionConstants.ERRORS, - ErrorConstants.INVALID_DEBTOR_ACC_SEC_IDENTIFICATION); - - return validationResponse; - } - - //Validate Sort Code number scheme - String schemeName = debtorAccount.getAsString(ConsentExtensionConstants.SCHEME_NAME); - String identification = debtorAccount.getAsString(ConsentExtensionConstants.IDENTIFICATION); - if (!checkSortCodeSchemeNameAndIdentificationValidity(schemeName, identification)) { - log.error(ErrorConstants.INVALID_IDENTIFICATION); - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - validationResponse.put(ConsentExtensionConstants.HTTP_CODE, ResponseStatus.BAD_REQUEST); - validationResponse.put(ConsentExtensionConstants.ERRORS, ErrorConstants.INVALID_IDENTIFICATION); - - return validationResponse; - } - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - - return validationResponse; - } - - /** - * Validate creditor account. - * - * @param creditorAccount Creditor Account object - * @return JSONObject Validation response - */ - public static JSONObject validateCreditorAccount(JSONObject creditorAccount) { - - JSONObject validationResponse = new JSONObject(); - //Check Creditor Account Scheme name exists - if (!creditorAccount.containsKey(ConsentExtensionConstants.SCHEME_NAME) || - StringUtils.isEmpty(creditorAccount.getAsString(ConsentExtensionConstants.SCHEME_NAME))) { - log.error(ErrorConstants.MISSING_CREDITOR_ACC_SCHEME_NAME); - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - validationResponse.put(ConsentExtensionConstants.HTTP_CODE, ResponseStatus.BAD_REQUEST); - validationResponse.put(ConsentExtensionConstants.ERRORS, ErrorConstants.MISSING_CREDITOR_ACC_SCHEME_NAME); - return validationResponse; - } - - //Validate Creditor Account Scheme name Length - if (creditorAccount.containsKey(ConsentExtensionConstants.SCHEME_NAME) && - !ConsentManageUtil.validateDebtorAccSchemeNameLength(creditorAccount - .getAsString(ConsentExtensionConstants.SCHEME_NAME))) { - log.error(ErrorConstants.INVALID_CREDITOR_ACC_SCHEME_NAME_LENGTH); - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - validationResponse.put(ConsentExtensionConstants.HTTP_CODE, ResponseStatus.BAD_REQUEST); - validationResponse.put(ConsentExtensionConstants.ERRORS, - ErrorConstants.INVALID_CREDITOR_ACC_SCHEME_NAME_LENGTH); - return validationResponse; - } - - //Validate Creditor Account Scheme name - if (creditorAccount.containsKey(ConsentExtensionConstants.SCHEME_NAME) && - (!(creditorAccount.get(ConsentExtensionConstants.SCHEME_NAME) instanceof String) || - !ConsentManageUtil.isDebtorAccSchemeNameValid(creditorAccount - .getAsString(ConsentExtensionConstants.SCHEME_NAME)))) { - log.error(ErrorConstants.INVALID_CREDITOR_ACC_SCHEME_NAME); - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - validationResponse.put(ConsentExtensionConstants.HTTP_CODE, ResponseStatus.BAD_REQUEST); - validationResponse.put(ConsentExtensionConstants.ERRORS, ErrorConstants.INVALID_CREDITOR_ACC_SCHEME_NAME); - return validationResponse; - } - - //Check Creditor Account Identification existing - if (!creditorAccount.containsKey(ConsentExtensionConstants.IDENTIFICATION) || - StringUtils.isEmpty(creditorAccount.getAsString(ConsentExtensionConstants.IDENTIFICATION))) { - log.error(ErrorConstants.MISSING_CREDITOR_ACC_IDENTIFICATION); - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - validationResponse.put(ConsentExtensionConstants.HTTP_CODE, ResponseStatus.BAD_REQUEST); - validationResponse.put(ConsentExtensionConstants.ERRORS, - ErrorConstants.MISSING_CREDITOR_ACC_IDENTIFICATION); - return validationResponse; - } - - //Validate Creditor Account Identification - if (creditorAccount.containsKey(ConsentExtensionConstants.IDENTIFICATION) && - (!(creditorAccount.get(ConsentExtensionConstants.IDENTIFICATION) instanceof String) || - !ConsentManageUtil.isDebtorAccIdentificationValid(creditorAccount - .getAsString(ConsentExtensionConstants.IDENTIFICATION)))) { - log.error(ErrorConstants.INVALID_CREDITOR_ACC_IDENTIFICATION); - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - validationResponse.put(ConsentExtensionConstants.HTTP_CODE, ResponseStatus.BAD_REQUEST); - validationResponse.put(ConsentExtensionConstants.ERRORS, - ErrorConstants.INVALID_CREDITOR_ACC_IDENTIFICATION); - return validationResponse; - } - - //Validate Creditor Account Name - if (creditorAccount.containsKey(ConsentExtensionConstants.NAME) && - (!(creditorAccount.get(ConsentExtensionConstants.NAME) instanceof String) || - !ConsentManageUtil.isDebtorAccNameValid(creditorAccount - .getAsString(ConsentExtensionConstants.NAME)))) { - log.error(ErrorConstants.INVALID_CREDITOR_ACC_NAME); - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - validationResponse.put(ConsentExtensionConstants.HTTP_CODE, ResponseStatus.BAD_REQUEST); - validationResponse.put(ConsentExtensionConstants.ERRORS, ErrorConstants.INVALID_CREDITOR_ACC_NAME); - return validationResponse; - } - - //Validate Creditor Account Secondary Identification - if (creditorAccount.containsKey(ConsentExtensionConstants.SECONDARY_IDENTIFICATION) && - (!(creditorAccount.get(ConsentExtensionConstants.SECONDARY_IDENTIFICATION) instanceof String) || - !ConsentManageUtil.isDebtorAccSecondaryIdentificationValid(creditorAccount - .getAsString(ConsentExtensionConstants.SECONDARY_IDENTIFICATION)))) { - log.error(ErrorConstants.INVALID_CREDITOR_ACC_SEC_IDENTIFICATION); - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - validationResponse.put(ConsentExtensionConstants.HTTP_CODE, ResponseStatus.BAD_REQUEST); - validationResponse.put(ConsentExtensionConstants.ERRORS, - ErrorConstants.INVALID_CREDITOR_ACC_SEC_IDENTIFICATION); - return validationResponse; - } - - //Validate Sort Code number scheme - String schemeName = creditorAccount.getAsString(ConsentExtensionConstants.SCHEME_NAME); - String identification = creditorAccount.getAsString(ConsentExtensionConstants.IDENTIFICATION); - if (!checkSortCodeSchemeNameAndIdentificationValidity(schemeName, identification)) { - log.error(ErrorConstants.INVALID_IDENTIFICATION); - validationResponse.put(ConsentExtensionConstants.IS_VALID, false); - validationResponse.put(ConsentExtensionConstants.HTTP_CODE, ResponseStatus.BAD_REQUEST); - validationResponse.put(ConsentExtensionConstants.ERRORS, ErrorConstants.INVALID_IDENTIFICATION); - return validationResponse; - } - - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } - - - /** - * Method to handle the Payment/cof Consent Delete requests. - * - * @param consentManageData Object containing request details - */ - public static void handleConsentManageDelete(ConsentManageData consentManageData) { - - String consentId = consentManageData.getRequestPath().split("/")[1]; - Boolean shouldRevokeTokens; - if (ConsentManageUtil.isConsentIdValid(consentId)) { - try { - ConsentResource consentResource = ConsentServiceUtil.getConsentService() - .getConsent(consentId, false); - - if (!consentResource.getClientID().equals(consentManageData.getClientId())) { - //Throwing this error in a generic manner since client will not be able to identify if consent - // exists if consent does not belong to them - throw new ConsentException(ResponseStatus.BAD_REQUEST, - ErrorConstants.NO_CONSENT_FOR_CLIENT_ERROR); - } - - if (ConsentExtensionConstants.REVOKED_STATUS.equals(consentResource.getCurrentStatus())) { - throw new ConsentException(ResponseStatus.BAD_REQUEST, - "Consent already in revoked state"); - } - - //Revoke tokens related to the consent if the flag 'shouldRevokeTokens' is true. - shouldRevokeTokens = ConsentExtensionConstants.AUTHORIZED_STATUS - .equals(consentResource.getCurrentStatus()); - - boolean success = ConsentExtensionsDataHolder.getInstance().getConsentCoreService() - .revokeConsent(consentId, ConsentExtensionConstants.REVOKED_STATUS, null, - shouldRevokeTokens); - if (!success) { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - "Token revocation unsuccessful"); - } - consentManageData.setResponseStatus(ResponseStatus.NO_CONTENT); - } catch (ConsentManagementException e) { - log.error(e.getMessage()); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, e.getMessage()); - } - } else { - throw new ConsentException(ResponseStatus.BAD_REQUEST, "Consent ID invalid"); - } - } - - /** - * Utility class to check whether the Debtor Account Scheme name length. - * - * @param debtorAccSchemeName Debtor Account Scheme Name - * @return boolean Whether the Debtor Account Scheme name length is valid - */ - public static boolean validateDebtorAccSchemeNameLength(String debtorAccSchemeName) { - if (log.isDebugEnabled()) { - log.debug("debtorAccSchemeName: " + debtorAccSchemeName); - } - - return (debtorAccSchemeName.length() <= 256); - } - - /** - * Utility class to check whether the Debtor Account Scheme name matches with Enum values. - * - * @param debtorAccSchemeName Debtor Account Scheme Name - * @return boolean Whether the Debtor Account Scheme name is valid - */ - public static boolean isDebtorAccSchemeNameValid(String debtorAccSchemeName) { - if (log.isDebugEnabled()) { - log.debug("debtorAccSchemeName: " + debtorAccSchemeName); - } - - EnumSet set = EnumSet.allOf(DebtorAccountSchemeNameEnum.class); - boolean result = set.contains(DebtorAccountSchemeNameEnum.fromValue(debtorAccSchemeName)); - if (log.isDebugEnabled()) { - log.debug("Result: " + result); - } - return result; - } - - /** - * Utility class to check whether the Debtor Account Identification is valid. - * - * @param debtorAccIdentification Debtor Account Identification - * @return boolean Whether the Debtor Account Identification is valid - */ - public static boolean isDebtorAccIdentificationValid(String debtorAccIdentification) { - if (log.isDebugEnabled()) { - log.debug("debtorAccIdentification: " + debtorAccIdentification); - } - - return (debtorAccIdentification.length() <= 256); - } - - /** - * Utility class to check whether the Debtor Account Name is valid. - * - * @param debtorAccName Debtor Account Name - * @return boolean Whether the Debtor Account Name is valid - */ - public static boolean isDebtorAccNameValid(String debtorAccName) { - if (log.isDebugEnabled()) { - log.debug("debtorAccName: " + debtorAccName); - } - - return (debtorAccName.length() <= 350); - } - - /** - * Utility class to check whether the Debtor AccountSecondary Identification is valid. - * - * @param debtorAccSecondaryIdentification Debtor Account Secondary Identification - * @return boolean Whether the Debtor Account Secondary Identification is valid - */ - public static boolean isDebtorAccSecondaryIdentificationValid(String debtorAccSecondaryIdentification) { - if (log.isDebugEnabled()) { - log.debug("debtorAccSecondaryIdentification: " + debtorAccSecondaryIdentification); - } - - return (debtorAccSecondaryIdentification.length() <= 34); - } - - /** - * Utility class to check whether the SortCode SchemeName and Identification is valid. - * - * @param schemeName Scheme name - * @param identification Identification - * @return - */ - private static boolean checkSortCodeSchemeNameAndIdentificationValidity(String schemeName, String identification) { - - boolean isValid = true; - if ((ConsentExtensionConstants.OB_SORT_CODE_ACCOUNT_NUMBER.equals(schemeName) - || ConsentExtensionConstants.SORT_CODE_ACCOUNT_NUMBER.equals(schemeName)) && - (StringUtils.isNotEmpty(identification) && - !(identification.length() == ConsentExtensionConstants.ACCOUNT_IDENTIFICATION_LENGTH && - identification.matches(ConsentExtensionConstants.SORT_CODE_PATTERN)))) { - isValid = false; - } - return isValid; - } - - /** - * Check whether the local instrument is supported. - * - * @param localInstrument Local Instrument value to validate - * @return Whether the local instrument is valid - */ - public static boolean validateLocalInstrument(String localInstrument) { - ArrayList defaultLocalInstrumentList = new ArrayList<>(Arrays.asList( - "OB.BACS", - "OB.BalanceTransfer", "OB.CHAPS", "OB.Euro1", "OB.FPS", "OB.Link", - "OB.MoneyTransfer", "OB.Paym", "OB.SEPACreditTransfer", - "OB.SEPAInstantCreditTransfer", "OB.SWIFT", "OB.Target2")); - - String customValues = (String) parser.getConfiguration().get( - ConsentExtensionConstants.CUSTOM_LOCAL_INSTRUMENT_VALUES); - if (customValues != null) { - - String[] customLocalInstrumentList = customValues.split("\\|"); - defaultLocalInstrumentList.addAll(Arrays.asList(customLocalInstrumentList)); - } - return defaultLocalInstrumentList.contains(localInstrument); - - } - - /** - * Check whether the amount is higher that the max instructed amount allowed by the bank. - * - * @param instructedAmount Instructed Amount to validate - * @return Whether the instructed amount is valid - */ - public static boolean validateMaxInstructedAmount(String instructedAmount) { - //This is a mandatory configuration in open-banking.xml. Hence can't be null. - String maxInstructedAmount = (String) parser.getConfiguration().get( - ConsentExtensionConstants.MAX_INSTRUCTED_AMOUNT); - return Double.parseDouble(instructedAmount) <= Double.parseDouble(maxInstructedAmount); - - } - - /** - * Method to construct Initiation response. - * - * @param response Response of the request - * @param createdConsent Consent response received from service layer - * @param consentManageData Request Details received - * @param type ConsentType - * @return JSONObject Initiation Response - */ - public static JSONObject getInitiationResponse(JSONObject response, DetailedConsentResource createdConsent, - ConsentManageData consentManageData, String type) { - JSONObject dataObject = (JSONObject) response.get(ConsentExtensionConstants.DATA); - dataObject.appendField(ConsentExtensionConstants.CONSENT_ID, createdConsent.getConsentID()); - dataObject.appendField("CreationDateTime", convertEpochDateTime(createdConsent.getCreatedTime())); - dataObject.appendField("StatusUpdateDateTime", convertEpochDateTime(createdConsent.getUpdatedTime())); - dataObject.appendField(ConsentExtensionConstants.STATUS, - ConsentExtensionUtils.getConsentStatus(createdConsent.getCurrentStatus())); - if (type.equals(ConsentExtensionConstants.PAYMENTS) && - ConsentExtensionUtils.isRequestAcceptedPastElapsedTime()) { - dataObject.appendField(ConsentExtensionConstants.CUT_OFF_DATE_TIME, ConsentExtensionUtils - .constructDateTime(0L, (String) parser.getConfiguration() - .get(OpenBankingConstants.DAILY_CUTOFF))); - } - - //add self link - JSONObject links = new JSONObject(); - links.put(ConsentExtensionConstants.SELF, - constructSelfLink(createdConsent.getConsentID(), consentManageData, type)); - response.appendField(ConsentExtensionConstants.LINKS, links); - - response.appendField(ConsentExtensionConstants.META, new JSONObject()); - - response.remove(ConsentExtensionConstants.DATA); - response.appendField(ConsentExtensionConstants.DATA, dataObject); - - return response; - } - - - /** - * Method to construct Retrieval Initiation response. - * - * @param receiptJSON Initiation of the request - * @param consent Consent response received from service layer - * @param consentManageData Request Details received - * @param type ConsentType - * @return JSONObject Initiation Response - */ - public static JSONObject getInitiationRetrievalResponse(JSONObject receiptJSON, ConsentResource consent, - ConsentManageData consentManageData, String type) { - - JSONObject dataObject = (JSONObject) receiptJSON.get(ConsentExtensionConstants.DATA); - dataObject.appendField(ConsentExtensionConstants.CONSENT_ID, consent.getConsentID()); - dataObject.appendField(ConsentExtensionConstants.STATUS, consent.getCurrentStatus()); - dataObject.appendField(ConsentExtensionConstants.STATUS_UPDATE_TIME, - ConsentExtensionUtils.convertToISO8601(consent.getUpdatedTime())); - dataObject.appendField(ConsentExtensionConstants.CREATION_TIME, - ConsentExtensionUtils.convertToISO8601(consent.getCreatedTime())); - - receiptJSON.remove(ConsentExtensionConstants.DATA); - receiptJSON.appendField(ConsentExtensionConstants.DATA, dataObject); - - JSONObject links = new JSONObject(); - links.put(ConsentExtensionConstants.SELF, - constructSelfLink(consent.getConsentID(), consentManageData, type)); - receiptJSON.appendField(ConsentExtensionConstants.LINKS, links); - - receiptJSON.appendField(ConsentExtensionConstants.META, new JSONObject()); - - return receiptJSON; - } - - private static String convertEpochDateTime(long epochTime) { - - int nanoOfSecond = 0; - ZoneOffset offset = ZoneOffset.UTC; - LocalDateTime ldt = LocalDateTime.ofEpochSecond(epochTime, nanoOfSecond, offset); - return DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ss'Z'").format(ldt); - } - - /** - * Method to construct the self link. - * - * @param consentId Consent ID - * @param consentManageData Request Details recieved - * @param type ConsentType - * @return Constructed Self Link - */ - public static String constructSelfLink(String consentId, ConsentManageData consentManageData, String type) { - - String baseUrl = ""; - if (ConsentExtensionConstants.ACCOUNTS.equals(type)) { - baseUrl = (String) parser.getConfiguration().get( - ConsentExtensionConstants.ACCOUNTS_SELF_LINK); - } else if (ConsentExtensionConstants.PAYMENTS.equals(type)) { - baseUrl = (String) parser.getConfiguration().get( - ConsentExtensionConstants.PAYMENT_SELF_LINK); - } else if (ConsentExtensionConstants.FUNDSCONFIRMATIONS.equals(type)) { - baseUrl = (String) parser.getConfiguration().get( - ConsentExtensionConstants.COF_SELF_LINK); - } else if (ConsentExtensionConstants.VRP.equals(type)) { - baseUrl = (String) parser.getConfiguration().get( - ConsentExtensionConstants.VRP_SELF_LINK); - } - - String requestPath = consentManageData.getRequestPath(); - return baseUrl.replaceFirst("\\{version}", "3.1") + requestPath + "/" + consentId; - } - - /** - * Validate the consent ID. - * - * @param consentId Consent Id to validate - * @return Whether the consent ID is valid - */ - public static boolean isConsentIdValid(String consentId) { - return (consentId.length() == 36 && Pattern.matches(ConsentExtensionConstants.UUID_REGEX, consentId)); - } - - /** - * Validate Expiration Date Time. - * - * @param expDateVal Expiration Date Time - * @return Whether the expiration date time is valid - */ - public static boolean isConsentExpirationTimeValid(String expDateVal) { - if (expDateVal == null) { - return true; - } - try { - OffsetDateTime expDate = OffsetDateTime.parse(expDateVal); - OffsetDateTime currDate = OffsetDateTime.now(expDate.getOffset()); - - return expDate.compareTo(currDate) > 0; - } catch (DateTimeParseException e) { - return false; - } - } - /** - * Validate whether the date is a valid ISO 8601 format. - * @param dateValue Date value to validate - * @return Whether the date is a valid ISO 8601 format - */ - public static boolean isValid8601(String dateValue) { - try { - OffsetDateTime.parse(dateValue); - return true; - } catch (DateTimeParseException e) { - return false; - } - } - - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/DebtorAccountSchemeNameEnum.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/DebtorAccountSchemeNameEnum.java deleted file mode 100644 index daccab47..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/DebtorAccountSchemeNameEnum.java +++ /dev/null @@ -1,61 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.util; - -import java.util.Arrays; -import java.util.List; -import java.util.Optional; - -/** - * Specifies the Schema Names of Debtor Account. - */ -public enum DebtorAccountSchemeNameEnum { - - BBAN("OB.BBAN"), - - IBAN("OB.IBAN"), - - PAN("OB.PAN"), - - PAYM("OB.Paym"), - - SORT_CODE_NUMBER("OB.SortCodeAccountNumber"); - - private String value; - - DebtorAccountSchemeNameEnum(String value) { - this.value = value; - } - - @Override - public String toString() { - return String.valueOf(value); - } - - public static DebtorAccountSchemeNameEnum fromValue(String text) { - - List accountList = Arrays.asList(DebtorAccountSchemeNameEnum.values()); - Optional accountOpt = accountList - .stream() - .filter(i -> String.valueOf(i.value).equals(text)) - .findAny(); - - return accountOpt.isPresent() ? accountOpt.get() : null; - } - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/PaymentPayloadValidator.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/PaymentPayloadValidator.java deleted file mode 100644 index 03e4782a..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/PaymentPayloadValidator.java +++ /dev/null @@ -1,97 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.util; - -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import net.minidev.json.JSONObject; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -/** - * Validator class to validate payment initiation payload. - */ -public class PaymentPayloadValidator { - - private static final Log log = LogFactory.getLog(PaymentPayloadValidator.class); - - /** - * Method to validate payment initiation payload. - * - * @param requestPath Request Path of the request - * @param initiation Initiation Object of the request - * @return JSONObject Validation Response - */ - public static JSONObject validatePaymentInitiationPayload(String requestPath, JSONObject initiation) { - - JSONObject validationResponse = new JSONObject(); - - //Validate DebtorAccount - if (initiation.containsKey(ConsentExtensionConstants.DEBTOR_ACC)) { - JSONObject debtorAccount = (JSONObject) initiation.get(ConsentExtensionConstants.DEBTOR_ACC); - JSONObject validationResult = ConsentManageUtil.validateDebtorAccount(debtorAccount); - if (!(boolean) validationResult.get(ConsentExtensionConstants.IS_VALID)) { - return validationResult; - } - } - - //Validate CreditorAccount - if (initiation.containsKey(ConsentExtensionConstants.CREDITOR_ACC)) { - JSONObject creditorAccount = (JSONObject) initiation.get(ConsentExtensionConstants.CREDITOR_ACC); - JSONObject validationResult = ConsentManageUtil.validateCreditorAccount(creditorAccount); - - if (!(boolean) validationResult.get(ConsentExtensionConstants.IS_VALID)) { - return validationResult; - } - } else { - if (!requestPath.contains(ConsentExtensionConstants.PAYMENTS)) { - log.error(ErrorConstants.MSG_MISSING_CREDITOR_ACC); - return ConsentManageUtil.getValidationResponse(ErrorConstants.FIELD_MISSING, - ErrorConstants.MSG_MISSING_CREDITOR_ACC, ErrorConstants.PATH_CREDIT_ACCOUNT); - } - } - - //Validate Local Instrument - if (initiation.containsKey(ConsentExtensionConstants.LOCAL_INSTRUMENT) && !ConsentManageUtil - .validateLocalInstrument(initiation.getAsString(ConsentExtensionConstants.LOCAL_INSTRUMENT))) { - log.error(ErrorConstants.INVALID_LOCAL_INSTRUMENT); - return ConsentManageUtil.getValidationResponse(ErrorConstants.UNSUPPORTED_LOCAL_INSTRUMENTS, - ErrorConstants.INVALID_LOCAL_INSTRUMENT, ErrorConstants.PATH_LOCAL_INSTRUMENT); - } - - - if (!requestPath.contains(ConsentExtensionConstants.PAYMENTS)) { - JSONObject instructedAmount = (JSONObject) initiation.get(ConsentExtensionConstants.INSTRUCTED_AMOUNT); - if (Double.parseDouble(instructedAmount.getAsString(ConsentExtensionConstants.AMOUNT)) < 1) { - log.error(ErrorConstants.INVALID_INSTRUCTED_AMOUNT); - return ConsentManageUtil.getValidationResponse(ErrorConstants.FIELD_INVALID, - ErrorConstants.INVALID_INSTRUCTED_AMOUNT, ErrorConstants.PATH_INSTRUCTED_AMOUNT); - } - - if (!ConsentManageUtil - .validateMaxInstructedAmount(instructedAmount.getAsString(ConsentExtensionConstants.AMOUNT))) { - log.error(ErrorConstants.MAX_INSTRUCTED_AMOUNT); - return ConsentManageUtil.getValidationResponse(ErrorConstants.FIELD_INVALID, - ErrorConstants.MAX_INSTRUCTED_AMOUNT, ErrorConstants.PATH_INSTRUCTED_AMOUNT); - } - - } - validationResponse.put(ConsentExtensionConstants.IS_VALID, true); - return validationResponse; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/PeriodicTypesEnum.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/PeriodicTypesEnum.java deleted file mode 100644 index 5ec0d221..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/PeriodicTypesEnum.java +++ /dev/null @@ -1,76 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - *

- * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.util; - -import java.time.LocalDate; - -/** - * This enum represents the different types of periods that can be used in the application. - * Each enum value is associated with a string representation and a method to calculate the divisor based - * on the period type. - * The divisor is used to convert other time units to this period type. - */ -public enum PeriodicTypesEnum { - - DAY("Day"), - - WEEK("Week"), - - FORTNIGHT("Fortnight"), - - MONTH("Month"), - - HALF_YEAR("Half-Year"), - - YEAR("Year"); - - private String value; - - PeriodicTypesEnum(String value) { - this.value = value; - } - - @Override - public String toString() { - return String.valueOf(value); - } - - /** - * Returns the divisor based on the period type. - * - * @return the divisor based on the period type - */ - public int getDivisor() { - switch (this) { - case DAY: - return 1; - case WEEK: - return 7; - case FORTNIGHT: - return 14; - case MONTH: - return LocalDate.now().lengthOfMonth(); - case HALF_YEAR: - return LocalDate.now().isLeapYear() ? 181 : 180; - case YEAR: - return LocalDate.now().isLeapYear() ? 366 : 365; - default: - throw new IllegalArgumentException("Invalid PeriodType"); - } - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/PeriodicalConsentJobActivator.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/PeriodicalConsentJobActivator.java deleted file mode 100644 index 2852b1ff..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/PeriodicalConsentJobActivator.java +++ /dev/null @@ -1,106 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.util; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.consent.extensions.util.jobs.ExpiredConsentStatusUpdateJob; -import com.wso2.openbanking.accelerator.consent.extensions.util.jobs.RetentionDatabaseSyncJob; -import com.wso2.openbanking.accelerator.consent.extensions.util.scheduler.PeriodicalConsentJobScheduler; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.quartz.CronScheduleBuilder; -import org.quartz.JobDetail; -import org.quartz.Scheduler; -import org.quartz.SchedulerException; -import org.quartz.Trigger; - -import static org.quartz.JobBuilder.newJob; -import static org.quartz.TriggerBuilder.newTrigger; - -/** - * Scheduled Task definition and trigger to perform expired consent status updateJob based on the cron string. - */ -public class PeriodicalConsentJobActivator { - - private static Log log = LogFactory.getLog(PeriodicalConsentJobActivator.class); - - /** - * activate the scheduler task. - */ - public void activate() { - - if (OpenBankingConfigParser.getInstance().isConsentExpirationPeriodicalJobEnabled()) { - JobDetail job = newJob(ExpiredConsentStatusUpdateJob.class) - .withIdentity("ConsentStatusUpdateJob", "group1") - .build(); - - Trigger trigger = newTrigger() - .withIdentity("periodicalTrigger", "group1") - .withSchedule(CronScheduleBuilder.cronSchedule( - OpenBankingConfigParser.getInstance().getConsentExpiryCronExpression())) - .build(); - - try { - Scheduler scheduler = PeriodicalConsentJobScheduler.getInstance().getScheduler(); - // this check is to remove already stored jobs in clustered mode. - if (scheduler.checkExists(job.getKey())) { - scheduler.deleteJob(job.getKey()); - } - - scheduler.scheduleJob(job, trigger); - if (log.isDebugEnabled()) { - log.debug("Periodical Consent Status Updater Started with cron : " - + OpenBankingConfigParser.getInstance().getConsentExpiryCronExpression()); - } - } catch (SchedulerException e) { - log.error("Error while creating and starting Periodical Consent Status Update Scheduled Task", e); - } - } - - if (OpenBankingConfigParser.getInstance().isRetentionDataDBSyncEnabled() && - OpenBankingConfigParser.getInstance().isConsentDataRetentionEnabled()) { - JobDetail job = newJob(RetentionDatabaseSyncJob.class) - .withIdentity("RetentionDatabaseSyncJob", "group1") - .build(); - - Trigger trigger = newTrigger() - .withIdentity("RetentionDatabaseSyncPeriodicalTrigger", "group1") - .withSchedule(CronScheduleBuilder.cronSchedule( - OpenBankingConfigParser.getInstance().getRetentionDataDBSyncCronExpression())) - .build(); - - try { - Scheduler scheduler = PeriodicalConsentJobScheduler.getInstance().getScheduler(); - // this check is to remove already stored jobs in clustered mode. - if (scheduler.checkExists(job.getKey())) { - scheduler.deleteJob(job.getKey()); - } - - scheduler.scheduleJob(job, trigger); - if (log.isDebugEnabled()) { - log.debug("Retention database sync job started with cron : " - + OpenBankingConfigParser.getInstance().getConsentExpiryCronExpression()); - } - } catch (SchedulerException e) { - log.error("Error while creating and starting retention database syncing scheduled Task", e); - } - } - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/jobs/ExpiredConsentStatusUpdateJob.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/jobs/ExpiredConsentStatusUpdateJob.java deleted file mode 100644 index fde76d8a..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/jobs/ExpiredConsentStatusUpdateJob.java +++ /dev/null @@ -1,158 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.util.jobs; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import com.wso2.openbanking.accelerator.consent.mgt.dao.constants.ConsentMgtDAOConstants; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentHistoryResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.ConsentCoreService; -import com.wso2.openbanking.accelerator.consent.mgt.service.constants.ConsentCoreServiceConstants; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.quartz.DisallowConcurrentExecution; -import org.quartz.Job; -import org.quartz.JobExecutionContext; -import org.quartz.JobExecutionException; - -import java.time.Instant; -import java.util.ArrayList; -import java.util.Map; - -/** - * Scheduled Task to read and update expired consents in the DB - * 1) Read the consents which has a expiry time attribute from the DB. - * 2) Check if expired, and collect expired consents - * 3) Update the expired statues in DB - * 4) Notify state change to relevant handler. - */ -@DisallowConcurrentExecution -public class ExpiredConsentStatusUpdateJob implements Job { - - private static Log log = LogFactory.getLog(ExpiredConsentStatusUpdateJob.class); - private static final String expiredConsentStatus = - OpenBankingConfigParser.getInstance().getStatusWordingForExpiredConsents(); - private static final String expirationEligibleConsentStatuses = - OpenBankingConfigParser.getInstance().getEligibleStatusesForConsentExpiry(); - - /** - * Method used to enforce periodic statues update of consents. - * - * @param jobExecutionContext Job Execution Context - * @throws JobExecutionException if an error occurs while executing the job - */ - public void execute(JobExecutionContext jobExecutionContext) throws JobExecutionException { - try { - updateExpiredStatues(); - } catch (ConsentManagementException e) { - log.error("Error occurred while updating status for expired consents", e); - } - - } - - /** - * Method to update statues of consents. - * @throws ConsentManagementException if an error occurs while updating the consent status - */ - public static void updateExpiredStatues() throws ConsentManagementException { - - log.debug("Expired Consent Status Update Scheduled Task is executing."); - // get consents which has a expiry time attribute - ArrayList consentsEligibleForExpiration = - ConsentExtensionsDataHolder.getInstance().getConsentCoreService() - .getConsentsEligibleForExpiration(expirationEligibleConsentStatuses); - // filter out expired consents and change the status of expired consents - for (DetailedConsentResource consentResource : consentsEligibleForExpiration) { - if (isExpired(consentResource)) { - String updatedConsentId = updateConsentExpiredStatus(consentResource); - if (log.isDebugEnabled()) { - log.debug("Expired status updated for consent : " + updatedConsentId); - } - } - } - log.debug("Expired Consent Status Update Scheduled Task is finished."); - } - - /** - * Check if the consents is expired based on the consent attribute value. - * - * @param detailedConsentResource - * @return - */ - private static boolean isExpired(DetailedConsentResource detailedConsentResource) { - - Map consentAttributes = detailedConsentResource.getConsentAttributes(); - if (consentAttributes.containsKey(ConsentMgtDAOConstants.CONSENT_EXPIRY_TIME_ATTRIBUTE)) { - // Read the UTC expiry timestamp in long - long expiryTimestamp = Long.parseLong( - consentAttributes.get(ConsentMgtDAOConstants.CONSENT_EXPIRY_TIME_ATTRIBUTE)); - // Compare with current UTC timestamp in long - Instant instant = Instant.now(); - long currentTimeStampSeconds = instant.getEpochSecond(); - if (currentTimeStampSeconds >= expiryTimestamp) { - log.info("Consent " + detailedConsentResource.getConsentID() + " is identified as expired based on the " - + "given consent expiration time : " + expiryTimestamp); - return true; - } - } - return false; - } - - /** - * Update the expired consents in DB. - * - * @param detailedConsentResource - * @return - */ - private static String updateConsentExpiredStatus(DetailedConsentResource detailedConsentResource) { - - try { - ConsentExtensionsDataHolder.getInstance().getConsentCoreService() - .updateConsentStatus(detailedConsentResource.getConsentID(), expiredConsentStatus); - - //since the consent status is changed during the consent expiration, the previous status will be saved - //in the consent history to properly back-track the previous status held in the consent - storeConsentStateChangeInConsentHistory(detailedConsentResource); - - } catch (ConsentManagementException e) { - log.error("Error occurred while updating status for consentId : " + - detailedConsentResource.getConsentID(), e); - } - return detailedConsentResource.getConsentID(); - } - - private static void storeConsentStateChangeInConsentHistory(DetailedConsentResource detailedConsentResource) - throws ConsentManagementException { - - if (OpenBankingConfigParser.getInstance().isConsentAmendmentHistoryEnabled()) { - - ConsentCoreService consentCoreService = ConsentExtensionsDataHolder.getInstance().getConsentCoreService(); - ConsentHistoryResource consentHistoryResource = new ConsentHistoryResource(); - consentHistoryResource.setTimestamp(System.currentTimeMillis() / 1000); - consentHistoryResource.setReason(ConsentCoreServiceConstants.AMENDMENT_REASON_CONSENT_EXPIRATION); - consentHistoryResource.setDetailedConsentResource(detailedConsentResource); - - consentCoreService.storeConsentAmendmentHistory(detailedConsentResource.getConsentID(), - consentHistoryResource, null); - } - } - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/jobs/RetentionDatabaseSyncJob.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/jobs/RetentionDatabaseSyncJob.java deleted file mode 100644 index 8ea9068e..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/jobs/RetentionDatabaseSyncJob.java +++ /dev/null @@ -1,68 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.util.jobs; - -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.quartz.DisallowConcurrentExecution; -import org.quartz.Job; -import org.quartz.JobExecutionContext; -import org.quartz.JobExecutionException; - -/** - * Scheduled Task to read and sync the temporary retention data in consent tables to retention database. - * 1) Read the consents in temporary retention tables in consennt DB. - * 2) Insert each consent data to retention database. - * 3) Delete the consent data from temporary retention tables. - */ -@DisallowConcurrentExecution -public class RetentionDatabaseSyncJob implements Job { - - private static Log log = LogFactory.getLog(RetentionDatabaseSyncJob.class); - - /** - * Method used to enforce sync the temporary retention data. - * - * @param jobExecutionContext Job Execution Context - * @throws JobExecutionException if an error occurs while executing the job - */ - public void execute(JobExecutionContext jobExecutionContext) throws JobExecutionException { - - try { - syncRetentionDatabase(); - } catch (ConsentManagementException e) { - log.error("Error occurred while retention database syncing", e); - } - } - - /** - * Method to sync the temporary retention data. - * @throws ConsentManagementException if an error occurs while syncing the retention database - */ - public static void syncRetentionDatabase() throws ConsentManagementException { - - log.debug("Retention database syncing scheduled task is executing."); - ConsentExtensionsDataHolder.getInstance().getConsentCoreService().syncRetentionDatabaseWithPurgedConsent(); - log.debug("Retention database syncing scheduled task is finished."); - } - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/scheduler/PeriodicalConsentJobScheduler.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/scheduler/PeriodicalConsentJobScheduler.java deleted file mode 100644 index 3131c3d9..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/util/scheduler/PeriodicalConsentJobScheduler.java +++ /dev/null @@ -1,102 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.util.scheduler; - -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.quartz.Scheduler; -import org.quartz.SchedulerException; -import org.quartz.impl.StdSchedulerFactory; -import org.wso2.carbon.utils.CarbonUtils; - -import java.io.File; -import java.nio.file.Paths; - -/** - * Periodic consent job scheduler class. - * This class initialize the scheduler and schedule configured jobs and triggers. - */ -public class PeriodicalConsentJobScheduler { - - private static volatile PeriodicalConsentJobScheduler instance; - private static final String QUARTZ_PROPERTY_FILE = "quartz.properties"; - - private static volatile Scheduler scheduler; - private static Log log = LogFactory.getLog(PeriodicalConsentJobScheduler.class); - - private PeriodicalConsentJobScheduler() { - - initScheduler(); - } - - /** - * Get an instance of the PeriodicalConsentJobScheduler. It implements a double checked locking initialization. - * - * @return PeriodicalConsentJobScheduler instance - */ - public static synchronized PeriodicalConsentJobScheduler getInstance() { - - if (instance == null) { - synchronized (PeriodicalConsentJobScheduler.class) { - if (instance == null) { - instance = new PeriodicalConsentJobScheduler(); - } - } - } - return instance; - } - - /** - * Initialize the scheduler. - */ - private void initScheduler() { - - if (instance != null) { - return; - } - synchronized (PeriodicalConsentJobScheduler.class) { - try { - String quartzConfigFile = Paths.get(CarbonUtils.getCarbonConfigDirPath()).toString() + "/" - + QUARTZ_PROPERTY_FILE; - boolean exists = new File(quartzConfigFile).exists(); - if (exists) { - StdSchedulerFactory stdSchedulerFactory = new StdSchedulerFactory(); - stdSchedulerFactory.initialize(quartzConfigFile); - scheduler = stdSchedulerFactory.getScheduler(); - } else { - scheduler = StdSchedulerFactory.getDefaultScheduler(); - } - scheduler.start(); - } catch (SchedulerException e) { - log.error("Exception while initializing the scheduler", e); - } - } - } - - /** - * Returns the scheduler. - * - * @return Scheduler scheduler. - */ - public Scheduler getScheduler() { - - return scheduler; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/builder/ConsentValidateBuilder.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/builder/ConsentValidateBuilder.java deleted file mode 100644 index de6b065f..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/builder/ConsentValidateBuilder.java +++ /dev/null @@ -1,57 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.validate.builder; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.common.util.OpenBankingUtils; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import com.wso2.openbanking.accelerator.consent.extensions.validate.model.ConsentValidator; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -/** - * Builder class for consent validator. - */ -public class ConsentValidateBuilder { - - private static final Log log = LogFactory.getLog(ConsentValidateBuilder.class); - private ConsentValidator consentValidator = null; - private String requestSignatureAlias = null; - private static String validatorConfigPath = "Consent.Validation.Validator"; - private static String signatureAliasConfigPath = "Consent.Validation.RequestSignatureAlias"; - - - public void build() { - - OpenBankingConfigurationService configurationService = - ConsentExtensionsDataHolder.getInstance().getOpenBankingConfigurationService(); - String handlerConfig = (String) configurationService.getConfigurations().get(validatorConfigPath); - consentValidator = (ConsentValidator) OpenBankingUtils.getClassInstanceFromFQN(handlerConfig); - requestSignatureAlias = (String) configurationService.getConfigurations().get(signatureAliasConfigPath); - log.debug("Admin handler loaded successfully"); - } - - public ConsentValidator getConsentValidator() { - return consentValidator; - } - - public String getRequestSignatureAlias() { - return requestSignatureAlias; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/impl/DefaultConsentValidator.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/impl/DefaultConsentValidator.java deleted file mode 100644 index 0692ec13..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/impl/DefaultConsentValidator.java +++ /dev/null @@ -1,508 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - *

- * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.validate.impl; - -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionUtils; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.consent.extensions.validate.model.ConsentValidateData; -import com.wso2.openbanking.accelerator.consent.extensions.validate.model.ConsentValidationResult; -import com.wso2.openbanking.accelerator.consent.extensions.validate.model.ConsentValidator; -import com.wso2.openbanking.accelerator.consent.extensions.validate.util.ConsentValidatorUtil; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.HttpStatus; - -import java.time.OffsetDateTime; -import java.time.format.DateTimeParseException; -import java.util.ArrayList; - -/** - * Consent validator default implementation. - */ -public class DefaultConsentValidator implements ConsentValidator { - - private static final Log log = LogFactory.getLog(DefaultConsentValidator.class); - private static final String ACCOUNTS_REGEX = "/accounts/[^/?]*"; - private static final String TRANSACTIONS_REGEX = "/accounts/[^/?]*/transactions"; - private static final String BALANCES_REGEX = "/accounts/[^/?]*/balances"; - private static final String PERMISSION_MISMATCH_ERROR = "Permission mismatch. Consent does not contain necessary " + - "permissions"; - private static final String INVALID_URI_ERROR = "Path requested is invalid"; - private static final String CONSENT_EXPIRED_ERROR = "Provided consent is expired"; - private static final String CONSENT_STATE_ERROR = "Provided consent not in authorised state"; - private static final String AUTHORISED_STATUS = "authorised"; - - @Override - public void validate(ConsentValidateData consentValidateData, ConsentValidationResult consentValidationResult) - throws ConsentException { - - String uri = consentValidateData.getRequestPath(); - JSONObject receiptJSON; - try { - receiptJSON = (JSONObject) (new JSONParser(JSONParser.MODE_PERMISSIVE)). - parse(consentValidateData.getComprehensiveConsent().getReceipt()); - - } catch (ParseException e) { - log.error(e.getMessage()); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Exception occurred while validating" + - " permissions"); - } - - //User Validation - String userIdFromToken = consentValidateData.getUserId(); - boolean userIdMatching = false; - ArrayList authResources = consentValidateData.getComprehensiveConsent() - .getAuthorizationResources(); - for (AuthorizationResource resource : authResources) { - if (userIdFromToken.contains(resource.getUserID())) { - userIdMatching = true; - break; - } - } - - if (!userIdMatching) { - log.error(ErrorConstants.INVALID_USER_ID); - consentValidationResult.setErrorMessage(ErrorConstants.INVALID_USER_ID); - consentValidationResult.setErrorCode(ErrorConstants.RESOURCE_CONSENT_MISMATCH); - consentValidationResult.setHttpCode(HttpStatus.SC_BAD_REQUEST); - return; - } - - String clientIdFromToken = consentValidateData.getClientId(); - String clientIdFromConsent = consentValidateData.getComprehensiveConsent().getClientID(); - if (clientIdFromToken == null || clientIdFromConsent == null || - !clientIdFromToken.equals(clientIdFromConsent)) { - log.error(ErrorConstants.MSG_INVALID_CLIENT_ID); - consentValidationResult.setErrorMessage(ErrorConstants.MSG_INVALID_CLIENT_ID); - consentValidationResult.setErrorCode(ErrorConstants.RESOURCE_CONSENT_MISMATCH); - consentValidationResult.setHttpCode(HttpStatus.SC_FORBIDDEN); - return; - } - - String requestType = consentValidateData.getComprehensiveConsent().getConsentType(); - - switch (requestType) { - case ConsentExtensionConstants.ACCOUNTS: - validateAccountSubmission(consentValidateData, receiptJSON, consentValidationResult); - break; - case ConsentExtensionConstants.PAYMENTS: - validatePaymentSubmission(consentValidateData, receiptJSON, consentValidationResult); - break; - case ConsentExtensionConstants.FUNDSCONFIRMATIONS: - validateFundsConfirmationSubmission(consentValidateData, receiptJSON, consentValidationResult); - break; - case ConsentExtensionConstants.VRP: - validateVRPSubmission(consentValidateData, receiptJSON, consentValidationResult); - break; - default: - log.error(ErrorConstants.INVALID_CONSENT_TYPE); - consentValidationResult.setErrorMessage(ErrorConstants.INVALID_CONSENT_TYPE); - consentValidationResult.setErrorCode(ErrorConstants.UNEXPECTED_ERROR); - consentValidationResult.setHttpCode(HttpStatus.SC_INTERNAL_SERVER_ERROR); - return; - } - } - - /** - * Validate Account Retrieval Request. - * - * @param consentValidateData Object with request data - * @param consentValidationResult Validation result object to return - */ - private void validateAccountSubmission(ConsentValidateData consentValidateData, JSONObject receiptJSON, - ConsentValidationResult consentValidationResult) { - - JSONArray permissions = (JSONArray) ((JSONObject) receiptJSON.get("Data")).get("Permissions"); - - // Perform URI Validation. - String uri = consentValidateData.getRequestPath(); - if (!(uri.matches(ACCOUNTS_REGEX) || uri.matches(TRANSACTIONS_REGEX) || uri.matches(BALANCES_REGEX))) { - consentValidationResult.setErrorMessage(INVALID_URI_ERROR); - consentValidationResult.setErrorCode("00013"); - consentValidationResult.setHttpCode(401); - return; - } - if ((uri.matches(ACCOUNTS_REGEX) && !permissions.contains("ReadAccountsDetail")) || - (uri.matches(TRANSACTIONS_REGEX) && !permissions.contains("ReadTransactionsDetail")) || - (uri.matches(BALANCES_REGEX)) && !permissions.contains("ReadBalances")) { - consentValidationResult.setErrorMessage(PERMISSION_MISMATCH_ERROR); - consentValidationResult.setErrorCode("00010"); - consentValidationResult.setHttpCode(401); - return; - } - - //Consent Status Validation - if (!ConsentExtensionConstants.AUTHORIZED_STATUS - .equalsIgnoreCase(consentValidateData.getComprehensiveConsent().getCurrentStatus())) { - consentValidationResult.setErrorMessage(ErrorConstants.ACCOUNT_CONSENT_STATE_INVALID); - consentValidationResult.setErrorCode(ErrorConstants.RESOURCE_INVALID_CONSENT_STATUS); - consentValidationResult.setHttpCode(HttpStatus.SC_BAD_REQUEST); - return; - } - - if (isConsentExpired(((JSONObject) receiptJSON.get("Data")).getAsString("ExpirationDateTime"))) { - consentValidationResult.setErrorMessage(CONSENT_EXPIRED_ERROR); - consentValidationResult.setErrorCode("00011"); - consentValidationResult.setHttpCode(401); - return; - } - consentValidationResult.setValid(true); - } - - - /** - * Validate Payment Retrieval Request. - * - * @param consentValidateData Object with request data - * @param consentValidationResult Validation result object to return - */ - private void validatePaymentSubmission(ConsentValidateData consentValidateData, JSONObject initiationJson, - ConsentValidationResult consentValidationResult) { - - DetailedConsentResource detailedConsentResource = consentValidateData.getComprehensiveConsent(); - - try { - // Rejecting consent if cut off time is elapsed and the policy is REJECT - // Updating the consent status to "Reject" if the above condition is true - if (ConsentExtensionUtils.shouldSubmissionRequestBeRejected(ConsentExtensionUtils - .convertToISO8601(detailedConsentResource.getCreatedTime()))) { - boolean success = ConsentExtensionUtils.getConsentService().revokeConsent( - detailedConsentResource.getConsentID(), ConsentExtensionConstants.REJECTED_STATUS); - if (!success) { - log.error(ErrorConstants.TOKEN_REVOKE_ERROR); - consentValidationResult.setErrorMessage(ErrorConstants.TOKEN_REVOKE_ERROR); - consentValidationResult.setErrorCode(ErrorConstants.UNEXPECTED_ERROR); - consentValidationResult.setHttpCode(HttpStatus.SC_INTERNAL_SERVER_ERROR); - return; - } - log.error(ErrorConstants.CUT_OFF_DATE_ELAPSED); - consentValidationResult.setErrorMessage(ErrorConstants.CUT_OFF_DATE_ELAPSED); - consentValidationResult.setErrorCode(ErrorConstants.RULES_CUTOFF); - consentValidationResult.setHttpCode(HttpStatus.SC_BAD_REQUEST); - return; - } - - // Check if requested consent ID matches to initiation consent ID. - if (consentValidateData.getConsentId() == null || detailedConsentResource.getConsentID() == null || - !consentValidateData.getConsentId().equals(detailedConsentResource.getConsentID())) { - log.error(ErrorConstants.MSG_INVALID_CONSENT_ID); - consentValidationResult.setErrorMessage(ErrorConstants.MSG_INVALID_CONSENT_ID); - consentValidationResult.setErrorCode(ErrorConstants.RESOURCE_CONSENT_MISMATCH); - consentValidationResult.setHttpCode(HttpStatus.SC_BAD_REQUEST); - return; - } - - if (consentValidateData.getRequestPath().contains(ConsentExtensionConstants.PAYMENT_COF_PATH)) { - PaymentFundsConfirmationPayloadValidator paymentFundsConfirmationValidator = - new PaymentFundsConfirmationPayloadValidator(); - paymentFundsConfirmationValidator.validatePaymentFundsConfirmationRequest(consentValidateData, - consentValidationResult, detailedConsentResource); - return; - } else { - if (!ConsentExtensionConstants.AUTHORIZED_STATUS - .equalsIgnoreCase(consentValidateData.getComprehensiveConsent().getCurrentStatus())) { - log.error(ErrorConstants.PAYMENT_CONSENT_STATE_INVALID); - consentValidationResult.setErrorMessage(ErrorConstants.PAYMENT_CONSENT_STATE_INVALID); - consentValidationResult.setErrorCode(ErrorConstants.RESOURCE_INVALID_CONSENT_STATUS); - consentValidationResult.setHttpCode(HttpStatus.SC_BAD_REQUEST); - return; - } - - JSONObject submissionJson = consentValidateData.getPayload(); - JSONObject submissionData = new JSONObject(); - JSONObject submissionInitiation = new JSONObject(); - - JSONObject requestInitiation = (JSONObject) ((JSONObject) initiationJson - .get(ConsentExtensionConstants.DATA)).get(ConsentExtensionConstants.INITIATION); - - if (submissionJson.containsKey(ConsentExtensionConstants.DATA) && - submissionJson.get(ConsentExtensionConstants.DATA) instanceof JSONObject) { - submissionData = (JSONObject) submissionJson.get(ConsentExtensionConstants.DATA); - } else { - log.error(ErrorConstants.DATA_NOT_FOUND); - consentValidationResult.setErrorMessage(ErrorConstants.DATA_NOT_FOUND); - consentValidationResult.setErrorCode(ErrorConstants.FIELD_MISSING); - consentValidationResult.setHttpCode(HttpStatus.SC_BAD_REQUEST); - return; - } - - if (submissionData.containsKey(ConsentExtensionConstants.INITIATION) && - submissionData.get(ConsentExtensionConstants.INITIATION) instanceof JSONObject) { - submissionInitiation = (JSONObject) submissionData.get(ConsentExtensionConstants.INITIATION); - } else { - log.error(ErrorConstants.INITIATION_NOT_FOUND); - consentValidationResult.setErrorMessage(ErrorConstants.INITIATION_NOT_FOUND); - consentValidationResult.setErrorCode(ErrorConstants.FIELD_MISSING); - consentValidationResult.setHttpCode(HttpStatus.SC_BAD_REQUEST); - return; - } - - // Check if requested consent ID in the body to initiation consent ID. - if (!submissionData.containsKey(ConsentExtensionConstants.CONSENT_ID_VALIDATION) || - submissionData.get(ConsentExtensionConstants.CONSENT_ID_VALIDATION) == null || - !submissionData.get(ConsentExtensionConstants.CONSENT_ID_VALIDATION) - .equals(detailedConsentResource.getConsentID())) { - log.error(ErrorConstants.MSG_INVALID_CONSENT_ID); - consentValidationResult.setErrorMessage(ErrorConstants.MSG_INVALID_CONSENT_ID); - consentValidationResult.setErrorCode(ErrorConstants.RESOURCE_CONSENT_MISMATCH); - consentValidationResult.setHttpCode(HttpStatus.SC_BAD_REQUEST); - return; - } - - PaymentSubmissionPayloadValidator validator = new PaymentSubmissionPayloadValidator(); - JSONObject initiationValidationResult = validator - .validateInitiation(submissionInitiation, requestInitiation); - - if (!(boolean) initiationValidationResult.get(ConsentExtensionConstants.IS_VALID_PAYLOAD)) { - log.error(initiationValidationResult.getAsString(ConsentExtensionConstants.ERROR_MESSAGE)); - consentValidationResult.setErrorMessage(initiationValidationResult - .getAsString(ConsentExtensionConstants.ERROR_MESSAGE)); - consentValidationResult.setErrorCode(initiationValidationResult - .getAsString(ConsentExtensionConstants.ERROR_CODE)); - consentValidationResult.setHttpCode(HttpStatus.SC_BAD_REQUEST); - return; - } - } - - } catch (ConsentManagementException e) { - log.error(e.getMessage()); - consentValidationResult.setErrorMessage(e.getMessage()); - consentValidationResult.setErrorCode(ErrorConstants.UNEXPECTED_ERROR); - consentValidationResult.setHttpCode(HttpStatus.SC_INTERNAL_SERVER_ERROR); - return; - } - consentValidationResult.setValid(true); - } - - private boolean isConsentExpired(String expDateVal) throws ConsentException { - - if (expDateVal != null && !expDateVal.isEmpty()) { - try { - OffsetDateTime expDate = OffsetDateTime.parse(expDateVal); - return OffsetDateTime.now().isAfter(expDate); - } catch (DateTimeParseException e) { - log.error("Error occurred while parsing the expiration date : " + expDateVal); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - "Error occurred while parsing the expiration date"); - } - } else { - return false; - } - - } - - /** - * Validate Funds Confirmation Retrieval Request. - * - * @param consentValidateData Object with request data - * @param consentValidationResult Validation result object to return - */ - private static void validateFundsConfirmationSubmission(ConsentValidateData consentValidateData, - JSONObject receiptJSON, - ConsentValidationResult consentValidationResult) { - - // Perform URI Validation. - String uri = consentValidateData.getRequestPath(); - if (uri == null || !ConsentValidatorUtil.isCOFURIValid(uri)) { - consentValidationResult.setErrorMessage(ErrorConstants.INVALID_URI_ERROR); - consentValidationResult.setErrorCode(ErrorConstants.RESOURCE_INVALID_FORMAT); - consentValidationResult.setHttpCode(401); - return; - } - - //Consent Status Validation - if (!ConsentExtensionConstants.AUTHORIZED_STATUS - .equalsIgnoreCase(consentValidateData.getComprehensiveConsent().getCurrentStatus())) { - consentValidationResult.setErrorMessage(ErrorConstants.COF_CONSENT_STATE_INVALID); - consentValidationResult.setErrorCode(ErrorConstants.RESOURCE_INVALID_CONSENT_STATUS); - consentValidationResult.setHttpCode(400); - return; - } - - //Validate whether the consent is expired - if (ConsentValidatorUtil - .isConsentExpired(((JSONObject) receiptJSON.get(ConsentExtensionConstants.DATA)) - .getAsString(ConsentExtensionConstants.EXPIRATION_DATE))) { - consentValidationResult.setErrorMessage(ErrorConstants.CONSENT_EXPIRED_ERROR); - consentValidationResult.setErrorCode(ErrorConstants.FIELD_INVALID); - consentValidationResult.setHttpCode(400); - return; - } - - // Check if requested consent ID in the token to initiation consent ID. - if (consentValidateData.getConsentId() == null || - consentValidateData.getComprehensiveConsent().getConsentID() == null || - !consentValidateData.getConsentId() - .equals(consentValidateData.getComprehensiveConsent().getConsentID())) { - log.error(ErrorConstants.MSG_INVALID_CONSENT_ID); - consentValidationResult.setErrorMessage(ErrorConstants.MSG_INVALID_CONSENT_ID); - consentValidationResult.setErrorCode(ErrorConstants.RESOURCE_CONSENT_MISMATCH); - consentValidationResult.setHttpCode(HttpStatus.SC_BAD_REQUEST); - return; - } - - JSONObject data = (JSONObject) consentValidateData.getPayload().get(ConsentExtensionConstants.DATA); - // Check if requested consent ID in the body to initiation consent ID. - if (!data.containsKey(ConsentExtensionConstants.CONSENT_ID_VALIDATION) || - data.get(ConsentExtensionConstants.CONSENT_ID_VALIDATION) == null || - !data.get(ConsentExtensionConstants.CONSENT_ID_VALIDATION) - .equals(consentValidateData.getComprehensiveConsent().getConsentID())) { - log.error(ErrorConstants.MSG_INVALID_CONSENT_ID); - consentValidationResult.setErrorMessage(ErrorConstants.MSG_INVALID_CONSENT_ID); - consentValidationResult.setErrorCode(ErrorConstants.RESOURCE_CONSENT_MISMATCH); - consentValidationResult.setHttpCode(HttpStatus.SC_BAD_REQUEST); - return; - } - - consentValidationResult.setValid(true); - - } - - /** - * Validate VRP Submission Request. - * - * @param consentValidateData Object with request data - * @param consentValidationResult Validation result object to return - */ - private void validateVRPSubmission(ConsentValidateData consentValidateData, JSONObject initiationJson, - ConsentValidationResult consentValidationResult) { - - DetailedConsentResource detailedConsentResource = consentValidateData.getComprehensiveConsent(); - - if (!ConsentExtensionConstants.AUTHORIZED_STATUS - .equals(consentValidateData.getComprehensiveConsent().getCurrentStatus())) { - log.error(ErrorConstants.VRP_CONSENT_STATUS_INVALID); - consentValidationResult.setErrorMessage(ErrorConstants.VRP_CONSENT_STATUS_INVALID); - consentValidationResult.setErrorCode(ErrorConstants.RESOURCE_INVALID_CONSENT_STATUS); - consentValidationResult.setHttpCode(HttpStatus.SC_BAD_REQUEST); - return; - } - - // Check if requested consent ID matches to initiation consent ID. - if (consentValidateData.getConsentId() == null || detailedConsentResource.getConsentID() == null || - !consentValidateData.getConsentId().equals(detailedConsentResource.getConsentID())) { - log.error(ErrorConstants.MSG_INVALID_CONSENT_ID); - consentValidationResult.setErrorMessage(ErrorConstants.MSG_INVALID_CONSENT_ID); - consentValidationResult.setErrorCode(ErrorConstants.RESOURCE_CONSENT_MISMATCH); - consentValidationResult.setHttpCode(HttpStatus.SC_BAD_REQUEST); - return; - } - - JSONObject submissionJson = consentValidateData.getPayload(); - - JSONObject dataValidationResults = VRPSubmissionPayloadValidator.validateSubmissionData(submissionJson); - if (!Boolean.parseBoolean(dataValidationResults. - getAsString(ConsentExtensionConstants.IS_VALID_PAYLOAD))) { - ConsentValidatorUtil.setErrorMessageForConsentValidationResult(dataValidationResults, - consentValidationResult); - return; - } - - JSONObject submissionData = (JSONObject) submissionJson.get(ConsentExtensionConstants.DATA); - - JSONObject initiationParameterValidationResults = VRPSubmissionPayloadValidator. - validateInitiationParameter(submissionData); - if (!Boolean.parseBoolean(initiationParameterValidationResults. - getAsString(ConsentExtensionConstants.IS_VALID_PAYLOAD))) { - ConsentValidatorUtil.setErrorMessageForConsentValidationResult(initiationParameterValidationResults, - consentValidationResult); - return; - } - - JSONObject instructionParameterValidationResults = VRPSubmissionPayloadValidator. - validateInstructionParameter(submissionData); - if (!Boolean.parseBoolean(instructionParameterValidationResults. - getAsString(ConsentExtensionConstants.IS_VALID_PAYLOAD))) { - ConsentValidatorUtil.setErrorMessageForConsentValidationResult(instructionParameterValidationResults, - consentValidationResult); - return; - } - - // Check if requested consent ID in the body to initiation consent ID. - if (!submissionData.containsKey(ConsentExtensionConstants.CONSENT_ID) || - !(submissionData.get(ConsentExtensionConstants.CONSENT_ID) instanceof String) || - !submissionData.get(ConsentExtensionConstants.CONSENT_ID) - .equals(detailedConsentResource.getConsentID())) { - log.error(ErrorConstants.INVALID_REQUEST_CONSENT_ID); - consentValidationResult.setErrorMessage(ErrorConstants.INVALID_REQUEST_CONSENT_ID); - consentValidationResult.setErrorCode(ErrorConstants.RESOURCE_CONSENT_MISMATCH); - consentValidationResult.setHttpCode(HttpStatus.SC_BAD_REQUEST); - return; - } - - JSONObject dataObject = (JSONObject) initiationJson.get(ConsentExtensionConstants.DATA); - JSONObject requestInitiation = (JSONObject) dataObject.get(ConsentExtensionConstants.INITIATION); - JSONObject submissionInitiation = (JSONObject) submissionData.get(ConsentExtensionConstants.INITIATION); - JSONObject submissionInstruction = (JSONObject) submissionData.get(ConsentExtensionConstants.INSTRUCTION); - - JSONObject initiationValidationResult = VRPSubmissionPayloadValidator - .validateInitiation(submissionInitiation, requestInitiation); - - if (!Boolean.parseBoolean(initiationValidationResult. - getAsString(ConsentExtensionConstants.IS_VALID_PAYLOAD))) { - ConsentValidatorUtil.setErrorMessageForConsentValidationResult(initiationValidationResult, - consentValidationResult); - return; - } - - // Here the requestInitiation is passed as a parameter inorder to compare the creditor account in - // the initiation payload present under the initiation parameter, with the submission payload present under the - // instruction parameter. - JSONObject instructionValidationResult = VRPSubmissionPayloadValidator. - validateInstruction(submissionInstruction, requestInitiation); - - if (!Boolean.parseBoolean(instructionValidationResult. - getAsString(ConsentExtensionConstants.IS_VALID_PAYLOAD))) { - ConsentValidatorUtil.setErrorMessageForConsentValidationResult(instructionValidationResult, - consentValidationResult); - return; - } - - JSONObject riskParameterValidationResults = VRPSubmissionPayloadValidator.validateRiskParameter(submissionJson); - if (!Boolean.parseBoolean(riskParameterValidationResults. - getAsString(ConsentExtensionConstants.IS_VALID_PAYLOAD))) { - ConsentValidatorUtil.setErrorMessageForConsentValidationResult(riskParameterValidationResults, - consentValidationResult); - return; - } - - JSONObject initiationRisk = (JSONObject) initiationJson.get(ConsentExtensionConstants.RISK); - JSONObject submissionRisk = (JSONObject) submissionJson.get(ConsentExtensionConstants.RISK); - JSONObject riskValidationResult = VRPSubmissionPayloadValidator.validateRisk(submissionRisk, - initiationRisk); - - if (!Boolean.parseBoolean(riskValidationResult. - getAsString(ConsentExtensionConstants.IS_VALID_PAYLOAD))) { - ConsentValidatorUtil.setErrorMessageForConsentValidationResult(riskValidationResult, - consentValidationResult); - return; - } - consentValidationResult.setValid(true); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/impl/PaymentFundsConfirmationPayloadValidator.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/impl/PaymentFundsConfirmationPayloadValidator.java deleted file mode 100644 index 968a7181..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/impl/PaymentFundsConfirmationPayloadValidator.java +++ /dev/null @@ -1,84 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.validate.impl; - -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.validate.model.ConsentValidateData; -import com.wso2.openbanking.accelerator.consent.extensions.validate.model.ConsentValidationResult; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.HttpStatus; - -/** - * Class for validating Payments funds confirmation requests. - */ -public class PaymentFundsConfirmationPayloadValidator { - - private static Log log = LogFactory.getLog(PaymentFundsConfirmationPayloadValidator.class); - - /** - * MEthod to validate Payment Funds Confirmation requests. - * - * @param consentValidateData Object with request data - * @param consentValidationResult Validation result object to return - * @param detailedConsentResource detailed consent resource retrieved from database - */ - public void validatePaymentFundsConfirmationRequest(ConsentValidateData consentValidateData, - ConsentValidationResult consentValidationResult, - DetailedConsentResource detailedConsentResource) { - - //Consent Status Validation - if (!ConsentExtensionConstants.AUTHORIZED_STATUS - .equalsIgnoreCase(consentValidateData.getComprehensiveConsent().getCurrentStatus())) { - consentValidationResult.setErrorMessage(ErrorConstants.PAYMENT_CONSENT_STATE_INVALID); - consentValidationResult.setErrorCode(ErrorConstants.RESOURCE_INVALID_CONSENT_STATUS); - consentValidationResult.setHttpCode(HttpStatus.SC_BAD_REQUEST); - return; - } - - //Validate Consent Id From path - //ResourcePath comes in format /pisp/domestic-scheduled-consents/{consentId}/funds-confirmation - String[] requestPathArray = ((String) consentValidateData.getResourceParams().get("ResourcePath")) - .trim().split("/"); - - if (requestPathArray.length != 5 || StringUtils.isEmpty(requestPathArray[3])) { - log.error(ErrorConstants.CONSENT_ID_NOT_FOUND); - consentValidationResult.setErrorMessage(ErrorConstants.CONSENT_ID_NOT_FOUND); - consentValidationResult.setErrorCode(ErrorConstants.RESOURCE_CONSENT_MISMATCH); - consentValidationResult.setHttpCode(HttpStatus.SC_BAD_REQUEST); - return; - } - - //Validate whether consentId from path matches - String consentIdFromPath = requestPathArray[3]; - if (consentIdFromPath == null || !consentIdFromPath.equals(detailedConsentResource.getConsentID())) { - log.error(ErrorConstants.MSG_INVALID_CONSENT_ID); - consentValidationResult.setErrorMessage(ErrorConstants.MSG_INVALID_CONSENT_ID); - consentValidationResult.setErrorCode(ErrorConstants.RESOURCE_CONSENT_MISMATCH); - consentValidationResult.setHttpCode(HttpStatus.SC_BAD_REQUEST); - return; - } - - consentValidationResult.setValid(true); - } -} - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/impl/PaymentSubmissionPayloadValidator.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/impl/PaymentSubmissionPayloadValidator.java deleted file mode 100644 index 421a13b3..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/impl/PaymentSubmissionPayloadValidator.java +++ /dev/null @@ -1,122 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.validate.impl; - -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.validate.util.ConsentValidatorUtil; -import com.wso2.openbanking.accelerator.consent.extensions.validate.util.PaymentSubmissionValidationUtil; -import net.minidev.json.JSONObject; - -/** - * Class for validating Payment submission requests. - */ -public class PaymentSubmissionPayloadValidator { - - /** - * Method to validate payment submission initiation payload. - * - * @param submission Submission Request - * @param initiation Initiation Request - * @return JSONObject Validation Response - */ - public JSONObject validateInitiation(JSONObject submission, JSONObject initiation) { - - JSONObject validationResult = new JSONObject(); - validationResult.put(ConsentExtensionConstants.IS_VALID_PAYLOAD, true); - validationResult.put(ConsentExtensionConstants.ERROR_CODE, ""); - validationResult.put(ConsentExtensionConstants.ERROR_MESSAGE, ""); - - if (submission != null && initiation != null) { - - //Validate Instruction Identification - JSONObject instructionIdentificationResult = PaymentSubmissionValidationUtil - .validateInstructionIdentification(submission, initiation); - if (!((boolean) instructionIdentificationResult.get(ConsentExtensionConstants.IS_VALID_PAYLOAD))) { - return instructionIdentificationResult; - } - - //Validate End to End Identification - JSONObject endToEndIdentificationResult = PaymentSubmissionValidationUtil - .validateEndToEndIdentification(submission, initiation); - if (!((boolean) endToEndIdentificationResult.get(ConsentExtensionConstants.IS_VALID_PAYLOAD))) { - return endToEndIdentificationResult; - } - - //Validate Instructed Amount - JSONObject instructedAmountResult = PaymentSubmissionValidationUtil - .validateInstructedAmount(submission, initiation); - if (!((boolean) instructedAmountResult.get(ConsentExtensionConstants.IS_VALID_PAYLOAD))) { - return instructedAmountResult; - } - - - //Validate Creditor Account - if (submission.containsKey(ConsentExtensionConstants.CREDITOR_ACC) && - initiation.containsKey(ConsentExtensionConstants.CREDITOR_ACC)) { - - JSONObject subCreditorAccount = (JSONObject) submission.get(ConsentExtensionConstants.CREDITOR_ACC); - JSONObject initCreditorAccount = (JSONObject) initiation.get(ConsentExtensionConstants.CREDITOR_ACC); - - JSONObject creditorAccValidationResult = ConsentValidatorUtil.validateCreditorAcc(subCreditorAccount, - initCreditorAccount); - if (!(boolean) creditorAccValidationResult.get(ConsentExtensionConstants.IS_VALID_PAYLOAD)) { - return creditorAccValidationResult; - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.CREDITOR_ACC_NOT_FOUND); - } - - //Validate Debtor Account - if ((!submission.containsKey(ConsentExtensionConstants.DEBTOR_ACC) && - initiation.containsKey(ConsentExtensionConstants.DEBTOR_ACC)) || - (submission.containsKey(ConsentExtensionConstants.DEBTOR_ACC) && - !initiation.containsKey(ConsentExtensionConstants.DEBTOR_ACC))) { - - return ConsentValidatorUtil.getValidationResult(ErrorConstants.RESOURCE_CONSENT_MISMATCH, - ErrorConstants.DEBTOR_ACC_MISMATCH); - } else if (submission.containsKey(ConsentExtensionConstants.DEBTOR_ACC) && - initiation.containsKey(ConsentExtensionConstants.DEBTOR_ACC)) { - - JSONObject subDebtorAccount = (JSONObject) submission.get(ConsentExtensionConstants.DEBTOR_ACC); - JSONObject initDebtorAccount = (JSONObject) initiation.get(ConsentExtensionConstants.DEBTOR_ACC); - - JSONObject debtorAccValidationResult = ConsentValidatorUtil.validateDebtorAcc(subDebtorAccount, - initDebtorAccount); - if (!(boolean) debtorAccValidationResult.get(ConsentExtensionConstants.IS_VALID_PAYLOAD)) { - return debtorAccValidationResult; - } - } - - //Validate Local Instrument - if (!ConsentValidatorUtil.compareOptionalParameter( - submission.getAsString(ConsentExtensionConstants.LOCAL_INSTRUMENT), - initiation.getAsString(ConsentExtensionConstants.LOCAL_INSTRUMENT))) { - - return ConsentValidatorUtil.getValidationResult(ErrorConstants.RESOURCE_CONSENT_MISMATCH, - ErrorConstants.LOCAL_INSTRUMENT_MISMATCH); - } - } - - return validationResult; - - } -} - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/impl/VRPSubmissionPayloadValidator.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/impl/VRPSubmissionPayloadValidator.java deleted file mode 100644 index d6431f1d..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/impl/VRPSubmissionPayloadValidator.java +++ /dev/null @@ -1,516 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - *

- * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.validate.impl; - -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.validate.util.ConsentValidatorUtil; -import net.minidev.json.JSONObject; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -/** - * Class for validating VRP submission request. - */ -public class VRPSubmissionPayloadValidator { - private static final Log log = LogFactory.getLog(VRPSubmissionPayloadValidator.class); - - /** - * Validates the initiation parameters between the initiation of submission request and the initiation parameters - * of consent initiation request. - * - * @param initiationOfSubmission The initiation parameters from the submission request. - * @param initiationParameterOfConsentInitiation The initiation parameters from the consent initiation request. - * @return A JSONObject indicating the validation result. It contains a boolean value under the key - * ConsentExtensionConstants.IS_VALID_PAYLOAD, indicating whether the payload is valid. If the - * validation fails, it returns a JSONObject containing error details with keys defined in ErrorConstants. - */ - public static JSONObject validateInitiation(JSONObject initiationOfSubmission, - JSONObject initiationParameterOfConsentInitiation) { - - if (initiationOfSubmission != null && initiationParameterOfConsentInitiation != null) { - - - - //Validate Creditor Account - if ((!initiationOfSubmission.containsKey(ConsentExtensionConstants.CREDITOR_ACC) && - initiationParameterOfConsentInitiation.containsKey(ConsentExtensionConstants.CREDITOR_ACC)) || - (initiationOfSubmission.containsKey(ConsentExtensionConstants.CREDITOR_ACC) && - !initiationParameterOfConsentInitiation. - containsKey(ConsentExtensionConstants.CREDITOR_ACC))) { - - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.CREDITOR_ACC_NOT_FOUND); - } else if (initiationOfSubmission.containsKey(ConsentExtensionConstants.CREDITOR_ACC) && - initiationParameterOfConsentInitiation.containsKey(ConsentExtensionConstants.CREDITOR_ACC)) { - - Object submissionCreditorAccounts = initiationOfSubmission. - get(ConsentExtensionConstants.CREDITOR_ACC); - Object consentInitiationCreditorAccounts = initiationParameterOfConsentInitiation. - get(ConsentExtensionConstants.CREDITOR_ACC); - - if (submissionCreditorAccounts instanceof JSONObject && - consentInitiationCreditorAccounts instanceof JSONObject) { - JSONObject submissionCreditorAccount = (JSONObject) initiationOfSubmission. - get(ConsentExtensionConstants.CREDITOR_ACC); - JSONObject consentInitiationCreditorAccount = (JSONObject) - initiationParameterOfConsentInitiation.get(ConsentExtensionConstants.CREDITOR_ACC); - - JSONObject creditorAccValidationResult = ConsentValidatorUtil. - validateCreditorAcc(submissionCreditorAccount, consentInitiationCreditorAccount); - if (!Boolean.parseBoolean(creditorAccValidationResult. - getAsString(ConsentExtensionConstants.IS_VALID_PAYLOAD))) { - return creditorAccValidationResult; - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.INITIATION_CREDITOR_ACC_NOT_JSON_ERROR); - } - } - - //Validate Debtor Account - // This code if condition checks whether the debtor account parameter is present in both the request - // payloads (Initiation and the submission payloads) since both the payloads as to be equal. - if ((!initiationOfSubmission.containsKey(ConsentExtensionConstants.DEBTOR_ACC) && - initiationParameterOfConsentInitiation.containsKey(ConsentExtensionConstants.DEBTOR_ACC)) || - (initiationOfSubmission.containsKey(ConsentExtensionConstants.DEBTOR_ACC) && - !initiationParameterOfConsentInitiation. - containsKey(ConsentExtensionConstants.DEBTOR_ACC))) { - - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.DEBTOR_ACC_NOT_FOUND); - } else if (initiationOfSubmission.containsKey(ConsentExtensionConstants.DEBTOR_ACC) && - initiationParameterOfConsentInitiation.containsKey(ConsentExtensionConstants.DEBTOR_ACC)) { - - Object submissionDebtorAccounts = initiationOfSubmission - .get(ConsentExtensionConstants.DEBTOR_ACC); - Object consentInitiationDebtorAccounts = initiationParameterOfConsentInitiation - .get(ConsentExtensionConstants.DEBTOR_ACC); - - if (submissionDebtorAccounts instanceof JSONObject && - consentInitiationDebtorAccounts instanceof JSONObject) { - JSONObject submissionDebtorAccount = (JSONObject) initiationOfSubmission - .get(ConsentExtensionConstants.DEBTOR_ACC); - JSONObject consentInitiationDebtorAccount = (JSONObject) initiationParameterOfConsentInitiation - .get(ConsentExtensionConstants.DEBTOR_ACC); - - JSONObject debtorAccValidationResult = ConsentValidatorUtil. - validateDebtorAcc(submissionDebtorAccount, consentInitiationDebtorAccount); - if (!Boolean.parseBoolean(debtorAccValidationResult. - getAsString(ConsentExtensionConstants.IS_VALID_PAYLOAD))) { - return debtorAccValidationResult; - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_INVALID, - ErrorConstants.DEBTOR_ACC_NOT_JSON_ERROR); - } - } - - if ((!initiationOfSubmission.containsKey(ConsentExtensionConstants.REMITTANCE_INFO) - && initiationParameterOfConsentInitiation.containsKey(ConsentExtensionConstants.REMITTANCE_INFO)) || - (initiationOfSubmission.containsKey(ConsentExtensionConstants.REMITTANCE_INFO) - && !initiationParameterOfConsentInitiation. - containsKey(ConsentExtensionConstants.REMITTANCE_INFO))) { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.REMITTANCE_INFO_NOT_FOUND); - } else if (initiationOfSubmission.containsKey(ConsentExtensionConstants.REMITTANCE_INFO) - && initiationParameterOfConsentInitiation. - containsKey(ConsentExtensionConstants.REMITTANCE_INFO)) { - - Object remittanceInformationSubmission = initiationOfSubmission - .get(ConsentExtensionConstants.REMITTANCE_INFO); - Object remittanceInformationInitiation = initiationParameterOfConsentInitiation - .get(ConsentExtensionConstants.REMITTANCE_INFO); - - if (remittanceInformationSubmission instanceof JSONObject && - remittanceInformationInitiation instanceof JSONObject) { - JSONObject remittanceInformationSub = (JSONObject) initiationOfSubmission - .get(ConsentExtensionConstants.REMITTANCE_INFO); - JSONObject remittanceInformationInit = (JSONObject) initiationParameterOfConsentInitiation - .get(ConsentExtensionConstants.REMITTANCE_INFO); - - JSONObject validateRemittanceInfoResult = VRPSubmissionPayloadValidator.validateRemittanceInfo - (remittanceInformationSub, remittanceInformationInit); - if (!Boolean.parseBoolean(validateRemittanceInfoResult. - getAsString(ConsentExtensionConstants.IS_VALID_PAYLOAD))) { - return validateRemittanceInfoResult; - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_INVALID, - ErrorConstants.INITIATION_REMITTANCE_INFO_NOT_JSON_ERROR); - } - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.INVALID_PARAMETER); - } - - JSONObject validationResult = new JSONObject(); - validationResult.put(ConsentExtensionConstants.IS_VALID_PAYLOAD, true); - return validationResult; - } - - /** - * Validates the instruction between submission and initiation JSONObjects. - * - * @param submission The submission JSONObject from submission request. - * @param initiation The initiation JSONObject from initiation request, here we consider the initiation parameter - * since the creditor account from the initiation request need to be retrieved. - * @return A JSONObject indicating the validation result. It contains a boolean value under the key - * ConsentExtensionConstants.IS_VALID_PAYLOAD, indicating whether the payload is valid. If the - * validation fails, it returns a JSONObject containing error details with keys defined in ErrorConstants. - */ - public static JSONObject validateInstruction(JSONObject submission, - JSONObject initiation) { - - if (submission != null && initiation != null) { - - if (!submission.containsKey(ConsentExtensionConstants.INSTRUCTED_AMOUNT)) { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.INSTRUCTED_AMOUNT_NOT_FOUND); - } else { - Object instructedAmountObject = submission.get(ConsentExtensionConstants.INSTRUCTED_AMOUNT); - - if (isValidJSONObject(instructedAmountObject)) { - JSONObject instructedAmount = (JSONObject) instructedAmountObject; - if (!instructedAmount.containsKey(ConsentExtensionConstants.AMOUNT)) { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.INSTRUCTED_AMOUNT_AMOUNT_NOT_FOUND); - } else { - Object amountValue = instructedAmount.get(ConsentExtensionConstants.AMOUNT); - if (!isValidString(amountValue)) { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_INVALID, - ErrorConstants.INSTRUCTED_AMOUNT_NOT_STRING); - } - - if (!instructedAmount.containsKey(ConsentExtensionConstants.CURRENCY)) { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.INSTRUCTED_AMOUNT_CURRENCY_NOT_FOUND); - } else { - Object currencyValue = instructedAmount.get(ConsentExtensionConstants.CURRENCY); - if (!isValidString(currencyValue)) { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_INVALID, - ErrorConstants.INSTRUCTED_AMOUNT_CURRENCY_NOT_STRING); - } - } - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_INVALID, - ErrorConstants.INSTRUCTED_AMOUNT_NOT_JSON_ERROR); - } - } - - //Validate Creditor Account - JSONObject validateCreditorAccResult = VRPSubmissionPayloadValidator.validateCreditorAcc - (submission, initiation); - if (!Boolean.parseBoolean(validateCreditorAccResult. - get(ConsentExtensionConstants.IS_VALID_PAYLOAD).toString())) { - return validateCreditorAccResult; - } - - if (submission.containsKey(ConsentExtensionConstants.INSTRUCTION_IDENTIFICATION)) { - Object value = submission.get(ConsentExtensionConstants.INSTRUCTION_IDENTIFICATION); - - // Check if the instruction_identification is an instance of a string - if (!isValidString(value)) { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_INVALID, - ErrorConstants.INVALID_SUBMISSION_TYPE); - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.INSTRUCTION_IDENTIFICATION_NOT_FOUND); - } - - if (submission.containsKey(ConsentExtensionConstants.END_TO_END_IDENTIFICATION)) { - Object endToEndIdentificationValue = submission. - get(ConsentExtensionConstants.END_TO_END_IDENTIFICATION); - if (!isValidString(endToEndIdentificationValue)) { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_INVALID, - ErrorConstants.INVALID_END_TO_END_IDENTIFICATION_TYPE); - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.END_TO_END_IDENTIFICATION_PARAMETER_NOT_FOUND); - } - - if ((!submission.containsKey(ConsentExtensionConstants.REMITTANCE_INFO) - && initiation.containsKey(ConsentExtensionConstants.REMITTANCE_INFO)) || - (submission.containsKey(ConsentExtensionConstants.REMITTANCE_INFO) - && !initiation.containsKey(ConsentExtensionConstants.REMITTANCE_INFO))) { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.REMITTANCE_INFO_NOT_FOUND); - } else if (submission.containsKey(ConsentExtensionConstants.REMITTANCE_INFO) - && initiation.containsKey(ConsentExtensionConstants.REMITTANCE_INFO)) { - Object remittanceInformationSubmission = submission - .get(ConsentExtensionConstants.REMITTANCE_INFO); - Object remittanceInformationInitiation = initiation - .get(ConsentExtensionConstants.REMITTANCE_INFO); - - if (remittanceInformationSubmission instanceof JSONObject && - remittanceInformationInitiation instanceof JSONObject) { - JSONObject remittanceInformationSub = (JSONObject) submission - .get(ConsentExtensionConstants.REMITTANCE_INFO); - JSONObject remittanceInformationInit = (JSONObject) initiation - .get(ConsentExtensionConstants.REMITTANCE_INFO); - - JSONObject remittanceInfoValidationResult = VRPSubmissionPayloadValidator.validateRemittanceInfo - (remittanceInformationSub, remittanceInformationInit); - if ((!Boolean.parseBoolean(remittanceInfoValidationResult. - get(ConsentExtensionConstants.IS_VALID_PAYLOAD).toString()))) { - return remittanceInfoValidationResult; - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_INVALID, - ErrorConstants.INSTRUCTION_REMITTANCE_INFO_NOT_JSON_ERROR); - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.INVALID_PARAMETER); - } - } - - JSONObject validationResult = new JSONObject(); - validationResult.put(ConsentExtensionConstants.IS_VALID_PAYLOAD, true); - return validationResult; - } - - /** - * Validates the remittance information between two remittance information JSONObjects. - * - * @param remittanceInformationSub The remittance information from the submission request. - * @param remittanceInformationInit The remittance information from the initiation request. - * @return A JSONObject indicating the validation result. It contains a boolean value under the key - * ConsentExtensionConstants.IS_VALID_PAYLOAD, indicating whether the payload is valid. If the - * validation fails, it returns a JSONObject containing error details with keys defined in ErrorConstants. - */ - public static JSONObject validateRemittanceInfo(JSONObject remittanceInformationSub, - JSONObject remittanceInformationInit) { - - if (!ConsentValidatorUtil.compareOptionalParameter( - remittanceInformationSub.getAsString(ConsentExtensionConstants.REFERENCE), - remittanceInformationInit.getAsString(ConsentExtensionConstants.REFERENCE))) { - - return ConsentValidatorUtil.getValidationResult(ErrorConstants.RESOURCE_CONSENT_MISMATCH, - ErrorConstants.REMITTANCE_INFO_MISMATCH); - } - - if (!ConsentValidatorUtil.compareOptionalParameter( - remittanceInformationSub.getAsString(ConsentExtensionConstants.UNSTRUCTURED), - remittanceInformationInit.getAsString(ConsentExtensionConstants.UNSTRUCTURED))) { - - return ConsentValidatorUtil.getValidationResult(ErrorConstants.RESOURCE_CONSENT_MISMATCH, - ErrorConstants.REMITTANCE_UNSTRUCTURED_MISMATCH); - } - - JSONObject validationResult = new JSONObject(); - validationResult.put(ConsentExtensionConstants.IS_VALID_PAYLOAD, true); - return validationResult; - } - - /** - * Validates the risk parameters between the risk of submission and the risk of initiation JSONObjects. - * - * @param riskOfSubmission The risk parameters from the submission. - * @param riskOfInitiation The risk parameters from the initiation. - * @return A JSONObject indicating the validation result. It contains a boolean value under the key - * ConsentExtensionConstants.IS_VALID_PAYLOAD, indicating whether the payload is valid. If the - * validation fails, it returns a JSONObject containing error details with keys defined in ErrorConstants. - */ - public static JSONObject validateRisk(JSONObject riskOfSubmission, - JSONObject riskOfInitiation) { - - if (!ConsentValidatorUtil.compareOptionalParameter( - riskOfSubmission.getAsString(ConsentExtensionConstants.CONTEXT_CODE), - riskOfInitiation.getAsString(ConsentExtensionConstants.CONTEXT_CODE))) { - - return ConsentValidatorUtil.getValidationResult(ErrorConstants.RESOURCE_CONSENT_MISMATCH, - ErrorConstants.RISK_PARAMETER_MISMATCH); - } - JSONObject validationResult = new JSONObject(); - validationResult.put(ConsentExtensionConstants.IS_VALID_PAYLOAD, true); - return validationResult; - } - - /** - * This method validates whether the risk parameter is present in the request and validates the risk parameter is an - * instance of JSONObject. - * - * @param submissionJson - * @return validationResult - */ - public static JSONObject validateRiskParameter(JSONObject submissionJson) { - - //Validate RISK - if (submissionJson.containsKey(ConsentExtensionConstants.RISK)) { - - Object dataObject = submissionJson.get(ConsentExtensionConstants.RISK); - // Check if the risk is valid JSON Object - if (!isValidJSONObject(dataObject)) { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_INVALID, - ErrorConstants.RISK_NOT_JSON_ERROR); - } - } else { - log.error(ErrorConstants.RISK_NOT_FOUND); - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.RISK_NOT_FOUND); - } - JSONObject validationResult = new JSONObject(); - validationResult.put(ConsentExtensionConstants.IS_VALID_PAYLOAD, true); - return validationResult; - } - - /** - * Checks if the given Object is a JSONObject and the JSONObject is non-empty , and it is an instance of a string. - * - * @param value The Object to be validated. - * @return true if the object is a non-null and non-empty JSONObject. - */ - public static boolean isValidString(Object value) { - return value instanceof String; - } - - /** - * Validates initiation parameter in the submission data. - * - * @param submissionData The JSONObject containing submission data. - * @return A JSONObject indicating the validation result. - */ - public static JSONObject validateInitiationParameter(JSONObject submissionData) { - - if (submissionData.containsKey(ConsentExtensionConstants.INITIATION)) { - - Object dataObject = submissionData.get(ConsentExtensionConstants.INITIATION); - - if (!isValidJSONObject(dataObject)) { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_INVALID, - ErrorConstants.INITIATION_NOT_JSON); - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.INITIATION_NOT_FOUND); - } - JSONObject validationResult = new JSONObject(); - validationResult.put(ConsentExtensionConstants.IS_VALID_PAYLOAD, true); - return validationResult; - } - - /** - * Validates instruction parameter in the submission data. - * - * @param submissionData The JSONObject containing submission data. - * @return A JSONObject indicating the validation result. - */ - public static JSONObject validateInstructionParameter(JSONObject submissionData) { - - if (submissionData.containsKey(ConsentExtensionConstants.INSTRUCTION)) { - - Object dataObject = submissionData.get(ConsentExtensionConstants.INSTRUCTION); - if (!isValidJSONObject(dataObject)) { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_INVALID, - ErrorConstants.INSTRUCTION_NOT_JSON); - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.INSTRUCTION_NOT_FOUND); - } - JSONObject validationResult = new JSONObject(); - validationResult.put(ConsentExtensionConstants.IS_VALID_PAYLOAD, true); - return validationResult; - } - - /** - * Extracts submission data from a JSONObject. - * - * @param submissionJson The JSONObject containing submission data. - * @return A JSONObject indicating the validation result. - */ - public static JSONObject validateSubmissionData(JSONObject submissionJson) { - - if (!submissionJson.containsKey(ConsentExtensionConstants.DATA) && - !(submissionJson.get(ConsentExtensionConstants.DATA) instanceof JSONObject)) { - log.error(ErrorConstants.DATA_NOT_FOUND); - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.DATA_NOT_FOUND); - } - JSONObject validationResult = new JSONObject(); - validationResult.put(ConsentExtensionConstants.IS_VALID_PAYLOAD, true); - return validationResult; - } - - /** - * Checks if the given object is a valid JSONObject. - * - * @param value The object to be checked. - * @return true if the object is a JSONObject, otherwise false. - */ - public static boolean isValidJSONObject(Object value) { - return value instanceof JSONObject; - } - - /** - * Validates the creditor account parameter between the creditor account of submission under instruction parameter - * and the creditor account of initiation JSONObjects. - * - * @param submission The creditor account parameters from the submission. - * @param initiation The creditor account parameters from the initiation. - * @return A JSONObject indicating the validation result. It contains a boolean value under the key - * ConsentExtensionConstants.IS_VALID_PAYLOAD, indicating whether the payload is valid. If the - * validation fails, it returns a JSONObject containing error details with keys defined in ErrorConstants. - */ - public static JSONObject validateCreditorAcc(JSONObject submission, - JSONObject initiation) { - JSONObject validationResult = new JSONObject(); - - if (submission.containsKey(ConsentExtensionConstants.CREDITOR_ACC)) { - // If the CreditorAccount was not specified in the consent initiation,the CreditorAccount must be specified - // in the instruction present in the submission payload. - if (!initiation.containsKey(ConsentExtensionConstants.CREDITOR_ACC)) { - validationResult.put(ConsentExtensionConstants.IS_VALID_PAYLOAD, true); - } else { - Object submissionCreditorAccounts = submission.get(ConsentExtensionConstants.CREDITOR_ACC); - Object consentInitiationCreditorAccounts = initiation.get(ConsentExtensionConstants.CREDITOR_ACC); - - if (submissionCreditorAccounts instanceof JSONObject && - consentInitiationCreditorAccounts instanceof JSONObject) { - JSONObject submissionCreditorAccount = (JSONObject) submission. - get(ConsentExtensionConstants.CREDITOR_ACC); - JSONObject consentInitiationCreditorAccount = (JSONObject) initiation. - get(ConsentExtensionConstants.CREDITOR_ACC); - - JSONObject creditorAccValidationResult = ConsentValidatorUtil. - validateCreditorAcc(submissionCreditorAccount, consentInitiationCreditorAccount); - if (!Boolean.parseBoolean(validationResult. - getAsString(ConsentExtensionConstants.IS_VALID_PAYLOAD))) { - return creditorAccValidationResult; - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_INVALID, - ErrorConstants.INSTRUCTION_CREDITOR_ACC_NOT_JSON_ERROR); - } - } - } else { - // Creditor account present under the instruction in the submission request - // is considered to be a mandatory parameter - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.CREDITOR_ACC_NOT_FOUND); - } - - validationResult.put(ConsentExtensionConstants.IS_VALID_PAYLOAD, true); - return validationResult; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/model/ConsentValidateData.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/model/ConsentValidateData.java deleted file mode 100644 index 14525e22..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/model/ConsentValidateData.java +++ /dev/null @@ -1,125 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.validate.model; - -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import net.minidev.json.JSONObject; - -import java.util.Map; -import java.util.TreeMap; - -/** - * Data wrapper for consent validate data. - */ -public class ConsentValidateData { - - private JSONObject headers; - private JSONObject payload; - private String requestPath; - private String consentId; - private String userId; - private String clientId; - private Map resourceParams; - private DetailedConsentResource comprehensiveConsent; - private TreeMap headersMap; - - public ConsentValidateData(JSONObject headers, JSONObject payload, String requestPath, String consentId, - String userId, String clientId, Map resourceParams) { - this.headers = headers; - this.payload = payload; - this.requestPath = requestPath; - this.consentId = consentId; - this.userId = userId; - this.clientId = clientId; - this.resourceParams = resourceParams; - } - - public ConsentValidateData(JSONObject headers, JSONObject payload, String requestPath, String consentId, - String userId, String clientId, Map resourceParams, - TreeMap headersMap) { - this.headers = headers; - this.payload = payload; - this.requestPath = requestPath; - this.consentId = consentId; - this.userId = userId; - this.clientId = clientId; - this.resourceParams = resourceParams; - this.headersMap = headersMap; - } - - public String getRequestPath() { - return requestPath; - } - - public JSONObject getPayload() { - return payload; - } - - public JSONObject getHeaders() { - return headers; - } - - public DetailedConsentResource getComprehensiveConsent() { - return comprehensiveConsent; - } - - public void setComprehensiveConsent(DetailedConsentResource comprehensiveConsent) { - this.comprehensiveConsent = comprehensiveConsent; - } - - public String getConsentId() { - return consentId; - } - - public void setConsentId(String consentId) { - this.consentId = consentId; - } - - public String getUserId() { - return userId; - } - - public void setUserId(String userId) { - this.userId = userId; - } - - public String getClientId() { - return clientId; - } - - public void setClientId(String clientId) { - this.clientId = clientId; - } - - public Map getResourceParams() { - return resourceParams; - } - - public void setResourceParams(Map resourceParams) { - this.resourceParams = resourceParams; - } - - public TreeMap getHeadersMap() { - return headersMap; - } - - private void setHeadersMap(TreeMap headersMap) { - this.headersMap = headersMap; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/model/ConsentValidationResult.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/model/ConsentValidationResult.java deleted file mode 100644 index f59bb436..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/model/ConsentValidationResult.java +++ /dev/null @@ -1,99 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.validate.model; - -import net.minidev.json.JSONObject; - -/** - * Data wrapper for the result of consent validation. - */ -public class ConsentValidationResult { - - private boolean isValid = false; - private JSONObject modifiedPayload = null; - private JSONObject consentInformation = new JSONObject(); - /** - * errorCode, errorMessage and httpCode have to be set in error/invalid scenarios. - */ - private String errorCode = null; - private String errorMessage = null; - private int httpCode = 0; - - public boolean isValid() { - return isValid; - } - - public void setValid(boolean valid) { - isValid = valid; - } - - public JSONObject getModifiedPayload() { - return modifiedPayload; - } - - public void setModifiedPayload(JSONObject modifiedPayload) { - this.modifiedPayload = modifiedPayload; - } - - public JSONObject getConsentInformation() { - return consentInformation; - } - - public void setConsentInformation(JSONObject consentInformation) { - this.consentInformation = consentInformation; - } - - public String getErrorCode() { - return errorCode; - } - - public void setErrorCode(String errorCode) { - this.errorCode = errorCode; - } - - public String getErrorMessage() { - return errorMessage; - } - - public void setErrorMessage(String errorMessage) { - this.errorMessage = errorMessage; - } - - public int getHttpCode() { - return httpCode; - } - - public void setHttpCode(int httpCode) { - this.httpCode = httpCode; - } - - public JSONObject generatePayload() throws Exception { - JSONObject payload = new JSONObject(); - payload.appendField("isValid", isValid); - if (modifiedPayload != null) { - payload.appendField("modifiedPayload", modifiedPayload); - } - if (errorCode != null && errorMessage != null && httpCode != 0) { - payload.appendField("errorCode", errorCode); - payload.appendField("errorMessage", errorMessage); - payload.appendField("httpCode", httpCode); - } - return payload; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/model/ConsentValidator.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/model/ConsentValidator.java deleted file mode 100644 index 9f7767d8..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/model/ConsentValidator.java +++ /dev/null @@ -1,38 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.validate.model; - -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; - -/** - * Consent validator interface. - */ -public interface ConsentValidator { - - /** - * Validate function to implement required validations. - * - * @param consentValidateData Object with data of the request and consent needed for validations - * @param consentValidationResult Object to set data with regard to the validation result - * @throws ConsentException Error object with data required for the error response - */ - public void validate(ConsentValidateData consentValidateData, ConsentValidationResult consentValidationResult) - throws ConsentException; - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/util/ConsentValidatorUtil.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/util/ConsentValidatorUtil.java deleted file mode 100644 index a80d3ba6..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/util/ConsentValidatorUtil.java +++ /dev/null @@ -1,302 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.validate.util; - -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.consent.extensions.validate.model.ConsentValidationResult; -import net.minidev.json.JSONObject; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.HttpStatus; - -import java.time.OffsetDateTime; -import java.time.format.DateTimeParseException; -import java.util.Arrays; -import java.util.List; - -/** - * Consent validate util class for accelerator. - */ -public class ConsentValidatorUtil { - - private static final Log log = LogFactory.getLog(ConsentValidatorUtil.class); - - /** - * Utility method to validate mandatory parameters. - * - * @param str1 First String to validate - * @param str2 Second String to validate - * @return Whether mandatory parameters are same - */ - public static boolean compareMandatoryParameter(String str1, String str2) { - - return (str1 == null) || (str2 == null) ? false : str1.equals(str2); - - } - - /** - * Method to construct the validation result. - * - * @param errorCode Error Code - * @param errorMessage Error Message - * @return Validation Result - */ - public static JSONObject getValidationResult(String errorCode, String errorMessage) { - - JSONObject validationResult = new JSONObject(); - log.error(errorMessage); - validationResult.put(ConsentExtensionConstants.IS_VALID_PAYLOAD, false); - validationResult.put(ConsentExtensionConstants.ERROR_CODE, errorCode); - validationResult.put(ConsentExtensionConstants.ERROR_MESSAGE, errorMessage); - - return validationResult; - } - - - /** - * Populates the provided consent validation result object with error information. - * - * @param errorResult the JSONObject containing error details, specifically error message and error code - * @param consentValidationResult the ConsentValidationResult object to be updated with error details - * - */ - public static void setErrorMessageForConsentValidationResult(JSONObject errorResult - , ConsentValidationResult consentValidationResult) { - - String errorMessage = errorResult.getAsString(ConsentExtensionConstants.ERROR_MESSAGE); - String errorCode = errorResult.getAsString(ConsentExtensionConstants.ERROR_CODE); - - consentValidationResult.setErrorMessage(errorMessage); - consentValidationResult.setErrorCode(errorCode); - consentValidationResult.setHttpCode(HttpStatus.SC_BAD_REQUEST); - } - - /** - * Method to construct the success validation result. - * - * @return Validation Result - */ - public static JSONObject getSuccessValidationResult() { - - JSONObject validationResult = new JSONObject(); - validationResult.put(ConsentExtensionConstants.IS_VALID_PAYLOAD, true); - - return validationResult; - } - - /** - * Validate whether fields in creditor account from initiation and submission are same. - * - * @param subCreditorAccount Creditor Account from submission request - * @param initCreditorAccount Creditor Account from initiation request - * @return Validation Result - */ - public static JSONObject validateCreditorAcc(JSONObject subCreditorAccount, JSONObject initCreditorAccount) { - - if (subCreditorAccount.containsKey(ConsentExtensionConstants.SCHEME_NAME)) { - if (StringUtils.isEmpty(subCreditorAccount.getAsString(ConsentExtensionConstants.SCHEME_NAME)) || - !ConsentValidatorUtil.compareMandatoryParameter( - subCreditorAccount.getAsString(ConsentExtensionConstants.SCHEME_NAME), - initCreditorAccount.getAsString(ConsentExtensionConstants.SCHEME_NAME))) { - - return ConsentValidatorUtil.getValidationResult(ErrorConstants.RESOURCE_CONSENT_MISMATCH, - ErrorConstants.CREDITOR_ACC_SCHEME_NAME_MISMATCH); - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.CREDITOR_ACC_SCHEME_NAME_NOT_FOUND); - } - - if (subCreditorAccount.containsKey(ConsentExtensionConstants.IDENTIFICATION)) { - if (StringUtils.isEmpty(subCreditorAccount.getAsString(ConsentExtensionConstants.IDENTIFICATION)) || - !ConsentValidatorUtil.compareMandatoryParameter( - subCreditorAccount.getAsString(ConsentExtensionConstants.IDENTIFICATION), - initCreditorAccount.getAsString(ConsentExtensionConstants.IDENTIFICATION))) { - - return ConsentValidatorUtil.getValidationResult(ErrorConstants.RESOURCE_CONSENT_MISMATCH, - ErrorConstants.CREDITOR_ACC_IDENTIFICATION_MISMATCH); - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.CREDITOR_ACC_IDENTIFICATION_NOT_FOUND); - } - - if (!ConsentValidatorUtil - .compareOptionalParameter(subCreditorAccount.getAsString(ConsentExtensionConstants.NAME), - initCreditorAccount.getAsString(ConsentExtensionConstants.NAME))) { - - return ConsentValidatorUtil.getValidationResult(ErrorConstants.RESOURCE_CONSENT_MISMATCH, - ErrorConstants.CREDITOR_ACC_NAME_MISMATCH); - } - - if (!ConsentValidatorUtil.compareOptionalParameter(subCreditorAccount - .getAsString(ConsentExtensionConstants.SECONDARY_IDENTIFICATION), - initCreditorAccount.getAsString(ConsentExtensionConstants.SECONDARY_IDENTIFICATION))) { - - return ConsentValidatorUtil - .getValidationResult(ErrorConstants.RESOURCE_CONSENT_MISMATCH, - ErrorConstants.CREDITOR_ACC_SEC_IDENTIFICATION_MISMATCH); - } - JSONObject validationResult = new JSONObject(); - validationResult.put(ConsentExtensionConstants.IS_VALID_PAYLOAD, true); - - return validationResult; - } - /** - * Utility method to validate optional parameters. - * - * @param str1 First String to validate - * @param str2 Second String to validate - * @return Whether optional parameters are same - */ - public static boolean compareOptionalParameter(String str1, String str2) { - - boolean isStr1Empty = StringUtils.isBlank(str1); - boolean isStr2Empty = StringUtils.isBlank(str2); - - if (!(isStr1Empty || isStr2Empty)) { - return str1.equals(str2); - } else { - return (isStr1Empty && isStr2Empty); - } - } - - /** - * Validate whether fields in debtor account from initiation and submission are same. - * - * @param subDebtorAccount Debtor Account from submission request - * @param initDebtorAccount Debtor Account from initiation request - * @return Validation Result - */ - public static JSONObject validateDebtorAcc(JSONObject subDebtorAccount, JSONObject initDebtorAccount) { - - if (subDebtorAccount.containsKey(ConsentExtensionConstants.SCHEME_NAME)) { - if (StringUtils.isEmpty(subDebtorAccount.getAsString(ConsentExtensionConstants.SCHEME_NAME)) || - !ConsentValidatorUtil.compareMandatoryParameter( - subDebtorAccount.getAsString(ConsentExtensionConstants.SCHEME_NAME), - initDebtorAccount.getAsString(ConsentExtensionConstants.SCHEME_NAME))) { - - return ConsentValidatorUtil.getValidationResult(ErrorConstants.RESOURCE_CONSENT_MISMATCH, - ErrorConstants.DEBTOR_ACC_SCHEME_NAME_MISMATCH); - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.DEBTOR_ACC_SCHEME_NAME_NOT_FOUND); - } - - if (subDebtorAccount.containsKey(ConsentExtensionConstants.IDENTIFICATION)) { - if (StringUtils.isEmpty(subDebtorAccount.getAsString(ConsentExtensionConstants.IDENTIFICATION)) || - !ConsentValidatorUtil.compareMandatoryParameter( - subDebtorAccount.getAsString(ConsentExtensionConstants.IDENTIFICATION), - initDebtorAccount.getAsString(ConsentExtensionConstants.IDENTIFICATION))) { - - return ConsentValidatorUtil - .getValidationResult(ErrorConstants.RESOURCE_CONSENT_MISMATCH, - ErrorConstants.DEBTOR_ACC_IDENTIFICATION_MISMATCH); - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.DEBTOR_ACC_IDENTIFICATION_NOT_FOUND); - } - - if (!ConsentValidatorUtil.compareOptionalParameter( - subDebtorAccount.getAsString(ConsentExtensionConstants.NAME), - initDebtorAccount.getAsString(ConsentExtensionConstants.NAME))) { - - return ConsentValidatorUtil.getValidationResult(ErrorConstants.RESOURCE_CONSENT_MISMATCH, - ErrorConstants.DEBTOR_ACC_NAME_MISMATCH); - } - - if (!ConsentValidatorUtil.compareOptionalParameter(subDebtorAccount - .getAsString(ConsentExtensionConstants.SECONDARY_IDENTIFICATION), - initDebtorAccount.getAsString(ConsentExtensionConstants.SECONDARY_IDENTIFICATION))) { - - return ConsentValidatorUtil - .getValidationResult(ErrorConstants.RESOURCE_CONSENT_MISMATCH, - ErrorConstants.DEBTOR_ACC_SEC_IDENTIFICATION_MISMATCH); - } - - JSONObject validationResult = new JSONObject(); - validationResult.put(ConsentExtensionConstants.IS_VALID_PAYLOAD, true); - - return validationResult; - } - /** - * Method provides API resource paths applicable for Confirmtaion of Funds API. - * - * @return map of API Resources. - */ - public static List getCOFAPIPathRegexArray() { - - List requestUrls = Arrays.asList(ConsentExtensionConstants.COF_CONSENT_INITIATION_PATH, - ConsentExtensionConstants.COF_CONSENT_CONSENT_ID_PATH, - ConsentExtensionConstants.COF_SUBMISSION_PATH); - - return requestUrls; - - } - - - /** - * Util method to validate the Confirmation of Funds request URI. - * - * @param uri Request URI - * @return Whether URI is valid - */ - public static boolean isCOFURIValid(String uri) { - - List accountPaths = getCOFAPIPathRegexArray(); - - for (String entry : accountPaths) { - if (uri.equals(entry)) { - return true; - } - } - - return false; - } - - /** - * Validate whether consent is expired. - * - * @param expDateVal Expiration Date Time - * @return Whether consent is expired - * @throws ConsentException if an error occurs while parsing expiration date - */ - public static boolean isConsentExpired(String expDateVal) throws ConsentException { - - if (expDateVal != null && !expDateVal.isEmpty()) { - try { - OffsetDateTime expDate = OffsetDateTime.parse(expDateVal); - return OffsetDateTime.now().isAfter(expDate); - } catch (DateTimeParseException e) { - log.error(ErrorConstants.EXP_DATE_PARSE_ERROR + " : " + expDateVal); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - ErrorConstants.EXP_DATE_PARSE_ERROR); - } - } else { - return false; - } - - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/util/PaymentSubmissionValidationUtil.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/util/PaymentSubmissionValidationUtil.java deleted file mode 100644 index dbfa8cb6..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/java/com/wso2/openbanking/accelerator/consent/extensions/validate/util/PaymentSubmissionValidationUtil.java +++ /dev/null @@ -1,103 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.validate.util; - -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import net.minidev.json.JSONObject; -import org.apache.commons.lang3.StringUtils; - -/** - * Util class for Payment Submission Validation. - */ -public class PaymentSubmissionValidationUtil { - - public static JSONObject validateInstructionIdentification (JSONObject submission, JSONObject initiation) { - if (submission.containsKey(ConsentExtensionConstants.INSTRUCTION_IDENTIFICATION)) { - if (StringUtils.isEmpty(submission.getAsString(ConsentExtensionConstants.INSTRUCTION_IDENTIFICATION)) - || !ConsentValidatorUtil.compareMandatoryParameter( - submission.getAsString(ConsentExtensionConstants.INSTRUCTION_IDENTIFICATION), - initiation.getAsString(ConsentExtensionConstants.INSTRUCTION_IDENTIFICATION))) { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.RESOURCE_CONSENT_MISMATCH, - ErrorConstants.INSTRUCTION_IDENTIFICATION_MISMATCH); - } - } - return ConsentValidatorUtil.getSuccessValidationResult(); - } - - public static JSONObject validateEndToEndIdentification (JSONObject submission, JSONObject initiation) { - - if (submission.containsKey(ConsentExtensionConstants.END_TO_END_IDENTIFICATION)) { - if (StringUtils.isEmpty(submission.getAsString(ConsentExtensionConstants.END_TO_END_IDENTIFICATION)) - || !ConsentValidatorUtil.compareMandatoryParameter( - submission.getAsString(ConsentExtensionConstants.END_TO_END_IDENTIFICATION), - initiation.getAsString(ConsentExtensionConstants.END_TO_END_IDENTIFICATION))) { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.RESOURCE_CONSENT_MISMATCH, - ErrorConstants.END_TO_END_IDENTIFICATION_MISMATCH); - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.END_TO_END_IDENTIFICATION_NOT_FOUND); - } - - return ConsentValidatorUtil.getSuccessValidationResult(); - } - - public static JSONObject validateInstructedAmount (JSONObject submission, JSONObject initiation) { - - if (submission.containsKey(ConsentExtensionConstants.INSTRUCTED_AMOUNT)) { - - JSONObject subInstrAmount = (JSONObject) submission.get(ConsentExtensionConstants.INSTRUCTED_AMOUNT); - JSONObject initInstrAmount = (JSONObject) initiation.get(ConsentExtensionConstants.INSTRUCTED_AMOUNT); - - if (subInstrAmount.containsKey(ConsentExtensionConstants.AMOUNT)) { - if (StringUtils.isEmpty(subInstrAmount.getAsString(ConsentExtensionConstants.AMOUNT)) || - !ConsentValidatorUtil.compareMandatoryParameter( - subInstrAmount.getAsString(ConsentExtensionConstants.AMOUNT), - initInstrAmount.getAsString(ConsentExtensionConstants.AMOUNT))) { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.RESOURCE_CONSENT_MISMATCH, - ErrorConstants.INSTRUCTED_AMOUNT_AMOUNT_MISMATCH); - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.INSTRUCTED_AMOUNT_AMOUNT_NOT_FOUND); - } - - if (subInstrAmount.containsKey(ConsentExtensionConstants.CURRENCY)) { - if (StringUtils.isEmpty(subInstrAmount.getAsString(ConsentExtensionConstants.CURRENCY)) || - !ConsentValidatorUtil.compareMandatoryParameter( - subInstrAmount.getAsString(ConsentExtensionConstants.CURRENCY), - initInstrAmount.getAsString(ConsentExtensionConstants.CURRENCY))) { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.RESOURCE_CONSENT_MISMATCH, - ErrorConstants.INSTRUCTED_AMOUNT_CURRENCY_MISMATCH); - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.INSTRUCTED_AMOUNT_CURRENCY_NOT_FOUND); - } - } else { - return ConsentValidatorUtil.getValidationResult(ErrorConstants.FIELD_MISSING, - ErrorConstants.INSTRUCTED_AMOUNT_NOT_FOUND); - } - - return ConsentValidatorUtil.getSuccessValidationResult(); - } - -} - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/resources/findbugs-exclude.xml b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/resources/findbugs-exclude.xml deleted file mode 100644 index 5160ee77..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/resources/findbugs-exclude.xml +++ /dev/null @@ -1,20 +0,0 @@ - - - - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/resources/findbugs-include.xml b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/resources/findbugs-include.xml deleted file mode 100644 index 8932a22e..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/vrp/persistence/flow/ConsentPersistStepTests.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/vrp/persistence/flow/ConsentPersistStepTests.java deleted file mode 100644 index db75703c..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/vrp/persistence/flow/ConsentPersistStepTests.java +++ /dev/null @@ -1,258 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authorize.vrp.persistence.flow; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.util.CarbonUtils; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.impl.DefaultConsentPersistStep; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentData; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentPersistData; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentServiceUtil; -import com.wso2.openbanking.accelerator.consent.extensions.utils.ConsentAuthorizeTestConstants; -import com.wso2.openbanking.accelerator.consent.extensions.utils.ConsentExtensionTestUtils; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import org.mockito.Mock; -import org.mockito.Mockito; -import org.mockito.MockitoAnnotations; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockObjectFactory; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; - -import java.util.HashMap; -import java.util.Map; - -import static org.powermock.api.mockito.PowerMockito.when; - -/** - * Test class for Consent Persistence. - */ -@PrepareForTest({OpenBankingConfigParser.class, ConsentServiceUtil.class}) -@PowerMockIgnore({"com.wso2.openbanking.accelerator.consent.extensions.common.*", "net.minidev.*", - "jdk.internal.reflect.*"}) -public class ConsentPersistStepTests { - - @Mock - OpenBankingConfigParser openBankingConfigParserMock; - @Mock - private static DefaultConsentPersistStep consentPersistStep; - @Mock - private static ConsentPersistData consentPersistDataMock; - @Mock - private static ConsentData consentDataMock; - @Mock - private static ConsentResource consentResourceMock; - @Mock - ConsentCoreServiceImpl consentCoreServiceMock; - private static Map configMap; - JSONParser parser = new JSONParser(JSONParser.MODE_PERMISSIVE); - - @BeforeClass - public void initTest() throws ReflectiveOperationException { - - MockitoAnnotations.initMocks(this); - - consentPersistStep = new DefaultConsentPersistStep(); - consentPersistDataMock = Mockito.mock(ConsentPersistData.class); - consentDataMock = Mockito.mock(ConsentData.class); - consentResourceMock = Mockito.mock(ConsentResource.class); - consentCoreServiceMock = Mockito.mock(ConsentCoreServiceImpl.class); - - configMap = new HashMap<>(); - configMap.put("ErrorURL", "https://localhost:8243/error"); - - //to execute util class initialization - new CarbonUtils(); - System.setProperty("some.property", "property.value"); - System.setProperty("carbon.home", "."); - ConsentExtensionTestUtils.injectEnvironmentVariable("CARBON_HOME", "."); - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - - return new PowerMockObjectFactory(); - } - - @BeforeMethod - public void initMethod() { - - openBankingConfigParserMock = Mockito.mock(OpenBankingConfigParser.class); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceMock); - } - - @Test(priority = 1, expectedExceptions = ConsentException.class) - public void testConsentPersistWithoutConsentId() { - - Mockito.doReturn(consentDataMock).when(consentPersistDataMock).getConsentData(); - consentPersistStep.execute(consentPersistDataMock); - } - - @Test(priority = 3, expectedExceptions = ConsentException.class) - public void testConsentPersistWithoutAuthResource() { - - Mockito.doReturn(consentDataMock).when(consentPersistDataMock).getConsentData(); - Mockito.doReturn("1234").when(consentDataMock).getConsentId(); - Mockito.doReturn(consentResourceMock).when(consentDataMock).getConsentResource(); - - consentPersistStep.execute(consentPersistDataMock); - } - - @Test(priority = 6, expectedExceptions = ConsentException.class) - public void testAccountConsentPersistWithoutAccountIDs() throws Exception { - - Mockito.doReturn(consentDataMock).when(consentPersistDataMock).getConsentData(); - Mockito.doReturn(ConsentAuthorizeTestConstants.CONSENT_ID).when(consentDataMock).getConsentId(); - Mockito.doReturn(ConsentAuthorizeTestConstants.USER_ID).when(consentDataMock).getUserId(); - Mockito.doReturn(ConsentAuthorizeTestConstants.CLIENT_ID).when(consentDataMock).getClientId(); - Mockito.doReturn(consentResourceMock).when(consentDataMock).getConsentResource(); - Mockito.doReturn(ConsentAuthorizeTestConstants.getAuthResource()).when(consentDataMock).getAuthResource(); - Mockito.doReturn(ConsentAuthorizeTestConstants.ACCOUNTS).when(consentResourceMock).getConsentType(); - Mockito.doReturn(true).when(consentPersistDataMock).getApproval(); - - JSONObject payload = (JSONObject) parser - .parse(ConsentAuthorizeTestConstants.ACCOUNT_PERSIST_PAYLOAD_WITHOUT_ACCOUNT_ID); - Mockito.doReturn(payload).when(consentPersistDataMock).getPayload(); - - consentPersistStep.execute(consentPersistDataMock); - } - - @Test(priority = 7, expectedExceptions = ConsentException.class) - public void testAccountConsentPersistWithNonStringAccountIDs() throws Exception { - - Mockito.doReturn(consentDataMock).when(consentPersistDataMock).getConsentData(); - Mockito.doReturn(ConsentAuthorizeTestConstants.CONSENT_ID).when(consentDataMock).getConsentId(); - Mockito.doReturn(ConsentAuthorizeTestConstants.USER_ID).when(consentDataMock).getUserId(); - Mockito.doReturn(ConsentAuthorizeTestConstants.CLIENT_ID).when(consentDataMock).getClientId(); - Mockito.doReturn(consentResourceMock).when(consentDataMock).getConsentResource(); - Mockito.doReturn(ConsentAuthorizeTestConstants.getAuthResource()).when(consentDataMock).getAuthResource(); - Mockito.doReturn(ConsentAuthorizeTestConstants.ACCOUNTS).when(consentResourceMock).getConsentType(); - Mockito.doReturn(true).when(consentPersistDataMock).getApproval(); - - JSONObject payload = (JSONObject) parser - .parse(ConsentAuthorizeTestConstants.PAYLOAD_WITH_NON_STRING_ACCOUNTID); - Mockito.doReturn(payload).when(consentPersistDataMock).getPayload(); - - consentPersistStep.execute(consentPersistDataMock); - } - - @Test(priority = 9, expectedExceptions = ConsentException.class) - public void testCOFConsentPersistWithoutCOFAccount() throws Exception { - - Mockito.doReturn(consentDataMock).when(consentPersistDataMock).getConsentData(); - Mockito.doReturn(ConsentAuthorizeTestConstants.CONSENT_ID).when(consentDataMock).getConsentId(); - Mockito.doReturn(ConsentAuthorizeTestConstants.USER_ID).when(consentDataMock).getUserId(); - Mockito.doReturn(ConsentAuthorizeTestConstants.CLIENT_ID).when(consentDataMock).getClientId(); - Mockito.doReturn(consentResourceMock).when(consentDataMock).getConsentResource(); - Mockito.doReturn(ConsentAuthorizeTestConstants.getAuthResource()).when(consentDataMock).getAuthResource(); - Mockito.doReturn(ConsentAuthorizeTestConstants.FUNDS_CONFIRMATIONS).when(consentResourceMock) - .getConsentType(); - Mockito.doReturn(true).when(consentPersistDataMock).getApproval(); - - JSONObject payload = (JSONObject) parser - .parse(ConsentAuthorizeTestConstants.COF_PERSIST_PAYLOAD_WITHOUT_COF_ACC); - Mockito.doReturn(payload).when(consentPersistDataMock).getPayload(); - - consentPersistStep.execute(consentPersistDataMock); - } - - @Test(priority = 10, expectedExceptions = ConsentException.class) - public void testCOFConsentPersistWithNonStringCOFAccount() throws Exception { - - Mockito.doReturn(consentDataMock).when(consentPersistDataMock).getConsentData(); - Mockito.doReturn(ConsentAuthorizeTestConstants.CONSENT_ID).when(consentDataMock).getConsentId(); - Mockito.doReturn(ConsentAuthorizeTestConstants.USER_ID).when(consentDataMock).getUserId(); - Mockito.doReturn(ConsentAuthorizeTestConstants.CLIENT_ID).when(consentDataMock).getClientId(); - Mockito.doReturn(consentResourceMock).when(consentDataMock).getConsentResource(); - Mockito.doReturn(ConsentAuthorizeTestConstants.getAuthResource()).when(consentDataMock).getAuthResource(); - Mockito.doReturn(ConsentAuthorizeTestConstants.FUNDS_CONFIRMATIONS).when(consentResourceMock) - .getConsentType(); - Mockito.doReturn(true).when(consentPersistDataMock).getApproval(); - - JSONObject payload = (JSONObject) parser - .parse(ConsentAuthorizeTestConstants.COF_PERSIST_PAYLOAD_WITH_NON_STRING_COF_ACC); - Mockito.doReturn(payload).when(consentPersistDataMock).getPayload(); - - consentPersistStep.execute(consentPersistDataMock); - } - - @Test(priority = 11, expectedExceptions = ConsentException.class) - public void testCOFPersistThrowingExceptionWhenConsentBinding() throws Exception { - - Mockito.doReturn(consentDataMock).when(consentPersistDataMock).getConsentData(); - Mockito.doReturn(ConsentAuthorizeTestConstants.CONSENT_ID).when(consentDataMock).getConsentId(); - Mockito.doReturn(ConsentAuthorizeTestConstants.USER_ID).when(consentDataMock).getUserId(); - Mockito.doReturn(ConsentAuthorizeTestConstants.CLIENT_ID).when(consentDataMock).getClientId(); - Mockito.doReturn(consentResourceMock).when(consentDataMock).getConsentResource(); - Mockito.doReturn(ConsentAuthorizeTestConstants.getAuthResource()).when(consentDataMock).getAuthResource(); - Mockito.doReturn(ConsentAuthorizeTestConstants.FUNDS_CONFIRMATIONS).when(consentResourceMock) - .getConsentType(); - Mockito.doReturn(false).when(consentPersistDataMock).getApproval(); - JSONParser parser = new JSONParser(JSONParser.MODE_PERMISSIVE); - JSONObject payload = (JSONObject) parser - .parse(ConsentAuthorizeTestConstants.COF_PERSIST_PAYLOAD); - Mockito.doReturn(payload).when(consentPersistDataMock).getPayload(); - - consentPersistStep.execute(consentPersistDataMock); -} - -// @Test -// public void testAccountConsentPersistSuccessScenarioWithApprovalTrue() -// throws ParseException, ConsentManagementException { -// -// Mockito.doReturn(consentDataMock).when(consentPersistDataMock).getConsentData(); -// Mockito.doReturn(ConsentAuthorizeTestConstants.CONSENT_ID).when(consentDataMock).getConsentId(); -// Mockito.doReturn(ConsentAuthorizeTestConstants.USER_ID).when(consentDataMock).getUserId(); -// Mockito.doReturn(ConsentAuthorizeTestConstants.CLIENT_ID).when(consentDataMock).getClientId(); -// Mockito.doReturn(consentResourceMock).when(consentDataMock).getConsentResource(); -// Mockito.doReturn(ConsentAuthorizeTestConstants.getAuthResource()).when(consentDataMock).getAuthResource(); -// Mockito.doReturn(ConsentAuthorizeTestConstants.ACCOUNTS).when(consentResourceMock).getConsentType(); -// Mockito.doReturn(true).when(consentPersistDataMock).getApproval(); -// -// Mockito.doReturn(true).when(consentCoreServiceMock).bindUserAccountsToConsent( -// Mockito.anyObject(), Mockito.anyString(), Mockito.anyString(), Mockito.anyMap(), -// Mockito.anyString(), Mockito.anyString()); -// -// PowerMockito.mockStatic(ConsentServiceUtil.class); -// PowerMockito.when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceMock); -// -// JSONObject payload = (JSONObject) parser.parse(ConsentAuthorizeTestConstants.ACCOUNT_PERSIST_PAYLOAD); -// Mockito.doReturn(payload).when(consentPersistDataMock).getPayload(); -// -// consentPersistStep.execute(consentPersistDataMock); -// } -} - - - - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/vrp/retrieval/flow/ConsentExtensionDataProvider.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/vrp/retrieval/flow/ConsentExtensionDataProvider.java deleted file mode 100644 index 098a8706..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/vrp/retrieval/flow/ConsentExtensionDataProvider.java +++ /dev/null @@ -1,36 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authorize.vrp.retrieval.flow; - -import com.wso2.openbanking.accelerator.consent.extensions.utils.ConsentAuthorizeTestConstants; -import org.testng.annotations.DataProvider; - -/** - * Data Provider for Consent Executor Tests. - */ -public class ConsentExtensionDataProvider { - - @DataProvider(name = "PaymentConsentDataDataProvider") - Object[][] getPaymentConsentDataDataProvider() { - - return new Object[][]{ - {ConsentAuthorizeTestConstants.PAYMENT_INITIATION}, - {ConsentAuthorizeTestConstants.INTERNATIONAL_PAYMENT_INITIATION} - }; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/vrp/retrieval/flow/VRPConsentRetrievalStepTest.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/vrp/retrieval/flow/VRPConsentRetrievalStepTest.java deleted file mode 100644 index f8b11e02..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/vrp/retrieval/flow/VRPConsentRetrievalStepTest.java +++ /dev/null @@ -1,282 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authorize.vrp.retrieval.flow; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.util.CarbonUtils; -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.impl.DefaultConsentRetrievalStep; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentData; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentServiceUtil; -import com.wso2.openbanking.accelerator.consent.extensions.utils.ConsentAuthorizeTestConstants; -import com.wso2.openbanking.accelerator.consent.extensions.utils.ConsentExtensionTestUtils; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentFile; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.ParseException; -import org.mockito.Mock; -import org.mockito.Mockito; -import org.mockito.MockitoAnnotations; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Map; - -/** - * Test class for Consent Retrieval Step. - */ -@PrepareForTest({OpenBankingConfigParser.class, OpenBankingConfigParser.class, ConsentServiceUtil.class}) -@PowerMockIgnore({"com.wso2.openbanking.accelerator.consent.extensions.common.*", "net.minidev.*", - "jdk.internal.reflect.*"}) -public class VRPConsentRetrievalStepTest extends PowerMockTestCase { - - private static DefaultConsentRetrievalStep defaultConsentRetrievalStep; - @Mock - private static ConsentData consentDataMock; - @Mock - private static ConsentResource consentResourceMock; - @Mock - private static AuthorizationResource authorizationResourceMock; - @Mock - ConsentFile consentFileMock; - @Mock - ConsentCoreServiceImpl consentCoreServiceMock; - @Mock - OpenBankingConfigParser openBankingConfigParserMock; - - private static Map configMap; - ArrayList authResources; - - @BeforeClass - public void initClass() { - - MockitoAnnotations.initMocks(this); - - defaultConsentRetrievalStep = new DefaultConsentRetrievalStep(); - consentDataMock = Mockito.mock(ConsentData.class); - consentResourceMock = Mockito.mock(ConsentResource.class); - authorizationResourceMock = Mockito.mock(AuthorizationResource.class); - consentFileMock = Mockito.mock(ConsentFile.class); - consentCoreServiceMock = Mockito.mock(ConsentCoreServiceImpl.class); - - configMap = new HashMap<>(); - openBankingConfigParserMock = Mockito.mock(OpenBankingConfigParser.class); - authResources = new ArrayList(); - } - - @BeforeMethod - public void initMethod() throws ReflectiveOperationException { - - //to execute util class initialization - new CarbonUtils(); - System.setProperty("some.property", "property.value"); - System.setProperty("carbon.home", "."); - ConsentExtensionTestUtils.injectEnvironmentVariable("CARBON_HOME", "."); - - OpenBankingConfigParser openBankingConfigParserMock = Mockito.mock(OpenBankingConfigParser.class); - Mockito.doReturn("jdbc/WSO2OB_DB").when(openBankingConfigParserMock).getDataSourceName(); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - - Mockito.doReturn(configMap).when(openBankingConfigParserMock).getConfiguration(); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - - @Test - public void testGetConsentDataSetForNonRegulatory() { - - JSONObject jsonObject = new JSONObject(); - Mockito.doReturn(false).when(consentDataMock).isRegulatory(); - defaultConsentRetrievalStep.execute(consentDataMock, jsonObject); - - Assert.assertTrue(jsonObject.isEmpty()); - } - - @Test - public void testConsentRetrievalWithEmptyConsentData() { - - JSONObject jsonObject = new JSONObject(); - Mockito.doReturn(true).when(consentDataMock).isRegulatory(); - defaultConsentRetrievalStep.execute(consentDataMock, jsonObject); - - Assert.assertNotNull(jsonObject.get(ConsentExtensionConstants.IS_ERROR)); - String errorMsg = (String) jsonObject.get(ConsentExtensionConstants.IS_ERROR); - Assert.assertFalse(errorMsg.contains(ErrorConstants.REQUEST_OBJ_EXTRACT_ERROR)); - } - - @Test - public void testConsentRetrievalWithNonJWTRequestObject() { - - JSONObject jsonObject = new JSONObject(); - Mockito.doReturn(true).when(consentDataMock).isRegulatory(); - Mockito.doReturn("request=qwertyuijhbvbn").when(consentDataMock).getSpQueryParams(); - defaultConsentRetrievalStep.execute(consentDataMock, jsonObject); - - Assert.assertNotNull(jsonObject.get(ConsentExtensionConstants.IS_ERROR)); - String errorMsg = (String) jsonObject.get(ConsentExtensionConstants.IS_ERROR); - Assert.assertTrue(errorMsg.contains(ErrorConstants.REQUEST_OBJ_NOT_SIGNED)); - } - - @Test - public void testConsentRetrievalWithInvalidRequestObject() { - - String request = "request=" + ConsentAuthorizeTestConstants.INVALID_REQUEST_OBJECT; - JSONObject jsonObject = new JSONObject(); - Mockito.doReturn(true).when(consentDataMock).isRegulatory(); - Mockito.doReturn(request).when(consentDataMock).getSpQueryParams(); - defaultConsentRetrievalStep.execute(consentDataMock, jsonObject); - - Assert.assertNotNull(jsonObject.get(ConsentExtensionConstants.IS_ERROR)); - String errorMsg = (String) jsonObject.get(ConsentExtensionConstants.IS_ERROR); - Assert.assertTrue(errorMsg.contains(ErrorConstants.NOT_JSON_PAYLOAD)); - } - - @Test - public void testConsentRetrievalWithValidRequestObject() throws ConsentManagementException { - - String request = "request=" + ConsentAuthorizeTestConstants.VALID_REQUEST_OBJECT; - JSONObject jsonObject = new JSONObject(); - Mockito.doReturn(true).when(consentDataMock).isRegulatory(); - Mockito.doReturn(request).when(consentDataMock).getSpQueryParams(); - - Mockito.doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(consentResourceMock).getCurrentStatus(); - Mockito.doReturn(ConsentExtensionConstants.ACCOUNTS).when(consentResourceMock).getConsentType(); - Mockito.doReturn(ConsentAuthorizeTestConstants.VALID_INITIATION_OBJECT).when(consentResourceMock) - .getReceipt(); - Mockito.doReturn(consentResourceMock).when(consentCoreServiceMock) - .getConsent(Mockito.anyString(), Mockito.anyBoolean()); - Mockito.doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(authorizationResourceMock) - .getAuthorizationStatus(); - authResources.add(authorizationResourceMock); - Mockito.doReturn(authResources).when(consentCoreServiceMock) - .searchAuthorizations(Mockito.anyString()); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - PowerMockito.when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceMock); - - defaultConsentRetrievalStep.execute(consentDataMock, jsonObject); - Assert.assertNotNull(jsonObject.get(ConsentExtensionConstants.IS_ERROR)); - } - - @Test - public void testGetConsentDataSetForAccounts() { - - Mockito.doReturn(ConsentExtensionConstants.ACCOUNTS).when(consentResourceMock).getConsentType(); - Mockito.doReturn(ConsentAuthorizeTestConstants.VALID_INITIATION_OBJECT).when(consentResourceMock) - .getReceipt(); - Mockito.doReturn(ConsentAuthorizeTestConstants.AWAITING_AUTH_STATUS).when(consentResourceMock) - .getCurrentStatus(); - - JSONArray accountConsentData = defaultConsentRetrievalStep.getConsentDataSet(consentResourceMock); - Assert.assertNotNull(accountConsentData); - } - - - - @Test(dataProvider = "PaymentConsentDataDataProvider", dataProviderClass = ConsentExtensionDataProvider.class) - public void testGetConsentDataSetForPayments(String paymentReceipt) throws ConsentManagementException, - ParseException { - - Mockito.doReturn(configMap).when(openBankingConfigParserMock).getConfiguration(); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - - Mockito.doReturn(ConsentExtensionConstants.PAYMENTS).when(consentResourceMock).getConsentType(); - Mockito.doReturn(paymentReceipt).when(consentResourceMock).getReceipt(); - Mockito.doReturn(ConsentAuthorizeTestConstants.CREATED_TIME).when(consentResourceMock) - .getCreatedTime(); - Mockito.doReturn(ConsentAuthorizeTestConstants.AWAITING_AUTH_STATUS).when(consentResourceMock) - .getCurrentStatus(); - - JSONArray paymentConsentData = defaultConsentRetrievalStep.getConsentDataSet(consentResourceMock); - Assert.assertNotNull(paymentConsentData); - } - - @Test - public void testGetConsentDataSetForFilePayments() { - - Mockito.doReturn(configMap).when(openBankingConfigParserMock).getConfiguration(); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - PowerMockito.when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceMock); - - Mockito.doReturn(ConsentExtensionConstants.PAYMENTS).when(consentResourceMock).getConsentType(); - Mockito.doReturn(ConsentAuthorizeTestConstants.CREATED_TIME).when(consentResourceMock) - .getCreatedTime(); - Mockito.doReturn(ConsentAuthorizeTestConstants.AWAITING_AUTH_STATUS).when(consentResourceMock) - .getCurrentStatus(); - - JSONArray paymentConsentData = defaultConsentRetrievalStep.getConsentDataSet(consentResourceMock); - Assert.assertNotNull(paymentConsentData); - } - - - @Test - public void testGetConsentDataSetForCOF() { - - Mockito.doReturn(ConsentExtensionConstants.FUNDSCONFIRMATIONS).when(consentResourceMock).getConsentType(); - Mockito.doReturn(ConsentAuthorizeTestConstants.COF_RECEIPT).when(consentResourceMock) - .getReceipt(); - Mockito.doReturn(ConsentAuthorizeTestConstants.AWAITING_AUTH_STATUS).when(consentResourceMock) - .getCurrentStatus(); - - JSONArray cofConsentData = defaultConsentRetrievalStep.getConsentDataSet(consentResourceMock); - Assert.assertNotNull(cofConsentData); - } - - @Test - public void testGetConsentDataSetForVRP() { - - Mockito.doReturn(ConsentExtensionConstants.VRP).when(consentResourceMock).getConsentType(); - Mockito.doReturn(ConsentAuthorizeTestConstants.VRP_INITIATION).when(consentResourceMock) - .getReceipt(); - Mockito.doReturn(ConsentAuthorizeTestConstants.AWAITING_AUTH_STATUS).when(consentResourceMock) - .getCurrentStatus(); - - JSONArray cofConsentData = defaultConsentRetrievalStep.getConsentDataSet(consentResourceMock); - Assert.assertNotNull(cofConsentData); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/vrp/retrieval/flow/VRPConsentRetrievalUtilTest.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/vrp/retrieval/flow/VRPConsentRetrievalUtilTest.java deleted file mode 100644 index 9ad35bf6..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authorize/vrp/retrieval/flow/VRPConsentRetrievalUtilTest.java +++ /dev/null @@ -1,333 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authorize.vrp.retrieval.flow; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.util.CarbonUtils; -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.impl.DefaultConsentRetrievalStep; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentData; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.utils.ConsentRetrievalUtil; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentServiceUtil; -import com.wso2.openbanking.accelerator.consent.extensions.utils.ConsentAuthorizeTestConstants; -import com.wso2.openbanking.accelerator.consent.extensions.utils.ConsentExtensionTestUtils; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import org.mockito.InjectMocks; -import org.mockito.Mock; -import org.mockito.Mockito; -import org.mockito.MockitoAnnotations; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Map; - - -import static com.wso2.openbanking.accelerator.consent.extensions.utils.ConsentAuthorizeTestConstants.VRP_WITHOUT_DATA; -import static org.mockito.Mockito.mock; -import static org.powermock.api.mockito.PowerMockito.doReturn; -import static org.powermock.api.mockito.PowerMockito.when; -import static org.testng.Assert.assertTrue; - -/** - * Test class for consentRetrievalUtil. - */ -@PowerMockIgnore({"com.wso2.openbanking.accelerator.consent.extensions.common.*", "net.minidev.*", - "jdk.internal.reflect.*"}) -@PrepareForTest({OpenBankingConfigParser.class, OpenBankingConfigParser.class, ConsentServiceUtil.class}) -public class VRPConsentRetrievalUtilTest extends PowerMockTestCase { - - @InjectMocks - private final DefaultConsentRetrievalStep defaultConsentRetrievalStep = new DefaultConsentRetrievalStep(); - private final ConsentRetrievalUtil consentRetrievalUtil = new ConsentRetrievalUtil(); - @Mock - private static ConsentData consentDataMock; - @Mock - private static ConsentResource consentResourceMock; - @Mock - private static AuthorizationResource authorizationResourceMock; - @Mock - ConsentCoreServiceImpl consentCoreServiceMock; - @Mock - OpenBankingConfigParser openBankingConfigParser; - @Mock - OpenBankingConfigParser openBankingConfigParse; - private static Map configMap; - ArrayList authResources; - - - @BeforeClass - public void initClass() { - MockitoAnnotations.initMocks(this); - consentDataMock = mock(ConsentData.class); - consentResourceMock = mock(ConsentResource.class); - authorizationResourceMock = mock(AuthorizationResource.class); - consentCoreServiceMock = mock(ConsentCoreServiceImpl.class); - configMap = new HashMap<>(); - authResources = new ArrayList(); - } - - @BeforeClass - public void setUp() throws ReflectiveOperationException { - - MockitoAnnotations.initMocks(this); - new CarbonUtils(); - System.setProperty("some.property", "property.value"); - System.setProperty("carbon.home", "."); - ConsentExtensionTestUtils.injectEnvironmentVariable("CARBON_HOME", "."); - - consentResourceMock = mock(ConsentResource.class); - - } - - @BeforeMethod - public void initMethod() { - openBankingConfigParser = mock(OpenBankingConfigParser.class); - doReturn(configMap).when(openBankingConfigParser).getConfiguration(); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParser); - - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - - @Test - public void testGetConsentDataSetForNonRegulatory() { - - JSONObject jsonObject = new JSONObject(); - Mockito.doReturn(false).when(consentDataMock).isRegulatory(); - defaultConsentRetrievalStep.execute(consentDataMock, jsonObject); - assertTrue(jsonObject.isEmpty()); - } - - @Test - public void testConsentRetrievalWithEmptyConsentData() { - - JSONObject jsonObject = new JSONObject(); - Mockito.doReturn(true).when(consentDataMock).isRegulatory(); - defaultConsentRetrievalStep.execute(consentDataMock, jsonObject); - - Assert.assertNotNull(jsonObject.get(ConsentExtensionConstants.IS_ERROR)); - String errorMsg = (String) jsonObject.get(ConsentExtensionConstants.IS_ERROR); - Assert.assertFalse(errorMsg.contains(ErrorConstants.REQUEST_OBJ_EXTRACT_ERROR)); - } - - @Test - public void testConsentRetrievalWithNonJWTRequestObject() { - - JSONObject jsonObject = new JSONObject(); - Mockito.doReturn(true).when(consentDataMock).isRegulatory(); - Mockito.doReturn("request=qwertyuijhbvbn").when(consentDataMock).getSpQueryParams(); - defaultConsentRetrievalStep.execute(consentDataMock, jsonObject); - - Assert.assertNotNull(jsonObject.get(ConsentExtensionConstants.IS_ERROR)); - String errorMsg = (String) jsonObject.get(ConsentExtensionConstants.IS_ERROR); - Assert.assertTrue(errorMsg.contains(ErrorConstants.REQUEST_OBJ_NOT_SIGNED)); - } - - @Test - public void testConsentRetrievalWithInvalidRequestObject() { - - String request = "request=" + ConsentAuthorizeTestConstants.INVALID_REQUEST_OBJECT; - JSONObject jsonObject = new JSONObject(); - Mockito.doReturn(true).when(consentDataMock).isRegulatory(); - Mockito.doReturn(request).when(consentDataMock).getSpQueryParams(); - defaultConsentRetrievalStep.execute(consentDataMock, jsonObject); - - Assert.assertNotNull(jsonObject.get(ConsentExtensionConstants.IS_ERROR)); - String errorMsg = (String) jsonObject.get(ConsentExtensionConstants.IS_ERROR); - assertTrue(errorMsg.contains(ErrorConstants.NOT_JSON_PAYLOAD)); - } - - @Test - public void testConsentRetrievalWithValidRequestObject() throws ConsentManagementException { - - String request = "request=" + ConsentAuthorizeTestConstants.VALID_REQUEST_OBJECT; - JSONObject jsonObject = new JSONObject(); - Mockito.doReturn(true).when(consentDataMock).isRegulatory(); - Mockito.doReturn(request).when(consentDataMock).getSpQueryParams(); - - Mockito.doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(consentResourceMock).getCurrentStatus(); - Mockito.doReturn(ConsentExtensionConstants.ACCOUNTS).when(consentResourceMock).getConsentType(); - Mockito.doReturn(ConsentAuthorizeTestConstants.VALID_INITIATION_OBJECT).when(consentResourceMock) - .getReceipt(); - Mockito.doReturn(consentResourceMock).when(consentCoreServiceMock) - .getConsent(Mockito.anyString(), Mockito.anyBoolean()); - Mockito.doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(authorizationResourceMock) - .getAuthorizationStatus(); - authResources.add(authorizationResourceMock); - Mockito.doReturn(authResources).when(consentCoreServiceMock) - .searchAuthorizations(Mockito.anyString()); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - PowerMockito.when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceMock); - - defaultConsentRetrievalStep.execute(consentDataMock, jsonObject); - Assert.assertNotNull(jsonObject.get(ConsentExtensionConstants.IS_ERROR)); - } - - @Test - public void testGetConsentDataSetForAccounts() { - Mockito.doReturn(ConsentExtensionConstants.ACCOUNTS).when(consentResourceMock).getConsentType(); - Mockito.doReturn(ConsentAuthorizeTestConstants.VALID_INITIATION_OBJECT).when(consentResourceMock) - .getReceipt(); - Mockito.doReturn(ConsentAuthorizeTestConstants.AWAITING_AUTH_STATUS).when(consentResourceMock) - .getCurrentStatus(); - - JSONArray accountConsentData = ConsentRetrievalUtil.getConsentData(consentResourceMock); - Assert.assertNotNull(accountConsentData); - } - - @Test(dataProvider = "PaymentConsentDataDataProvider", dataProviderClass = ConsentExtensionDataProvider.class) - public void testGetConsentDataSetForPayments(String paymentReceipt) { - - Mockito.doReturn(configMap).when(openBankingConfigParse).getConfiguration(); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParse); - - Mockito.doReturn(ConsentExtensionConstants.PAYMENTS).when(consentResourceMock).getConsentType(); - Mockito.doReturn(paymentReceipt).when(consentResourceMock).getReceipt(); - Mockito.doReturn(ConsentAuthorizeTestConstants.CREATED_TIME).when(consentResourceMock) - .getCreatedTime(); - Mockito.doReturn(ConsentAuthorizeTestConstants.AWAITING_AUTH_STATUS).when(consentResourceMock) - .getCurrentStatus(); - - JSONArray paymentConsentData = ConsentRetrievalUtil.getConsentData(consentResourceMock); - Assert.assertNotNull(paymentConsentData); - } - - @Test - public void testGetConsentDataSetForCOF() { - - Mockito.doReturn(ConsentExtensionConstants.FUNDSCONFIRMATIONS).when(consentResourceMock).getConsentType(); - Mockito.doReturn(ConsentAuthorizeTestConstants.COF_RECEIPT).when(consentResourceMock) - .getReceipt(); - Mockito.doReturn(ConsentAuthorizeTestConstants.AWAITING_AUTH_STATUS).when(consentResourceMock) - .getCurrentStatus(); - - JSONArray cofConsentData = ConsentRetrievalUtil.getConsentData(consentResourceMock); - Assert.assertNotNull(cofConsentData); - } - - @Test - public void testGetConsentDataSetForVRPData() { - - Mockito.doReturn(ConsentExtensionConstants.VRP).when(consentResourceMock).getConsentType(); - Mockito.doReturn(ConsentAuthorizeTestConstants.VRP_INITIATION).when(consentResourceMock) - .getReceipt(); - Mockito.doReturn(ConsentAuthorizeTestConstants.AWAITING_AUTH_STATUS).when(consentResourceMock) - .getCurrentStatus(); - JSONArray vrpConsentData = ConsentRetrievalUtil.getConsentData(consentResourceMock); - Assert.assertNotNull(vrpConsentData); - } - - @Test(expectedExceptions = ConsentException.class) - public void testConsentDataWithInvalidJson() { - String invalidJsonString = "Invalid JSON String"; - Mockito.when(consentResourceMock.getReceipt()).thenReturn(invalidJsonString); - - JSONArray consentDataJSON = ConsentRetrievalUtil.getConsentData(consentResourceMock); - Assert.assertNotNull(consentDataJSON); - } - - @Test(expectedExceptions = ConsentException.class) - public void testGetConsentDataSetForVRPDataParameter() { - - Mockito.doReturn(ConsentExtensionConstants.VRP).when(consentResourceMock).getConsentType(); - Mockito.doReturn(VRP_WITHOUT_DATA).when(consentResourceMock) - .getReceipt(); - Mockito.doReturn(ConsentAuthorizeTestConstants.AWAITING_AUTH_STATUS).when(consentResourceMock) - .getCurrentStatus(); - JSONArray vrpConsentData = ConsentRetrievalUtil.getConsentData(consentResourceMock); - Assert.assertNotNull(vrpConsentData); - } - - @Test(expectedExceptions = ConsentException.class) - public void testGetConsentDataSetForVRPDataWithoutControlParameters() { - - Mockito.doReturn(ConsentExtensionConstants.VRP).when(consentResourceMock).getConsentType(); - Mockito.doReturn(ConsentAuthorizeTestConstants.VRP_WITHOUT_CONTROLPARAMETERS).when(consentResourceMock) - .getReceipt(); - Mockito.doReturn(ConsentAuthorizeTestConstants.AWAITING_AUTH_STATUS).when(consentResourceMock) - .getCurrentStatus(); - - JSONArray vrpConsentData = ConsentRetrievalUtil.getConsentData(consentResourceMock); - Assert.assertNotNull(vrpConsentData); - } - - @Test(expectedExceptions = ConsentException.class) - public void testGetConsentDataSetForAccount() { - - Mockito.doReturn(ConsentExtensionConstants.ACCOUNTS).when(consentResourceMock).getConsentType(); - Mockito.doReturn(ConsentAuthorizeTestConstants.INVALID_INITIATION_OBJECT).when(consentResourceMock) - .getReceipt(); - Mockito.doReturn(ConsentAuthorizeTestConstants.AWAITING_AUTH_STATUS).when(consentResourceMock) - .getCurrentStatus(); - - JSONArray accountConsentData = ConsentRetrievalUtil.getConsentData(consentResourceMock); - Assert.assertNotNull(accountConsentData); - } - - @Test(expectedExceptions = ConsentException.class) - public void testGetConsentDataSetForCOFs() { - - Mockito.doReturn(ConsentExtensionConstants.FUNDSCONFIRMATIONS).when(consentResourceMock).getConsentType(); - Mockito.doReturn(ConsentAuthorizeTestConstants.INVALID_COF_RECEIPT).when(consentResourceMock) - .getReceipt(); - Mockito.doReturn(ConsentAuthorizeTestConstants.AWAITING_AUTH_STATUS).when(consentResourceMock) - .getCurrentStatus(); - - JSONArray cofConsentData = ConsentRetrievalUtil.getConsentData(consentResourceMock); - Assert.assertNotNull(cofConsentData); - } - - @Test(expectedExceptions = ConsentException.class) - public void testGetConsentDataSetForCOFNull() { - - Mockito.doReturn(ConsentExtensionConstants.FUNDSCONFIRMATIONS).when(consentResourceMock).getConsentType(); - Mockito.doReturn(ConsentAuthorizeTestConstants.NULL_COF_RECEIPT).when(consentResourceMock) - .getReceipt(); - Mockito.doReturn(ConsentAuthorizeTestConstants.AWAITING_AUTH_STATUS).when(consentResourceMock) - .getCurrentStatus(); - - JSONArray cofConsentData = ConsentRetrievalUtil.getConsentData(consentResourceMock); - Assert.assertNotNull(cofConsentData); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/AuthServletTest.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/AuthServletTest.java deleted file mode 100644 index 3fe0b66a..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/AuthServletTest.java +++ /dev/null @@ -1,181 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authservlet.impl; - -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.utils.AuthServletTestConstants; -import org.json.JSONObject; -import org.junit.Assert; -import org.mockito.Mock; -import org.mockito.Mockito; -import org.mockito.MockitoAnnotations; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -import java.util.Map; -import java.util.ResourceBundle; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpSession; - -/** - * Test class for OB Auth Servlet. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -public class AuthServletTest { - - OBDefaultAuthServletImpl obAuthServlet; - @Mock - HttpServletRequest httpServletRequestMock; - @Mock - ResourceBundle resourceBundle; - - @BeforeClass - public void initClass() { - - MockitoAnnotations.initMocks(this); - - obAuthServlet = new OBDefaultAuthServletImpl(); - httpServletRequestMock = Mockito.mock(HttpServletRequest.class); - resourceBundle = Mockito.mock(ResourceBundle.class); - } - - @Test - public void testUpdateRequestAttributeForAccounts() { - - JSONObject accountObj = new JSONObject(AuthServletTestConstants.ACCOUNT_DATA); - - Map requestAttributes = obAuthServlet.updateRequestAttribute(httpServletRequestMock, - accountObj, resourceBundle); - - Assert.assertFalse(requestAttributes.isEmpty()); - Assert.assertTrue(requestAttributes.containsKey(ConsentExtensionConstants.DATA_REQUESTED)); - } - - @Test - public void testUpdateRequestAttributeForCOF() { - - JSONObject cofObj = new JSONObject(AuthServletTestConstants.COF_DATA); - - Map requestAttributes = obAuthServlet.updateRequestAttribute(httpServletRequestMock, - cofObj, resourceBundle); - - Assert.assertFalse(requestAttributes.isEmpty()); - Assert.assertTrue(requestAttributes.containsKey(ConsentExtensionConstants.DATA_REQUESTED)); - } - - @Test - public void testUpdateRequestAttributeForPayments() { - - JSONObject paymentObj = new JSONObject(AuthServletTestConstants.PAYMENT_DATA); - HttpSession session = Mockito.mock(HttpSession.class); - Mockito.doReturn(session).when(httpServletRequestMock).getSession(); - - Map requestAttributes = obAuthServlet.updateRequestAttribute(httpServletRequestMock, - paymentObj, resourceBundle); - - Assert.assertFalse(requestAttributes.isEmpty()); - Assert.assertTrue(requestAttributes.containsKey(ConsentExtensionConstants.DATA_REQUESTED)); - } - - @Test - public void testUpdateRequestAttributeForPaymentsWithoutDebtorAccInPayload() { - - JSONObject paymentObj = new JSONObject(AuthServletTestConstants.PAYMENT_DATA_WITHOUT_DEBTOR_ACC); - HttpSession session = Mockito.mock(HttpSession.class); - Mockito.doReturn(session).when(httpServletRequestMock).getSession(); - - Map requestAttributes = obAuthServlet.updateRequestAttribute(httpServletRequestMock, - paymentObj, resourceBundle); - - Assert.assertTrue(requestAttributes.isEmpty()); - Assert.assertFalse(requestAttributes.containsKey(ConsentExtensionConstants.DATA_REQUESTED)); - } - - @Test - public void testUpdateRequestAttributeForVRP() { - - JSONObject paymentObj = new JSONObject(AuthServletTestConstants.VRP_DATA); - HttpSession session = Mockito.mock(HttpSession.class); - Mockito.doReturn(session).when(httpServletRequestMock).getSession(); - - Map requestAttributes = obAuthServlet.updateRequestAttribute(httpServletRequestMock, - paymentObj, resourceBundle); - - Assert.assertFalse(requestAttributes.isEmpty()); - Assert.assertTrue(requestAttributes.containsKey(ConsentExtensionConstants.DATA_REQUESTED)); - } - - @Test - public void testUpdateRequestAttributeForVRPWithoutDebtorAcc() { - - JSONObject paymentObj = new JSONObject(AuthServletTestConstants.VRP_DATA_WITHOUT_DEBTOR_ACC); - HttpSession session = Mockito.mock(HttpSession.class); - Mockito.doReturn(session).when(httpServletRequestMock).getSession(); - - Map requestAttributes = obAuthServlet.updateRequestAttribute(httpServletRequestMock, - paymentObj, resourceBundle); - - Assert.assertFalse(requestAttributes.isEmpty()); - Assert.assertTrue(requestAttributes.containsKey(ConsentExtensionConstants.DATA_REQUESTED)); - } - - @Test - public void testUpdateRequestAttributeForNonExistingType() { - - JSONObject object = new JSONObject(AuthServletTestConstants.JSON_WITH_TYPE); - - Map requestAttributes = obAuthServlet.updateRequestAttribute(httpServletRequestMock, - object, resourceBundle); - - Assert.assertTrue(requestAttributes.isEmpty()); - } - - @Test - public void testUpdateConsentData() { - - String param = "Test_parameter"; - Mockito.doReturn(param).when(httpServletRequestMock).getParameter(Mockito.anyString()); - HttpSession session = Mockito.mock(HttpSession.class); - Mockito.doReturn(session).when(httpServletRequestMock).getSession(); - - Map consentData = obAuthServlet.updateConsentData(httpServletRequestMock); - Assert.assertFalse(consentData.isEmpty()); - Assert.assertTrue(consentData.containsKey(ConsentExtensionConstants.ACCOUNT_IDS)); - Assert.assertFalse(consentData.containsKey(ConsentExtensionConstants.PAYMENT_ACCOUNT)); - Assert.assertFalse(consentData.containsKey(ConsentExtensionConstants.COF_ACCOUNT)); - } - - @Test - public void testUpdateConsentMetaData() { - - Map consentMetadata = obAuthServlet.updateConsentMetaData(httpServletRequestMock); - - Assert.assertTrue(consentMetadata.isEmpty()); - } - - @Test - public void testUpdateSessionAttribute() { - - Map sessionAttributes = obAuthServlet.updateSessionAttribute(httpServletRequestMock, - new JSONObject(), resourceBundle); - - Assert.assertTrue(sessionAttributes.isEmpty()); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/ConsentMgrAuthServletImplTest.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/ConsentMgrAuthServletImplTest.java deleted file mode 100644 index 9427bea5..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/authservlet/impl/ConsentMgrAuthServletImplTest.java +++ /dev/null @@ -1,88 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.authservlet.impl; - -import com.wso2.openbanking.accelerator.consent.extensions.authservlet.impl.util.Constants; -import com.wso2.openbanking.accelerator.consent.extensions.authservlet.model.OBAuthServletInterface; -import org.json.JSONArray; -import org.json.JSONObject; -import org.mockito.Mockito; -import org.testng.Assert; -import org.testng.annotations.Test; - -import java.util.List; -import java.util.Map; -import java.util.ResourceBundle; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpSession; - -/** - * Test for consent management auth servlet implementation. - */ -public class ConsentMgrAuthServletImplTest { - OBAuthServletInterface uut = new ConsentMgrAuthServletImpl(); - - @Test - public void testUpdateRequestAttribute() { - JSONArray scopeArray = new JSONArray(); - scopeArray.put("scope1"); - scopeArray.put("scope2"); - scopeArray.put("scope3"); - - JSONObject dataset = new JSONObject(); - dataset.put("openid_scopes", scopeArray); - - // mock - HttpServletRequest servletRequestMock = Mockito.mock(HttpServletRequest.class); - HttpSession httpSessionMock = Mockito.mock(HttpSession.class); - ResourceBundle resourceBundleMock = Mockito.mock(ResourceBundle.class); - - // when - Mockito.when(httpSessionMock.getAttribute("displayScopes")).thenReturn(true); - Mockito.when(servletRequestMock.getSession()).thenReturn(httpSessionMock); - - // assert - Map returnMap = uut - .updateRequestAttribute(servletRequestMock, dataset, resourceBundleMock); - Assert.assertNotNull(returnMap); - List oidScopes = (List) returnMap.get(Constants.OIDC_SCOPES); - Assert.assertTrue(oidScopes.contains("scope1")); - Assert.assertTrue(oidScopes.contains("scope2")); - Assert.assertTrue(oidScopes.contains("scope3")); - } - - @Test(description = "when displayScopes is false, OIDCScopes should be null") - public void testUpdateRequestAttributeWithFalseDisplayScope() { - // mock - HttpServletRequest servletRequestMock = Mockito.mock(HttpServletRequest.class); - HttpSession httpSessionMock = Mockito.mock(HttpSession.class); - ResourceBundle resourceBundleMock = Mockito.mock(ResourceBundle.class); - - // when - Mockito.when(httpSessionMock.getAttribute("displayScopes")).thenReturn(false); - Mockito.when(servletRequestMock.getSession()).thenReturn(httpSessionMock); - - // assert - Map returnMap = uut - .updateRequestAttribute(servletRequestMock, null, resourceBundleMock); - Assert.assertNotNull(returnMap); - Assert.assertNull(returnMap.get(Constants.OIDC_SCOPES)); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/authenticator/CIBAPushAuthenticatorTests.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/authenticator/CIBAPushAuthenticatorTests.java deleted file mode 100644 index 9a84e09e..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/ciba/authenticator/CIBAPushAuthenticatorTests.java +++ /dev/null @@ -1,258 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.ciba.authenticator; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentData; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentCache; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionUtils; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import com.wso2.openbanking.accelerator.identity.cache.IdentityCache; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import net.minidev.json.JSONObject; -import org.mockito.Mock; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.annotations.DataProvider; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.application.authentication.framework.cache.AuthenticationContextCache; -import org.wso2.carbon.identity.application.authentication.framework.config.model.ApplicationConfig; -import org.wso2.carbon.identity.application.authentication.framework.config.model.SequenceConfig; -import org.wso2.carbon.identity.application.authentication.framework.context.AuthenticationContext; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticationRequest; -import org.wso2.carbon.identity.application.authentication.framework.util.FrameworkUtils; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.oauth.cache.SessionDataCache; -import org.wso2.carbon.identity.oauth.cache.SessionDataCacheEntry; -import org.wso2.carbon.identity.oauth2.model.OAuth2Parameters; - -import java.io.Serializable; -import java.util.HashMap; -import java.util.Map; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; - -import static org.mockito.Mockito.doNothing; -import static org.mockito.Mockito.doReturn; -import static org.mockito.Mockito.mock; -import static org.mockito.Mockito.spy; -import static org.mockito.Mockito.when; -import static org.powermock.api.mockito.PowerMockito.mockStatic; -import static org.testng.Assert.assertEquals; -import static org.testng.Assert.assertNotNull; -import static org.testng.Assert.assertTrue; - -/** - * Test class for CIBAPushAuthenticator. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({SessionDataCache.class, AuthenticationContextCache.class, ConsentExtensionUtils.class, - ConsentCache.class, ConsentData.class, IdentityCommonUtil.class, FrameworkUtils.class, - AuthenticationContext.class, SequenceConfig.class, ServiceProvider.class, ApplicationConfig.class, - AuthenticationRequest.class, SessionDataCacheEntry.class, IdentityCache.class, OAuth2Parameters.class, - HttpServletRequest.class, ConsentCoreServiceImpl.class, ConsentExtensionsDataHolder.class, - OpenBankingConfigurationService.class, OpenBankingConfigParser.class}) -public class CIBAPushAuthenticatorTests extends PowerMockTestCase { - - private final String dummyString = "dummyString"; - - @Mock - OpenBankingConfigurationService openBankingConfigurationService; - - @Mock - ConsentExtensionsDataHolder consentExtensionsDataHolder; - - @DataProvider(name = "splitQueryParams") - public Object[][] getSplitQueryParams() { - - String validQueryParms = "key1=val1&key2=val2"; - Map validQueryParamMap = new HashMap<>(); - validQueryParamMap.put("key1", "val1"); - validQueryParamMap.put("key2", "val2"); - - String invalidQueryParams = "key1:val1&key2:val2"; - Map invalidQueryParamMap = new HashMap<>(); - invalidQueryParamMap.put("key1:val1", null); - invalidQueryParamMap.put("key2:val2", null); - - return new Object[][]{ - {validQueryParms, validQueryParamMap}, - {invalidQueryParams, invalidQueryParamMap} - }; - } - - @Test(dataProvider = "splitQueryParams") - public void splitQueryValidQueryTest(String dummyQueryParms, Map queryParamMap) throws Exception { - CIBAPushAuthenticator cibaPushAuthenticator = new CIBAPushAuthenticator(); - Map splitQueryParamMap = cibaPushAuthenticator.splitQuery(dummyQueryParms); - - assertEquals(queryParamMap, splitQueryParamMap); - } - - @Test - public void handlePreConsentTest() { - AuthenticationContext mockAuthnCtxt = spy(AuthenticationContext.class); - - Map params = new HashMap() { - { - put(CIBAPushAuthenticatorConstants.LOGIN_HINT, dummyString); - put(CIBAPushAuthenticatorConstants.REQUEST_OBJECT, dummyString); - put(CIBAPushAuthenticatorConstants.SCOPE, dummyString); - } - }; - - Map queryParamMap = new HashMap<>(); - String[] queryParamArray = new String[1]; - queryParamArray[0] = dummyString; - queryParamMap.put(CIBAPushAuthenticatorConstants.NONCE, queryParamArray); - - SequenceConfig mockSequenceConfig = mock(SequenceConfig.class); - ApplicationConfig mockApplicationConfig = mock(ApplicationConfig.class); - ServiceProvider mockServiceProvider = mock(ServiceProvider.class); - AuthenticationRequest mockAuthenticationRequest = mock(AuthenticationRequest.class); - - when(mockAuthnCtxt.getSequenceConfig()).thenReturn(mockSequenceConfig); - when(mockSequenceConfig.getApplicationConfig()).thenReturn(mockApplicationConfig); - when(mockApplicationConfig.getServiceProvider()).thenReturn(mockServiceProvider); - when(mockServiceProvider.getApplicationName()).thenReturn(dummyString); - when(mockAuthnCtxt.getAuthenticationRequest()).thenReturn(mockAuthenticationRequest); - when(mockAuthenticationRequest.getRequestQueryParams()).thenReturn(queryParamMap); - - int initialSize = mockAuthnCtxt.getEndpointParams().size(); - CIBAPushAuthenticator cibaPushAuthenticator = new CIBAPushAuthenticator(); - cibaPushAuthenticator.handlePreConsent(mockAuthnCtxt, params); - int finalSize = mockAuthnCtxt.getEndpointParams().size(); - - assertTrue(finalSize > initialSize); - } - - @Test - public void setMetadataTest() throws Exception { - CIBAPushAuthenticator mockAuthenticator = spy(new CIBAPushAuthenticatorMock()); - - mockStatic(SessionDataCache.class); - mockStatic(AuthenticationContextCache.class); - mockStatic(FrameworkUtils.class); - - SessionDataCache sessionDataCache = mock(SessionDataCache.class); - SessionDataCacheEntry sessionDataCacheEntry = mock(SessionDataCacheEntry.class); - AuthenticationContextCache authenticationContextCache = mock(AuthenticationContextCache.class); - HttpServletRequest httpServletRequest = mock(HttpServletRequest.class); - HttpServletResponse httpServletResponse = mock(HttpServletResponse.class); - when(httpServletRequest.getParameter(CIBAPushAuthenticatorConstants.BINDING_MESSAGE)).thenReturn(dummyString); - - when(mockAuthenticator.splitQuery(Mockito.anyString())).thenReturn(new HashMap<>()); - when(FrameworkUtils.getQueryStringWithFrameworkContextId - (Mockito.anyObject(), Mockito.anyObject(), Mockito.anyObject())).thenReturn(dummyString); - when(SessionDataCache.getInstance()).thenReturn(sessionDataCache); - when(sessionDataCache.getValueFromCache(Mockito.anyObject())).thenReturn(sessionDataCacheEntry); - when(AuthenticationContextCache.getInstance()).thenReturn(authenticationContextCache); - - doNothing().when(mockAuthenticator).handlePreConsent(Mockito.anyObject(), Mockito.anyObject()); - doNothing().when(authenticationContextCache).addToCache(Mockito.anyObject(), Mockito.anyObject()); - JSONObject jsonObject = new JSONObject(); - jsonObject.put(dummyString, dummyString); - doReturn(jsonObject).when(mockAuthenticator).retrieveConsent(Mockito.anyObject(), - Mockito.anyObject(), Mockito.anyString()); - - assertNotNull(mockAuthenticator.getAdditionalInfo(httpServletRequest, httpServletResponse, - dummyString)); - } - - @Test - public void retrieveConsentTest() throws OpenBankingException { - - CIBAPushAuthenticator mockAuthenticator = spy(new CIBAPushAuthenticatorMock()); - Map configs = new HashMap<>(); - configs.put("Consent.PreserveConsentLink", "true"); - mockStatic(ConsentExtensionUtils.class); - mockStatic(ConsentCoreServiceImpl.class); - mockStatic(ConsentExtensionsDataHolder.class); - - OpenBankingConfigParser openBankingConfigParserMock = Mockito.mock(OpenBankingConfigParser.class); - Mockito.doReturn(configs).when(openBankingConfigParserMock).getConfiguration(); - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - - mockStatic(ConsentCache.class); - mockStatic(ConsentData.class); - mockStatic(IdentityCommonUtil.class); - - ConsentData consentData = mock(ConsentData.class); - IdentityCache identityCache = mock(IdentityCache.class); - SessionDataCacheEntry cacheEntry = mock(SessionDataCacheEntry.class); - OAuth2Parameters oAuth2Parameters = mock(OAuth2Parameters.class); - HttpServletRequest httpServletRequest = mock(HttpServletRequest.class); - IdentityCommonUtil identityCommonUtil = mock(IdentityCommonUtil.class); - - when(ConsentCache.getInstance()).thenReturn(identityCache); - when(ConsentCache.getCacheEntryFromSessionDataKey(Mockito.anyString())).thenReturn(cacheEntry); - when(cacheEntry.getoAuth2Parameters()).thenReturn(oAuth2Parameters); - when(oAuth2Parameters.getRedirectURI()).thenReturn(dummyString); - when(oAuth2Parameters.getClientId()).thenReturn(dummyString); - when(oAuth2Parameters.getState()).thenReturn(dummyString); - when(identityCommonUtil.getRegulatoryFromSPMetaData(dummyString)).thenReturn(true); - when(consentData.getType()).thenReturn(dummyString); - when(consentData.getApplication()).thenReturn(dummyString); - - Map sensitiveDataMap = new HashMap<>(); - Map headers = new HashMap<>(); - sensitiveDataMap.put(CIBAPushAuthenticatorConstants.IS_ERROR, "false"); - sensitiveDataMap.put(CIBAPushAuthenticatorConstants.LOGGED_IN_USER, dummyString); - sensitiveDataMap.put(CIBAPushAuthenticatorConstants.SP_QUERY_PARAMS, dummyString); - sensitiveDataMap.put(CIBAPushAuthenticatorConstants.SCOPE, dummyString); - - when(ConsentExtensionUtils.getSensitiveDataWithConsentKey(Mockito.anyString())).thenReturn(sensitiveDataMap); - when(ConsentExtensionUtils.getHeaders(httpServletRequest)).thenReturn(headers); - when(mockAuthenticator.createConsentData(Mockito.anyString(), Mockito.anyString(), Mockito.anyString(), - Mockito.anyString(), Mockito.anyString(), Mockito.anyObject())).thenReturn(consentData); - - doNothing().when(consentData).setSensitiveDataMap(Mockito.anyObject()); - doNothing().when(consentData).setRedirectURI(Mockito.anyObject()); - doNothing().when(consentData).setRegulatory(Mockito.anyObject()); - doNothing().when(mockAuthenticator).executeRetrieval(Mockito.anyObject(), Mockito.anyObject()); - - assertNotNull(mockAuthenticator.retrieveConsent(Mockito.anyObject(), Mockito.anyObject(), Mockito.anyString())); - - } - -} - -class CIBAPushAuthenticatorMock extends CIBAPushAuthenticator { - - @Override - protected AuthenticationContext getAutenticationContext(String sessionDataKey) { - - return mock(AuthenticationContext.class); - } - - @Override - protected ConsentData createConsentData(String sessionDataKey, String loggedInUser, String spQueryParams, - String scopeString, String app, HttpServletRequest request) { - return mock(ConsentData.class); - } - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidatorTests.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidatorTests.java deleted file mode 100644 index f6afd117..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/common/idempotency/IdempotencyValidatorTests.java +++ /dev/null @@ -1,303 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.common.idempotency; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import com.wso2.openbanking.accelerator.consent.extensions.manage.model.ConsentManageData; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import org.mockito.Mock; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -import java.time.OffsetDateTime; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Map; -import java.util.UUID; - -/** - * Test class for IdempotencyValidator. - */ -@PrepareForTest({OpenBankingConfigParser.class, ConsentExtensionsDataHolder.class}) -@PowerMockIgnore("jdk.internal.reflect.*") -public class IdempotencyValidatorTests extends PowerMockTestCase { - - @Mock - private ConsentManageData consentManageData; - private ConsentCoreServiceImpl consentCoreServiceImpl; - private ArrayList consentIdList; - private Map attributeList; - private String consentId; - private Map configs; - private Map headers; - private static final String CLIENT_ID = "testClientId"; - - private static final String PAYLOAD = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"Yes\",\n" + - " \"Initiation\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"165.88\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"UK.OBIE.SortCodeAccountNumber\",\n" + - " \"Identification\": \"08080021325698\",\n" + - " \"Name\": \"ACME Inc\",\n" + - " \"SecondaryIdentification\": \"0002\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"FRESCO-101\",\n" + - " \"Unstructured\": \"Internal ops code 5120101\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " }\n" + - " }\n" + - "}"; - - private static final String DIFFERENT_PAYLOAD = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"No\",\n" + - " \"Initiation\": {\n" + - " \"InstructionIdentification\": \"ACME413\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"165.88\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"UK.OBIE.SortCodeAccountNumber\",\n" + - " \"Identification\": \"08080021325698\",\n" + - " \"Name\": \"ACME Inc\",\n" + - " \"SecondaryIdentification\": \"0002\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"FRESCO-101\",\n" + - " \"Unstructured\": \"Internal ops code 5120101\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " }\n" + - " }\n" + - "}"; - - - @BeforeClass - public void beforeTest() { - configs = new HashMap<>(); - - headers = new HashMap<>(); - headers.put(IdempotencyConstants.X_IDEMPOTENCY_KEY, "123456"); - headers.put(IdempotencyConstants.CONTENT_TYPE_TAG, "application/json"); - - consentManageData = Mockito.mock(ConsentManageData.class); - consentCoreServiceImpl = Mockito.mock(ConsentCoreServiceImpl.class); - - consentId = UUID.randomUUID().toString(); - consentIdList = new ArrayList<>(); - consentIdList.add(consentId); - - attributeList = new HashMap<>(); - attributeList.put(consentId, "123456"); - } - - @BeforeMethod - public void beforeMethod() { - OpenBankingConfigParser openBankingConfigParserMock = PowerMockito.mock(OpenBankingConfigParser.class); - Mockito.doReturn(configs).when(openBankingConfigParserMock).getConfiguration(); - Mockito.doReturn(true).when(openBankingConfigParserMock).isIdempotencyValidationEnabled(); - Mockito.doReturn("1").when(openBankingConfigParserMock).getIdempotencyAllowedTime(); - ConsentExtensionsDataHolder consentExtensionsDataHolderMock = PowerMockito - .mock(ConsentExtensionsDataHolder.class); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - - PowerMockito.mockStatic(ConsentExtensionsDataHolder.class); - PowerMockito.when(ConsentExtensionsDataHolder.getInstance()).thenReturn(consentExtensionsDataHolderMock); - PowerMockito.when(consentExtensionsDataHolderMock.getConsentCoreService()).thenReturn(consentCoreServiceImpl); - } - - @Test - public void testValidateIdempotency() throws ConsentManagementException, IdempotencyValidationException { - OffsetDateTime offsetDateTime = OffsetDateTime.now(); - - Mockito.doReturn(consentIdList).when(consentCoreServiceImpl) - .getConsentIdByConsentAttributeNameAndValue(Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(getConsent(offsetDateTime.toEpochSecond())).when(consentCoreServiceImpl) - .getDetailedConsent(Mockito.anyString()); - Mockito.doReturn(headers).when(consentManageData).getHeaders(); - Mockito.doReturn(CLIENT_ID).when(consentManageData).getClientId(); - Mockito.doReturn(PAYLOAD).when(consentManageData).getPayload(); - IdempotencyValidationResult result = new IdempotencyValidator().validateIdempotency(consentManageData); - - Assert.assertTrue(result.isIdempotent()); - Assert.assertTrue(result.isValid()); - Assert.assertNotNull(result.getConsent()); - Assert.assertEquals(consentId, result.getConsentId()); - } - - @Test(expectedExceptions = IdempotencyValidationException.class) - public void testValidateIdempotencyForRequestsWithoutPayload() throws ConsentManagementException, - IdempotencyValidationException { - OffsetDateTime offsetDateTime = OffsetDateTime.now(); - - Mockito.doReturn(attributeList).when(consentCoreServiceImpl).getConsentAttributesByName(Mockito.anyString()); - Mockito.doReturn(getConsent(offsetDateTime.toEpochSecond())).when(consentCoreServiceImpl) - .getDetailedConsent(Mockito.anyString()); - Mockito.doReturn(headers).when(consentManageData).getHeaders(); - Mockito.doReturn(CLIENT_ID).when(consentManageData).getClientId(); - Mockito.doReturn("{}").when(consentManageData).getPayload(); - Mockito.doReturn("{}").when(consentManageData).getPayload(); - Mockito.doReturn("/payments/".concat(consentId)).when(consentManageData).getRequestPath(); - new IdempotencyValidator().validateIdempotency(consentManageData); - } - - @Test - public void testValidateIdempotencyWithoutIdempotencyKeyValue() throws IdempotencyValidationException { - - Mockito.doReturn(new HashMap<>()).when(consentManageData).getHeaders(); - Mockito.doReturn(CLIENT_ID).when(consentManageData).getClientId(); - Mockito.doReturn(PAYLOAD).when(consentManageData).getPayload(); - IdempotencyValidationResult result = new IdempotencyValidator().validateIdempotency(consentManageData); - - Assert.assertFalse(result.isIdempotent()); - } - - @Test - public void testValidateIdempotencyWithoutRequest() throws IdempotencyValidationException { - Mockito.doReturn(headers).when(consentManageData).getHeaders(); - Mockito.doReturn(CLIENT_ID).when(consentManageData).getClientId(); - Mockito.doReturn("").when(consentManageData).getPayload(); - IdempotencyValidationResult result = new IdempotencyValidator().validateIdempotency(consentManageData); - - Assert.assertFalse(result.isIdempotent()); - } - - @Test - public void testValidateIdempotencyRetrievingAttributesWithException() - throws ConsentManagementException, IdempotencyValidationException { - - Mockito.doThrow(ConsentManagementException.class).when(consentCoreServiceImpl) - .getConsentIdByConsentAttributeNameAndValue(Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(headers).when(consentManageData).getHeaders(); - Mockito.doReturn(CLIENT_ID).when(consentManageData).getClientId(); - Mockito.doReturn(PAYLOAD).when(consentManageData).getPayload(); - IdempotencyValidationResult result = new IdempotencyValidator().validateIdempotency(consentManageData); - - Assert.assertFalse(result.isIdempotent()); - } - - @Test - public void testValidateIdempotencyWithoutAttribute() - throws ConsentManagementException, IdempotencyValidationException { - - Mockito.doReturn(new ArrayList<>()).when(consentCoreServiceImpl) - .getConsentIdByConsentAttributeNameAndValue(Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(headers).when(consentManageData).getHeaders(); - Mockito.doReturn(CLIENT_ID).when(consentManageData).getClientId(); - Mockito.doReturn(PAYLOAD).when(consentManageData).getPayload(); - IdempotencyValidationResult result = new IdempotencyValidator().validateIdempotency(consentManageData); - - Assert.assertFalse(result.isIdempotent()); - } - - @Test(expectedExceptions = IdempotencyValidationException.class) - public void testValidateIdempotencyWithNullConsentRequest() - throws ConsentManagementException, IdempotencyValidationException { - - Mockito.doReturn(consentIdList).when(consentCoreServiceImpl) - .getConsentIdByConsentAttributeNameAndValue(Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(headers).when(consentManageData).getHeaders(); - Mockito.doReturn(CLIENT_ID).when(consentManageData).getClientId(); - Mockito.doReturn(PAYLOAD).when(consentManageData).getPayload(); - Mockito.doReturn(null).when(consentCoreServiceImpl).getDetailedConsent(Mockito.anyString()); - new IdempotencyValidator().validateIdempotency(consentManageData); - } - - @Test(expectedExceptions = IdempotencyValidationException.class) - public void testValidateIdempotencyWithNonMatchingClientId() - throws ConsentManagementException, IdempotencyValidationException { - - Mockito.doReturn(consentIdList).when(consentCoreServiceImpl) - .getConsentIdByConsentAttributeNameAndValue(Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(headers).when(consentManageData).getHeaders(); - Mockito.doReturn("sampleClientID").when(consentManageData).getClientId(); - Mockito.doReturn(PAYLOAD).when(consentManageData).getPayload(); - Mockito.doReturn(null).when(consentCoreServiceImpl).getDetailedConsent(Mockito.anyString()); - new IdempotencyValidator().validateIdempotency(consentManageData); - } - - @Test(expectedExceptions = IdempotencyValidationException.class) - public void testValidateIdempotencyAfterAllowedTime() - throws ConsentManagementException, IdempotencyValidationException { - - OffsetDateTime offsetDateTime = OffsetDateTime.now().minusHours(2); - - Mockito.doReturn(consentIdList).when(consentCoreServiceImpl) - .getConsentIdByConsentAttributeNameAndValue(Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(getConsent(offsetDateTime.toEpochSecond())).when(consentCoreServiceImpl) - .getDetailedConsent(Mockito.anyString()); - Mockito.doReturn(headers).when(consentManageData).getHeaders(); - Mockito.doReturn(CLIENT_ID).when(consentManageData).getClientId(); - Mockito.doReturn(PAYLOAD).when(consentManageData).getPayload(); - new IdempotencyValidator().validateIdempotency(consentManageData); - } - - @Test(expectedExceptions = IdempotencyValidationException.class) - public void testValidateIdempotencyWithNonMatchingPayload() - throws ConsentManagementException, IdempotencyValidationException { - - OffsetDateTime offsetDateTime = OffsetDateTime.now(); - - Mockito.doReturn(consentIdList).when(consentCoreServiceImpl) - .getConsentIdByConsentAttributeNameAndValue(Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(getConsent(offsetDateTime.toEpochSecond())).when(consentCoreServiceImpl) - .getDetailedConsent(Mockito.anyString()); - Mockito.doReturn(headers).when(consentManageData).getHeaders(); - Mockito.doReturn(CLIENT_ID).when(consentManageData).getClientId(); - Mockito.doReturn(DIFFERENT_PAYLOAD).when(consentManageData).getPayload(); - new IdempotencyValidator().validateIdempotency(consentManageData); - - } - - private DetailedConsentResource getConsent(long createdTime) { - DetailedConsentResource consent = new DetailedConsentResource(); - consent.setConsentID(consentId); - consent.setReceipt(PAYLOAD); - consent.setClientID(CLIENT_ID); - consent.setCreatedTime(createdTime); - return consent; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/event/executors/ConsentAmendmentHistoryEventExecutorTests.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/event/executors/ConsentAmendmentHistoryEventExecutorTests.java deleted file mode 100644 index 0eb78099..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/event/executors/ConsentAmendmentHistoryEventExecutorTests.java +++ /dev/null @@ -1,149 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.event.executors; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.event.executor.model.OBEvent; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import org.apache.logging.log4j.LogManager; -import org.apache.logging.log4j.Logger; -import org.mockito.Mockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -import java.io.ByteArrayOutputStream; -import java.io.PrintStream; -import java.util.HashMap; -import java.util.Map; -import java.util.UUID; - -import static org.powermock.api.mockito.PowerMockito.mock; -import static org.powermock.api.mockito.PowerMockito.mockStatic; -import static org.powermock.api.mockito.PowerMockito.when; -/** - * Test class for ConsentAmendmentHistoryEventExecutor. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({OpenBankingConfigParser.class, ConsentExtensionsDataHolder.class}) -public class ConsentAmendmentHistoryEventExecutorTests extends PowerMockTestCase { - - private static ByteArrayOutputStream outContent; - private static Logger logger = null; - private static PrintStream printStream; - private ConsentCoreServiceImpl consentCoreServiceImpl; - private String sampleConsentID; - - @BeforeClass - public void initTest() { - - consentCoreServiceImpl = Mockito.mock(ConsentCoreServiceImpl.class); - } - - @BeforeClass - public void beforeTests() { - - sampleConsentID = UUID.randomUUID().toString(); - outContent = new ByteArrayOutputStream(); - printStream = new PrintStream(outContent); - System.setOut(printStream); - logger = LogManager.getLogger(ConsentAmendmentHistoryEventExecutorTests.class); - } - - @Test - public void testProcessEventSuccess() throws Exception { - - ConsentAmendmentHistoryEventExecutor consentAmendmentHistoryEventExecutorSpy = - Mockito.spy(new ConsentAmendmentHistoryEventExecutor()); - - outContent.reset(); - - OpenBankingConfigParser openBankingConfigParserMock = mock(OpenBankingConfigParser.class); - ConsentExtensionsDataHolder consentExtensionsDataHolderMock = mock(ConsentExtensionsDataHolder.class); - mockStatic(OpenBankingConfigParser.class); - when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - when(openBankingConfigParserMock.isConsentAmendmentHistoryEnabled()).thenReturn(true); - - mockStatic(ConsentExtensionsDataHolder.class); - when(ConsentExtensionsDataHolder.getInstance()).thenReturn(consentExtensionsDataHolderMock); - when(consentExtensionsDataHolderMock.getConsentCoreService()).thenReturn(consentCoreServiceImpl); - - Map eventData = new HashMap<>(); - eventData.put("Reason", "Amended by the user"); - eventData.put("ConsentId", sampleConsentID); - eventData.put("ClientId", "dummyClientId"); - - Map consentDataMap = new HashMap<>(); - consentDataMap.put("ConsentResource", new DetailedConsentResource()); - consentDataMap.put("ConsentAmendmentHistory", new DetailedConsentResource()); - consentDataMap.put("ConsentAmendmentTime", System.currentTimeMillis()); - eventData.put("ConsentDataMap", consentDataMap); - - OBEvent obEvent = new OBEvent("amended", eventData); - Mockito.doReturn(true).when(consentCoreServiceImpl) - .storeConsentAmendmentHistory(Mockito.anyString(), Mockito.anyObject(), Mockito.anyObject()); - consentAmendmentHistoryEventExecutorSpy.processEvent(obEvent); - - Assert.assertTrue(outContent.toString().contains("Consent Amendment History of consentID:")); - } - - @Test - public void testProcessEventFailure() throws ConsentManagementException { - - ConsentAmendmentHistoryEventExecutor consentAmendmentHistoryEventExecutorSpy = - Mockito.spy(new ConsentAmendmentHistoryEventExecutor()); - - outContent.reset(); - - OpenBankingConfigParser openBankingConfigParserMock = mock(OpenBankingConfigParser.class); - ConsentExtensionsDataHolder consentExtensionsDataHolderMock = mock(ConsentExtensionsDataHolder.class); - mockStatic(OpenBankingConfigParser.class); - when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - when(openBankingConfigParserMock.isConsentAmendmentHistoryEnabled()).thenReturn(true); - - mockStatic(ConsentExtensionsDataHolder.class); - when(ConsentExtensionsDataHolder.getInstance()).thenReturn(consentExtensionsDataHolderMock); - when(consentExtensionsDataHolderMock.getConsentCoreService()).thenReturn(consentCoreServiceImpl); - - Map eventData = new HashMap<>(); - eventData.put("Reason", "Amended by the user"); - eventData.put("ConsentId", sampleConsentID); - eventData.put("ClientId", "dummyClientId"); - - Map consentDataMap = new HashMap<>(); - consentDataMap.put("ConsentResource", new DetailedConsentResource()); - consentDataMap.put("ConsentAmendmentHistory", new DetailedConsentResource()); - consentDataMap.put("ConsentAmendmentTime", System.currentTimeMillis()); - eventData.put("ConsentDataMap", consentDataMap); - - OBEvent obEvent = new OBEvent("amended", eventData); - Mockito.doThrow(ConsentManagementException.class).when(consentCoreServiceImpl) - .storeConsentAmendmentHistory(Mockito.anyString(), Mockito.anyObject(), Mockito.anyObject()); - consentAmendmentHistoryEventExecutorSpy.processEvent(obEvent); - - Assert.assertTrue(outContent.toString().contains("An error occurred while persisting consent amendment " + - "history data.")); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/manage/vrp/VRPConsentHandlerTest.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/manage/vrp/VRPConsentHandlerTest.java deleted file mode 100644 index 48e8ab94..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/manage/vrp/VRPConsentHandlerTest.java +++ /dev/null @@ -1,149 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.manage.vrp; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.util.CarbonUtils; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentServiceUtil; -import com.wso2.openbanking.accelerator.consent.extensions.internal.ConsentExtensionsDataHolder; -import com.wso2.openbanking.accelerator.consent.extensions.manage.impl.VRPConsentRequestHandler; -import com.wso2.openbanking.accelerator.consent.extensions.manage.model.ConsentManageData; -import com.wso2.openbanking.accelerator.consent.extensions.utils.ConsentExtensionTestUtils; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import org.mockito.InjectMocks; -import org.mockito.Mock; -import org.mockito.MockitoAnnotations; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; - -import java.util.HashMap; -import java.util.Map; -import java.util.UUID; - -import static org.mockito.Matchers.anyBoolean; -import static org.mockito.Matchers.anyString; -import static org.mockito.Mockito.mock; -import static org.powermock.api.mockito.PowerMockito.doReturn; -import static org.powermock.api.mockito.PowerMockito.when; - -/** - * Test class for VRPConsentRequestHandler. - */ -@PowerMockIgnore({"jdk.internal.reflect.*", -"com.wso2.openbanking.accelerator.consent.extensions.common.*"}) -@PrepareForTest({OpenBankingConfigParser.class, ConsentServiceUtil.class, - ConsentExtensionsDataHolder.class}) -public class VRPConsentHandlerTest extends PowerMockTestCase { - - @InjectMocks - private final VRPConsentRequestHandler handler = new VRPConsentRequestHandler(); - - @Mock - private ConsentManageData consentManageData; - - @Mock - OpenBankingConfigParser openBankingConfigParser; - - @Mock - ConsentCoreServiceImpl consentCoreServiceImpl; - - private static Map configMap; - - - @ObjectFactory - public IObjectFactory getObjectFactory() { - - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - - @BeforeClass - public void setUp() throws ReflectiveOperationException { - MockitoAnnotations.initMocks(this); - - configMap = new HashMap<>(); - - new CarbonUtils(); - System.setProperty("some.property", "property.value"); - System.setProperty("carbon.home", "."); - ConsentExtensionTestUtils.injectEnvironmentVariable("CARBON_HOME", "."); - - consentManageData = mock(ConsentManageData.class); - } - - @BeforeMethod - public void initMethod() { - - openBankingConfigParser = mock(OpenBankingConfigParser.class); - doReturn(configMap).when(openBankingConfigParser).getConfiguration(); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParser); - - } - - @Test(expectedExceptions = ConsentException.class) - public void testHandleConsentManageGetWithValidConsentIdAndMatchingClientId() throws ConsentManagementException { - UUID consentIdUUID = UUID.randomUUID(); - doReturn("vrp-consent/".concat(consentIdUUID.toString())).when(consentManageData).getRequestPath(); - ConsentResource consent = mock(ConsentResource.class); - doReturn("5678").when(consent).getClientID(); - - consentCoreServiceImpl = mock(ConsentCoreServiceImpl.class); - doReturn(consent).when(consentCoreServiceImpl).getConsent(anyString(), anyBoolean()); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceImpl); - - String expectedClientId = "matchingClientId"; - doReturn(expectedClientId).when(consentManageData).getClientId(); - - handler.handleConsentManageGet(consentManageData); - } - - - @Test(expectedExceptions = ConsentException.class) - public void testHandleConsentManageDeleteWithValidConsent() throws ConsentManagementException { - - UUID consentIdUUID = UUID.randomUUID(); - doReturn("vrp-consent/".concat(consentIdUUID.toString())).when(consentManageData).getRequestPath(); - ConsentResource consent = mock(ConsentResource.class); - doReturn("5678").when(consent).getClientID(); - - consentCoreServiceImpl = mock(ConsentCoreServiceImpl.class); - doReturn(consent).when(consentCoreServiceImpl).getConsent(anyString(), anyBoolean()); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceImpl); - - String expectedClientId = "6788"; - doReturn(expectedClientId).when(consentManageData).getClientId(); - - handler.handleConsentManageDelete(consentManageData); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/manage/vrp/VRPConsentRequestValidatorTest.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/manage/vrp/VRPConsentRequestValidatorTest.java deleted file mode 100644 index 12ec982a..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/manage/vrp/VRPConsentRequestValidatorTest.java +++ /dev/null @@ -1,2009 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - *

- * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.manage.vrp; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.util.CarbonUtils; -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.manage.model.ConsentManageData; -import com.wso2.openbanking.accelerator.consent.extensions.manage.validator.VRPConsentRequestValidator; -import com.wso2.openbanking.accelerator.consent.extensions.util.ConsentManageUtil; -import com.wso2.openbanking.accelerator.consent.extensions.utils.ConsentExtensionTestUtils; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import net.minidev.json.JSONValue; -import org.mockito.Mock; -import org.mockito.Mockito; -import org.mockito.MockitoAnnotations; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -import java.util.HashMap; -import java.util.Map; - -import static org.mockito.Mockito.mock; -import static org.testng.Assert.assertTrue; -import static org.testng.AssertJUnit.assertFalse; - -/** - * Test class for VRPConsentRequestValidator. - */ -@PowerMockIgnore({"jdk.internal.reflect.*"}) -@PrepareForTest({OpenBankingConfigParser.class}) -public class VRPConsentRequestValidatorTest extends PowerMockTestCase { - - @Mock - private ConsentManageData consentManageData; - - @Mock - OpenBankingConfigParser openBankingConfigParser; - - private static Map configMap; - - @BeforeClass - public void setUp() throws ReflectiveOperationException { - MockitoAnnotations.initMocks(this); - - configMap = new HashMap<>(); - //to execute util class initialization - new CarbonUtils(); - System.setProperty("some.property", "property.value"); - System.setProperty("carbon.home", "."); - ConsentExtensionTestUtils.injectEnvironmentVariable("CARBON_HOME", "."); - - consentManageData = mock(ConsentManageData.class); - } - - @BeforeMethod - public void initMethod() { - - openBankingConfigParser = mock(OpenBankingConfigParser.class); - Mockito.doReturn(configMap).when(openBankingConfigParser).getConfiguration(); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParser); - } - - @Test - public void testVrpPayload() { - - VRPConsentRequestValidator handler = new VRPConsentRequestValidator(); - JSONObject response = handler.validateVRPPayload("payload"); - Assert.assertFalse((boolean) response.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR, response.get(ConsentExtensionConstants.ERRORS)); - - } - - @Test - public void testVrpEmptyPayload() { - - JSONObject response = VRPConsentRequestValidator.validateVRPPayload(""); - Assert.assertFalse((boolean) response.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR, response.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiation() { - - String initiationPayloads = VRPTestConstants.vrpInitiationPayloadWithoutData; - JSONObject response = VRPConsentRequestValidator.validateVRPPayload(JSONValue. - parse(initiationPayloads)); - Assert.assertFalse((boolean) response.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR, response.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpControlParameters() { - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_CONTROL_PARAMETERS; - JSONObject response = VRPConsentRequestValidator.validateControlParameters((JSONObject) JSONValue. - parse(initiationPayloads)); - Assert.assertFalse((boolean) response.get(ConsentExtensionConstants.IS_VALID)); - } - - @Test - public void testVrpEmptyData() { - String initiationPayloads = VRPTestConstants.vrpInitiationPayloadWithStringData; - JSONObject response = VRPConsentRequestValidator.validateVRPPayload(JSONValue. - parse(initiationPayloads)); - Assert.assertFalse((boolean) response.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR, response.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpDataIsJsonObject() { - String initiationPayloads = VRPTestConstants.vrpInitiationPayloadWithOutJsonObject; - JSONObject response = VRPConsentRequestValidator.validateVRPPayload(JSONValue. - parse(initiationPayloads)); - Assert.assertFalse((boolean) response.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR, response.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadWithoutControlParameterKey() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_CONTROL_PARAMETERS; - JSONObject result = VRPConsentRequestValidator.validateControlParameters((JSONObject) JSONValue. - parse(initiationPayloads)); - JSONObject result2 = VRPConsentRequestValidator.validateMaximumIndividualAmount((JSONObject) JSONValue. - parse(initiationPayloads)); - JSONObject result3 = VRPConsentRequestValidator. - validateMaximumIndividualAmountCurrency((JSONObject) JSONValue.parse(initiationPayloads)); - - Assert.assertTrue(true); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) result2.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) result3.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.MISSING_MAXIMUM_INDIVIDUAL_AMOUNT, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateAmountCurrencyWithCurrencyKeys() { - - JSONObject jsonObject = new JSONObject(); - jsonObject.put("Currency", "USD"); - - JSONArray jsonArray = new JSONArray(); - jsonArray.add(jsonObject); - - JSONObject result = VRPConsentRequestValidator. - validateAmountCurrencyPeriodicLimits(jsonArray, "Currency", String.class); - Assert.assertTrue((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - } - - @Test - public void testValidateAmountCurrencyWithInvalidKey() { - - JSONObject jsonObject = new JSONObject(); - jsonObject.put("InvalidKey", "USD"); - - JSONArray jsonArray = new JSONArray(); - jsonArray.add(jsonObject); - - JSONObject result = VRPConsentRequestValidator. - validateAmountCurrencyPeriodicLimits(jsonArray, "Currency", String.class); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Mandatory parameter 'Currency' is not present in payload", - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadWithoutPeriodicLimitCurrency() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_PERIODIC_LIMIT_CURRENCY; - JSONObject results = VRPConsentRequestValidator.validateControlParameters((JSONObject) JSONValue. - parse(initiationPayloads)); - JSONObject result = VRPConsentRequestValidator.validateCurrencyPeriodicLimit((JSONObject) JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) results.get(ConsentExtensionConstants.IS_VALID)); - assertTrue(true); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("parameter passed in is null", - result.get(ConsentExtensionConstants.ERRORS)); - - } - - @Test - public void testVrpInitiationPayloadWithoutPeriodicLimitAmount() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_PERIODIC_LIMIT_AMOUNT; - JSONObject results = VRPConsentRequestValidator.validateControlParameters((JSONObject) JSONValue. - parse(initiationPayloads)); - JSONObject result = VRPConsentRequestValidator.validateMaximumIndividualAmountCurrency((JSONObject) JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) results.get(ConsentExtensionConstants.IS_VALID)); - assertTrue(true); - - } - - @Test - public void testValidateAmountCurrencyPeriodicLimitsWithInvalidValue() { - - JSONObject jsonObject = new JSONObject(); - jsonObject.put("Currency", 123); - - JSONArray jsonArray = new JSONArray(); - jsonArray.add(jsonObject); - - JSONObject result = VRPConsentRequestValidator. - validateAmountCurrencyPeriodicLimits(jsonArray, "Currency", String.class); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - } - - @Test - public void testValidateAmountCurrencyPeriodicLimitsWithInvalidKey() { - - JSONArray testData = new JSONArray(); - JSONObject limit = new JSONObject(); - limit.put("anotherKey", "USD"); - testData.add(limit); - - JSONObject result = VRPConsentRequestValidator. - validateAmountCurrencyPeriodicLimits(testData, "currency", String.class); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Mandatory parameter 'currency' is not present in payload", - result.get(ConsentExtensionConstants.ERRORS)); - } - - - @Test - public void testValidationFailureForCurrency() { - - JSONObject limit = new JSONObject(); - limit.put(ConsentExtensionConstants.CURRENCY, 123); - - JSONArray periodicLimits = new JSONArray(); - periodicLimits.add(limit); - - JSONObject result = VRPConsentRequestValidator.validateAmountCurrencyPeriodicLimits(periodicLimits, - ConsentExtensionConstants.CURRENCY, String.class); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("The value of 'Currency' is not of type String", - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateAmountCurrencyPeriodicLimitsWithCurrencyKey() { - - // Test case 2: Invalid currency key (empty value) - JSONArray testData2 = new JSONArray(); - JSONObject limit2 = new JSONObject(); - limit2.put("currency", ""); - testData2.add(limit2); - - JSONObject result2 = VRPConsentRequestValidator. - validateAmountCurrencyPeriodicLimits(testData2, "0", String.class); - Assert.assertFalse((boolean) result2.get(ConsentExtensionConstants.IS_VALID)); - JSONArray testData3 = new JSONArray(); - - JSONObject result3 = VRPConsentRequestValidator. - validateAmountCurrencyPeriodicLimits(testData3, "0", String.class); - Assert.assertTrue((boolean) result3.get(ConsentExtensionConstants.IS_VALID)); - - JSONObject result4 = VRPConsentRequestValidator. - validateAmountCurrencyPeriodicLimits(null, "currency", String.class); - Assert.assertFalse((boolean) result4.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("parameter passed in is null", - result4.get(ConsentExtensionConstants.ERRORS)); - - - } - - @Test - public void testVrpInitiationPayloadWithoutRisk() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_RISK; - JSONObject result = VRPConsentRequestValidator.validateConsentRisk((JSONObject) JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_RISK, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testIsValidObjectDebAcc() { - String initiationPayloads = VRPTestConstants.METADATA_VRP_DEBTOR_ACCOUNT; - JSONObject result = VRPConsentRequestValidator.validateVRPInitiationPayload((JSONObject) JSONValue. - parse(initiationPayloads)); - - // Test case 3: Non-JSONObject value - String nonJsonObject = "not a JSONObject"; - Assert.assertFalse(VRPConsentRequestValidator.isValidJSONObject(nonJsonObject), - ConsentExtensionConstants.IS_VALID); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - - // Test case 4: Null value - Object nullValue = null; - Assert.assertFalse(VRPConsentRequestValidator.isValidJSONObject(nullValue), - ConsentExtensionConstants.IS_VALID); - - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_DEBTOR_ACC, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testIsValidObjectDebtorAcc() { - String initiationPayloads = VRPTestConstants.METADATA_VRP_DEBTOR_ACCOUNT; - JSONObject result = VRPConsentRequestValidator.validateVRPInitiationPayload((JSONObject) JSONValue. - parse(initiationPayloads)); - - // Test case 3: Non-JSONObject value - String nonJsonObject = "not a JSONObject"; - Assert.assertFalse(VRPConsentRequestValidator.isValidJSONObject(nonJsonObject), - ConsentExtensionConstants.IS_VALID); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - // Test case 4: Null value - Object nullValue = null; - Assert.assertFalse(VRPConsentRequestValidator.isValidJSONObject(nullValue), - ConsentExtensionConstants.IS_VALID); - } - - @Test - public void testVrpInitiationPayloadWithoutDebtorAcc() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_DEBTOR_ACCOUNT; - JSONObject result = VRPConsentRequestValidator.validateVRPInitiationPayload((JSONObject) JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_DEBTOR_ACC, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadWithoutCreditAcc() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_CREDITOR_ACCOUNT; - JSONObject result = VRPConsentRequestValidator.validateConsentInitiation((JSONObject) JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadWithoutCreditorAcc() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_CREDITOR_ACCOUNT; - JSONObject result = VRPConsentRequestValidator.validateVRPInitiationPayload((JSONObject) JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_DEBTOR_ACC, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadWithoutSchemeName() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_DEBTOR_ACCOUNT_SCHEME_NAME; - JSONObject result = ConsentManageUtil.validateDebtorAccount((JSONObject) JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.MISSING_DEBTOR_ACC_SCHEME_NAME, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateDebtorAccount_InvalidSchemeName() { - JSONObject debtorAccount = new JSONObject(); - debtorAccount.put(ConsentExtensionConstants.SCHEME_NAME, ""); - debtorAccount.put(ConsentExtensionConstants.IDENTIFICATION, "ValidIdentification"); - debtorAccount.put(ConsentExtensionConstants.NAME, "ValidName"); - - JSONObject result = ConsentManageUtil.validateDebtorAccount(debtorAccount); - - Assert.assertFalse(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(result.get(ConsentExtensionConstants.ERRORS), - ErrorConstants.MISSING_DEBTOR_ACC_SCHEME_NAME); - } - - @Test - public void testVrpInitiationPayloadWithoutIdentification() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_DEBTOR_ACCOUNT_IDENTIFICATION; - JSONObject result = VRPConsentRequestValidator.validateVRPInitiationPayload((JSONObject) JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - } - - @Test - public void testVrpInitiationPayloadCreditorAccWithoutSchemeName() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_CREDITOR_ACCOUNT_SCHEME_NAME; - JSONObject result = VRPConsentRequestValidator.validateVRPPayload((JSONValue. - parse(initiationPayloads))); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - - } - - @Test - public void testVrpInitiationPayloadCreditorAccWithoutIdentification() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_CREDITOR_ACCOUNT_IDENTIFICATION; - JSONObject result = VRPConsentRequestValidator.validateVRPPayload(JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - - } - - @Test - public void testValidatePeriodicLimits() { - - JSONObject invalidLimit = new JSONObject(); - invalidLimit.put("someKey", "someValue"); - - JSONObject validationResult = VRPConsentRequestValidator.validatePeriodicLimits(invalidLimit); - - Assert.assertFalse(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.MISSING_PERIOD_LIMITS, - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateAmountCurrencyPeriodicLimitsWithValidKey() { - - JSONObject jsonObject = new JSONObject(); - jsonObject.put("Currency", "USD"); - - JSONArray jsonArray = new JSONArray(); - jsonArray.add(jsonObject); - - JSONObject result = VRPConsentRequestValidator. - validateAmountCurrencyPeriodicLimits(jsonArray, "Currency", String.class); - Assert.assertTrue(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - } - - @Test - public void testValidateAmountCurrencyPeriodicLimitsWithInvalidKeys() { - - JSONObject jsonObject = new JSONObject(); - jsonObject.put("InvalidKey", "USD"); - - JSONArray jsonArray = new JSONArray(); - jsonArray.add(jsonObject); - - JSONObject result = VRPConsentRequestValidator. - validateAmountCurrencyPeriodicLimits(jsonArray, "Currency", String.class); - - Assert.assertFalse(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals("Mandatory parameter 'Currency' is not present in payload", - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateAmountCurrencyPeriodicLimitsWithEmptyArray() { - - JSONArray jsonArray = new JSONArray(); - - JSONObject result = VRPConsentRequestValidator. - validateAmountCurrencyPeriodicLimits(jsonArray, "Currency", String.class); - - Assert.assertTrue(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(null, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateAmountCurrencyPeriodicLimitsWithNullArray() { - - JSONObject result = VRPConsentRequestValidator. - validateAmountCurrencyPeriodicLimits(null, "Currency", String.class); - - Assert.assertFalse(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals("parameter passed in is null", - result.get(ConsentExtensionConstants.ERRORS)); - } - - - @Test - public void testVrpPeriodicTypeJsonArray() { - - Object invalidObject = "Not a JSONArray"; - boolean isValidInvalidObject = VRPConsentRequestValidator.isValidJSONArray(invalidObject); - - // Test case 2: Missing period type key - JSONObject missingKeyObject = new JSONObject(); - JSONObject result2 = VRPConsentRequestValidator.validatePeriodType(missingKeyObject); - Assert.assertFalse(Boolean.parseBoolean(result2.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.MISSING_PERIOD_TYPE, result2.get(ConsentExtensionConstants.ERRORS)); - - // Test case 3: Null period type - JSONObject nullPeriodTypeObject = new JSONObject(); - nullPeriodTypeObject.put(ConsentExtensionConstants.PERIOD_TYPE, null); - JSONObject result3 = VRPConsentRequestValidator.validatePeriodType(nullPeriodTypeObject); - Assert.assertFalse(Boolean.parseBoolean(result3.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.MISSING_PERIOD_TYPE, result2.get(ConsentExtensionConstants.ERRORS)); - - // Test case 4: Empty period type - JSONObject emptyPeriodTypeObject = new JSONObject(); - emptyPeriodTypeObject.put(ConsentExtensionConstants.PERIOD_TYPE, ""); - JSONObject result4 = VRPConsentRequestValidator.validatePeriodType(emptyPeriodTypeObject); - Assert.assertFalse(Boolean.parseBoolean(result4.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.MISSING_PERIOD_TYPE, result2.get(ConsentExtensionConstants.ERRORS)); - - // Test case 5: Invalid period type - JSONObject invalidPeriodTypeObject = new JSONObject(); - invalidPeriodTypeObject.put(ConsentExtensionConstants.PERIOD_TYPE, "InvalidType"); - JSONObject result5 = VRPConsentRequestValidator.validatePeriodType(invalidPeriodTypeObject); - Assert.assertFalse(Boolean.parseBoolean(result5.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.MISSING_PERIOD_TYPE, result2.get(ConsentExtensionConstants.ERRORS)); - - Assert.assertFalse(isValidInvalidObject, ConsentExtensionConstants.IS_VALID); - } - - @Test - public void testDataContainsKey_InitiationNotPresent() { - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_INITIATION; - JSONObject result = VRPConsentRequestValidator.validateConsentInitiation((JSONObject) JSONValue. - parse(initiationPayloads)); - - boolean containsKey = result.containsKey(ConsentExtensionConstants.INITIATION); - Assert.assertFalse(containsKey, ConsentExtensionConstants.IS_VALID); - Assert.assertEquals("Missing mandatory parameter Initiation in the payload", - result.get(ConsentExtensionConstants.ERRORS)); - } - - - @Test - public void testDataContainsKey_ControlParametersNotPresent() { - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_CONTROL_PARAMETERS; - JSONObject result = VRPConsentRequestValidator.validateConsentControlParameters((JSONObject) JSONValue. - parse(initiationPayloads)); - boolean containsKey = result.containsKey(ConsentExtensionConstants.CONTROL_PARAMETERS); - Assert.assertFalse(containsKey, ConsentExtensionConstants.IS_VALID); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_CONTROL_PARAMETER, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadMaximumIndividualAmountNotJsonObject() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITH_EMPTY_MAX_INDIVIDUAL_AMOUNT; - String date = VRPTestConstants.METADATA_VRP_WITHOUT_VALID_FROM_DATE; - String date2 = VRPTestConstants.METADATA_VRP_WITHOUT_VALID_TO_DATE; - - JSONObject result = VRPConsentRequestValidator.validateMaximumIndividualAmount((JSONObject) JSONValue. - parse(initiationPayloads)); - JSONObject results = VRPConsentRequestValidator.validateMaximumIndividualAmountCurrency((JSONObject) JSONValue. - parse(initiationPayloads)); - JSONObject result2 = VRPConsentRequestValidator.validateControlParameters((JSONObject) JSONValue. - parse(date)); - JSONObject result3 = VRPConsentRequestValidator.validateControlParameters((JSONObject) JSONValue. - parse(date)); - - boolean isValidNonJSONObject = VRPConsentRequestValidator.isValidJSONObject(initiationPayloads); - Assert.assertFalse(isValidNonJSONObject, (ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - - - boolean isValidNonJSONObjects = VRPConsentRequestValidator.isValidJSONObject(initiationPayloads); - Assert.assertFalse(isValidNonJSONObjects, (ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) results.get(ConsentExtensionConstants.IS_VALID)); - - boolean obj2 = VRPConsentRequestValidator.isValidJSONObject(date); - Assert.assertFalse(obj2, (ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) result2.get(ConsentExtensionConstants.IS_VALID)); - - boolean obj3 = VRPConsentRequestValidator.isValidJSONObject(date2); - Assert.assertFalse(obj3, (ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) result3.get(ConsentExtensionConstants.IS_VALID)); - } - - @Test - public void testVrpInitiationPayloadDebAcc() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_DEBTOR_ACC; - JSONObject result = VRPConsentRequestValidator.validateConsentInitiation((JSONObject) JSONValue. - parse(initiationPayloads)); - boolean isValidNonJSONObject = VRPConsentRequestValidator.isValidJSONObject(initiationPayloads); - Assert.assertFalse(isValidNonJSONObject); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_DEBTOR_ACC, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadDebAccs() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_DEB_ACC; - JSONObject result = VRPConsentRequestValidator.validateVRPPayload(JSONValue. - parse(initiationPayloads)); - boolean isValidNonJSONObject = VRPConsentRequestValidator.isValidJSONObject(initiationPayloads); - Assert.assertFalse(isValidNonJSONObject); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Parameter 'debtor account' passed in is null, empty, or not a JSONObject", - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationMax() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITH_EMPTY_MAX_INDIVIDUAL_AMOUNT; - JSONObject result = VRPConsentRequestValidator.validateVRPPayload(JSONValue. - parse(initiationPayloads)); - boolean isValidNonJSONObject = VRPConsentRequestValidator.isValidJSONObject(initiationPayloads); - Assert.assertFalse(isValidNonJSONObject, (ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR, result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadValidateDebAcc() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_DEB_ACC; - JSONObject result = VRPConsentRequestValidator.validateVRPInitiationPayload((JSONObject) JSONValue. - parse(initiationPayloads)); - boolean isValidNonJSONObject = VRPConsentRequestValidator.isValidJSONObject(initiationPayloads); - Assert.assertFalse(isValidNonJSONObject, (ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_DEBTOR_ACC, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadCreditorAcc() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_CREDITOR_ACC; - JSONObject result = VRPConsentRequestValidator.validateVRPInitiationPayload((JSONObject) JSONValue. - parse(initiationPayloads)); - boolean isValidNonJSONObject = VRPConsentRequestValidator.isValidJSONObject(initiationPayloads); - Assert.assertFalse(isValidNonJSONObject); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_DEBTOR_ACC, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadCreditorAccs() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_CREDITOR_ACC; - JSONObject result = VRPConsentRequestValidator.validateVRPPayload(JSONValue. - parse(initiationPayloads)); - boolean isValidNonJSONObject = VRPConsentRequestValidator.isValidJSONObject(initiationPayloads); - - Assert.assertFalse(isValidNonJSONObject); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Parameter 'creditor account' passed in is null, empty, or not a JSONObject", - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadDebtorAccs() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_DEB_ACC; - JSONObject result = VRPConsentRequestValidator.validateVRPPayload(JSONValue. - parse(initiationPayloads)); - boolean isValidNonJSONObject = VRPConsentRequestValidator.isValidJSONObject(initiationPayloads); - - Assert.assertFalse(isValidNonJSONObject); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - - Assert.assertEquals("Parameter 'debtor account' passed in is null, empty, or not a JSONObject", - result.get(ConsentExtensionConstants.ERRORS)); - } - - - @Test - public void testIsValidObject_NegativeScenarios() { - - String nonJSONObject = "Not a JSONObject"; - JSONObject validInitiationObject = new JSONObject(); - - boolean isValidNonJSONObject = VRPConsentRequestValidator.isValidJSONObject(nonJSONObject); - boolean isValidNonJSONObject1 = VRPConsentRequestValidator.isValidJSONObject(validInitiationObject); - Assert.assertFalse(isValidNonJSONObject1, (ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse(isValidNonJSONObject, (ConsentExtensionConstants.IS_VALID)); - } - - @Test - public void testVrpInitiationPayloadInitiationNotJsonObject() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_EMPTY_INITIATION; - JSONObject result = VRPConsentRequestValidator.validateConsentInitiation((JSONObject) JSONValue. - parse(initiationPayloads)); - boolean isValidNonJSONObject = VRPConsentRequestValidator.isValidJSONObject(initiationPayloads); - Assert.assertFalse(isValidNonJSONObject, (ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Parameter 'initiation' passed in is null, empty, or not a JSONObject", - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadMaximumIndividualNotJsonObject() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITH_INVALID_MAX_INDIVIDUAL_AMOUNT; - JSONObject result = VRPConsentRequestValidator.validateMaximumIndividualAmount((JSONObject) JSONValue. - parse(initiationPayloads)); - boolean isValidNonJSONObject = VRPConsentRequestValidator.isValidJSONObject(initiationPayloads); - Assert.assertFalse(isValidNonJSONObject, (ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.MISSING_MAXIMUM_INDIVIDUAL_AMOUNT, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadCurrencyNotJsonObject() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITH_INVALID_MAX_INDIVIDUAL_AMOUNT; - JSONObject result = VRPConsentRequestValidator.validateControlParameters((JSONObject) JSONValue. - parse(initiationPayloads)); - boolean isValidNonJSONObject = VRPConsentRequestValidator.isValidJSONObject(initiationPayloads); - Assert.assertFalse(isValidNonJSONObject, (ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.MISSING_MAXIMUM_INDIVIDUAL_AMOUNT, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadControlParametersNotJsonObject() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITH_EMPTY_CONTROL_PARAMETERS; - JSONObject result = VRPConsentRequestValidator.validateConsentControlParameters((JSONObject) JSONValue. - parse(initiationPayloads)); - boolean isValidNonJSONObject = VRPConsentRequestValidator.isValidJSONObject(initiationPayloads); - Assert.assertFalse(isValidNonJSONObject, (ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Parameter 'control parameters' passed in is null, empty, or not a JSONObject", - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadWithoutDate() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITH_INVALID_VALID_FROM_DATETIME; - JSONObject result = VRPConsentRequestValidator.validateParameterDateTime((JSONObject) JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.MISSING_VALID_TO_DATE_TIME, - result.get(ConsentExtensionConstants.ERRORS)); - } - - - @Test - public void testVrpInitiationPayloadWithoutValidToDate() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_VALID_TO_DATE; - JSONObject result = VRPConsentRequestValidator.validateParameterDateTime((JSONObject) JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.MISSING_VALID_TO_DATE_TIME, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadMaximumIndividualAmountIsJsonObject() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITH_EMPTY_MAX_INDIVIDUAL_AMOUNT; - JSONObject result = VRPConsentRequestValidator.validateControlParameters((JSONObject) JSONValue. - parse(initiationPayloads)); - boolean isValidNonJSONObject = VRPConsentRequestValidator.isValidJSONObject(initiationPayloads); - Assert.assertFalse(isValidNonJSONObject, (ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.MISSING_MAXIMUM_INDIVIDUAL_AMOUNT, - result.get(ConsentExtensionConstants.ERRORS)); - } - - - @Test - public void testIsValidDateTimeObjectNegativeScenarios() { - // Test case 1: Empty string - String emptyString = ""; - JSONObject resultEmptyString = VRPConsentRequestValidator.isValidDateTimeObject(emptyString); - Assert.assertFalse((boolean) resultEmptyString.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.MISSING_DATE_TIME_FORMAT, - resultEmptyString.get(ConsentExtensionConstants.ERRORS)); - - // Test case 2: Null value - Object nullValue = null; - boolean resultNullValue = false; - Assert.assertFalse(resultNullValue, "Expected false for a null value"); - - // Test case 3: Non-string value - Object nonStringValue = 123; // Assuming an integer, but could be any non-string type - JSONObject resultNonStringValue = VRPConsentRequestValidator.isValidDateTimeObject(nonStringValue); - Assert.assertFalse((boolean) resultNonStringValue.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.MISSING_DATE_TIME_FORMAT, - resultNonStringValue.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateAmountCurrencyPeriodicLimits() { - - // Test case 2: Key is null - JSONArray testData2 = new JSONArray(); - JSONObject result2 = VRPConsentRequestValidator. - validateAmountCurrencyPeriodicLimits(testData2, null, String.class); - Assert.assertTrue((boolean) result2.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(null, - result2.get(ConsentExtensionConstants.ERRORS)); - - // Test case 3: ParentObj is null - JSONObject result3 = VRPConsentRequestValidator.validateAmountCurrencyPeriodicLimits(null, "0", String.class); - Assert.assertTrue((boolean) result2.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(null, - result2.get(ConsentExtensionConstants.ERRORS)); - - // Test case 4: Key is not present in parentObj - JSONArray testData4 = new JSONArray(); - JSONObject result4 = VRPConsentRequestValidator. - validateAmountCurrencyPeriodicLimits(testData4, "nonExistentKey", String.class); - Assert.assertTrue((boolean) result2.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(null, - result2.get(ConsentExtensionConstants.ERRORS)); - - // Test case 5: Value is an empty String - JSONArray testData5 = new JSONArray(); - testData5.add(""); - JSONObject result5 = VRPConsentRequestValidator. - validateAmountCurrencyPeriodicLimits(testData5, "0", String.class); - Assert.assertTrue((boolean) result2.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(null, - result2.get(ConsentExtensionConstants.ERRORS)); - } - - - @Test - public void testValidateKeyAndNonEmptyStringValue() { - - // Test case 2: Key is null - JSONArray testData2 = new JSONArray(); - JSONObject result2 = VRPConsentRequestValidator. - validateAmountCurrencyPeriodicLimits(testData2, null, String.class); - Assert.assertTrue((boolean) result2.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(null, - result2.get(ConsentExtensionConstants.ERRORS)); - - // Test case 3: ParentObj is null - JSONObject result3 = VRPConsentRequestValidator.validateAmountCurrencyPeriodicLimits(null, "0", String.class); - Assert.assertFalse((boolean) result3.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("parameter passed in is null", - result3.get(ConsentExtensionConstants.ERRORS)); - - // Test case 4: Key is not present in parentObj - JSONArray testData4 = new JSONArray(); - JSONObject result4 = VRPConsentRequestValidator. - validateAmountCurrencyPeriodicLimits(testData4, "nonExistentKey", String.class); - Assert.assertTrue((boolean) result4.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(null, - result4.get(ConsentExtensionConstants.ERRORS)); - - // Test case 5: Value is an empty String - JSONArray testData5 = new JSONArray(); - testData5.add(""); - JSONObject result5 = VRPConsentRequestValidator. - validateAmountCurrencyPeriodicLimits(testData5, "0", String.class); - Assert.assertTrue((boolean) result5.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(null, - result5.get(ConsentExtensionConstants.ERRORS)); - - - // Test case 7: Value is not a String - JSONArray testData7 = new JSONArray(); - testData7.add(123); // Assuming the value should be a String, but it's an integer in this case - JSONObject result7 = VRPConsentRequestValidator. - validateAmountCurrencyPeriodicLimits(testData7, "0", String.class); - Assert.assertTrue((boolean) result7.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(null, - result7.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testInvalidPeriodicLimitsFormat() { - - JSONObject controlParameters = new JSONObject(); - controlParameters.put(ConsentExtensionConstants.PERIODIC_LIMITS, "invalid-format"); - JSONObject validationResult = VRPConsentRequestValidator.validatePeriodicLimits(controlParameters); - - Assert.assertFalse((boolean) validationResult.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.INVALID_PARAMETER_PERIODIC_LIMITS, - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testInvalidMaxAmountFormat() { - - JSONObject controlParameters = new JSONObject(); - controlParameters.put(ConsentExtensionConstants.MAXIMUM_INDIVIDUAL_AMOUNT, "invalid-format"); - JSONObject validationResult = VRPConsentRequestValidator.validateMaximumIndividualAmount(controlParameters); - - Assert.assertFalse((boolean) validationResult.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Parameter 'maximum individual amount' passed in is null, empty, or not a JSONObject", - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testInvalidMaxAmountFormatPeriodicLimit() { - - JSONObject controlParameters = new JSONObject(); - controlParameters.put(ConsentExtensionConstants.PERIODIC_LIMITS, "invalid-format"); - JSONObject validationResults = VRPConsentRequestValidator. - validateControlParameters(controlParameters); - JSONObject validationResult = VRPConsentRequestValidator. - validateMaximumIndividualAmountCurrency(controlParameters); - - Assert.assertFalse((boolean) validationResults.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) validationResult.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("parameter passed in is null", - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testInvalidMaxAmountFormats() { - - JSONObject controlParameters = new JSONObject(); - controlParameters.put(ConsentExtensionConstants.PERIODIC_LIMITS, "invalid-format"); - JSONObject validationResult = VRPConsentRequestValidator.validatePeriodicLimits(controlParameters); - - Assert.assertFalse((boolean) validationResult.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.INVALID_PARAMETER_PERIODIC_LIMITS, - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testInvalidJSONObject() { - - String invalidJSONObject = "not a JSON object"; - boolean isValid = VRPConsentRequestValidator.isValidJSONObject(invalidJSONObject); - - Assert.assertFalse(isValid); - } - - @Test - public void testEmptyJSONObject() { - JSONObject emptyJSONObject = new JSONObject(); - - boolean isValid = VRPConsentRequestValidator.isValidJSONObject(emptyJSONObject); - Assert.assertFalse(isValid); - } - - @Test - public void testInvalidPeriodicAlignment() { - // Arrange - JSONObject invalidLimit = new JSONObject(); - invalidLimit.put(ConsentExtensionConstants.PERIOD_ALIGNMENT, "InvalidAlignment"); - - JSONObject isValid = VRPConsentRequestValidator.validatePeriodicLimits(invalidLimit); - - Assert.assertFalse((boolean) isValid.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.MISSING_PERIOD_LIMITS, - isValid.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testInvalidDateTimeRange() { - - JSONObject controlParameters = new JSONObject(); - controlParameters.put(ConsentExtensionConstants.VALID_FROM_DATE_TIME, "2023-01-01T00:00:00Z"); - controlParameters.put(ConsentExtensionConstants.VALID_TO_DATE_TIME, "2022-01-01T00:00:00Z"); - - boolean hasValidFromDate = controlParameters.containsKey(ConsentExtensionConstants.VALID_FROM_DATE_TIME); - boolean hasValidToDate = controlParameters.containsKey(ConsentExtensionConstants.VALID_TO_DATE_TIME); - - - assertTrue(hasValidFromDate && hasValidToDate); - } - - @Test - public void testVrpInitiationPayloadWithoutControlParameterss() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_CURRENCY; - JSONObject result = VRPConsentRequestValidator.validateMaximumIndividualAmount((JSONObject) JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Missing mandatory parameter Maximum Individual Amount", - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadWithoutPeriodicType() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_PERIODIC_TYPE; - JSONObject result = VRPConsentRequestValidator.validateControlParameters((JSONObject) JSONValue. - parse(initiationPayloads)); - JSONObject result2 = VRPConsentRequestValidator.validatePeriodicLimits((JSONObject) JSONValue. - parse(initiationPayloads)); - JSONObject result3 = VRPConsentRequestValidator.validatePeriodType((JSONObject) JSONValue. - parse(initiationPayloads)); - - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) result2.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.MISSING_MAXIMUM_INDIVIDUAL_AMOUNT, - result.get(ConsentExtensionConstants.ERRORS)); - } - - - @Test - public void testValidationFailureForNullCurrencyKey() { - - JSONArray periodicLimits = new JSONArray(); - periodicLimits.add(new JSONObject()); - - JSONObject result = VRPConsentRequestValidator.validateAmountCurrencyPeriodicLimits(periodicLimits, - ConsentExtensionConstants.CURRENCY, String.class); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Mandatory parameter 'Currency' is not present in payload", - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadWithoutPeriodicTypeCurrency() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_PERIODIC_TYPE_CURRENCY; - JSONObject result = VRPConsentRequestValidator.validateControlParameters((JSONObject) JSONValue. - parse(initiationPayloads)); - JSONObject result2 = VRPConsentRequestValidator.validatePeriodicLimits((JSONObject) JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) result2.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.MISSING_MAXIMUM_INDIVIDUAL_AMOUNT, - result.get(ConsentExtensionConstants.ERRORS)); - } - - - @Test - public void testValidationFailureForMissingKey() { - - JSONArray periodicLimits = new JSONArray(); - JSONObject jsonObject = new JSONObject(); - jsonObject.put("otherKey", "someValue"); - periodicLimits.add(jsonObject); - - - JSONObject result = VRPConsentRequestValidator.validateAmountCurrencyPeriodicLimits(periodicLimits, - ConsentExtensionConstants.CURRENCY, String.class); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Mandatory parameter 'Currency' is not present in payload", - result.get(ConsentExtensionConstants.ERRORS)); - } - - - @Test - public void testValidateControlParameters() { - - JSONObject controlParameters = new JSONObject(); - - JSONObject result = VRPConsentRequestValidator.validateControlParameters(controlParameters); - - assertTrue(result.containsKey(ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.MISSING_MAXIMUM_INDIVIDUAL_AMOUNT, - result.get(ConsentExtensionConstants.ERRORS)); - } - - - @Test - public void testValidateAmountCurrencyPeriodicLimits_Invalid() { - - JSONObject controlParameters = new JSONObject(); - JSONArray periodicLimits = new JSONArray(); - - JSONObject invalidPeriodicLimit = new JSONObject(); - periodicLimits.add(invalidPeriodicLimit); - - controlParameters.put(ConsentExtensionConstants.PERIODIC_LIMITS, periodicLimits); - - JSONObject result = VRPConsentRequestValidator.validateCurrencyPeriodicLimit(controlParameters); - - assertTrue(result.containsKey(ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Mandatory parameter 'Currency' is not present in payload", - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateAmountCurrencyPeriodicLimit_WithErrors() { - - JSONObject controlParameters = new JSONObject(); - JSONArray periodicLimits = new JSONArray(); - - JSONObject invalidPeriodicLimit = new JSONObject(); - periodicLimits.add(invalidPeriodicLimit); - - controlParameters.put(ConsentExtensionConstants.PERIODIC_LIMITS, periodicLimits); - - JSONObject periodicLimitType = VRPConsentRequestValidator. - validateCurrencyPeriodicLimit(controlParameters); - - Assert.assertFalse((boolean) periodicLimitType.get(ConsentExtensionConstants.IS_VALID)); - - assertTrue(periodicLimitType.containsKey(ConsentExtensionConstants.ERRORS)); - Assert.assertEquals("Mandatory parameter 'Currency' is not present in payload", - periodicLimitType.get(ConsentExtensionConstants.ERRORS)); - - } - - @Test - public void testValidateConsentRisk_ValidRequest() { - - JSONObject validRequest = new JSONObject(); - JSONObject data = new JSONObject(); - data.put("someKey", "someValue"); - - validRequest.put(ConsentExtensionConstants.DATA, data); - validRequest.put(ConsentExtensionConstants.RISK, new JSONObject()); - - JSONObject validationResponse = VRPConsentRequestValidator.validateConsentRisk(validRequest); - - assertTrue((boolean) validationResponse.get(ConsentExtensionConstants.IS_VALID)); - } - - @Test - public void testValidateConsentControlParameters_InvalidControlParameters() { - - JSONObject invalidControlParametersObject = new JSONObject(); - invalidControlParametersObject.put("invalidParam", "value"); - - JSONObject invalidDataObject = new JSONObject(); - invalidDataObject.put(ConsentExtensionConstants.CONTROL_PARAMETERS, invalidControlParametersObject); - - - JSONObject invalidRequestObject = new JSONObject(); - invalidRequestObject.put(ConsentExtensionConstants.DATA, invalidDataObject); - - JSONObject validationResult = VRPConsentRequestValidator.validateConsentControlParameters(invalidRequestObject); - - Assert.assertFalse(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.MISSING_MAXIMUM_INDIVIDUAL_AMOUNT, - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidatePeriodicLimits_Valid() { - - JSONObject controlParametersObject = new JSONObject(); - JSONArray periodicLimitsArray = new JSONArray(); - - JSONObject periodicLimit1 = new JSONObject(); - periodicLimit1.put(ConsentExtensionConstants.PERIOD_ALIGNMENT, "ALIGNMENT1"); - periodicLimit1.put(ConsentExtensionConstants.PERIOD_TYPE, "TYPE1"); - - JSONObject periodicLimit2 = new JSONObject(); - periodicLimit2.put(ConsentExtensionConstants.PERIOD_ALIGNMENT, "ALIGNMENT2"); - periodicLimit2.put(ConsentExtensionConstants.PERIOD_TYPE, "TYPE2"); - - periodicLimitsArray.add(periodicLimit1); - periodicLimitsArray.add(periodicLimit2); - - controlParametersObject.put(ConsentExtensionConstants.PERIODIC_LIMITS, periodicLimitsArray); - - JSONObject validationResult = VRPConsentRequestValidator.validatePeriodicLimits(controlParametersObject); - JSONObject validationResults = VRPConsentRequestValidator.validatePeriodAlignment(controlParametersObject); - - Assert.assertFalse(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertFalse(Boolean.parseBoolean(validationResults.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.MISSING_PERIOD_ALIGNMENT, - validationResults.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidatePeriodicLimits_InvalidFormat() { - JSONObject controlParametersObject = new JSONObject(); - controlParametersObject.put(ConsentExtensionConstants.PERIODIC_LIMITS, "InvalidFormat"); - - JSONObject validationResult = VRPConsentRequestValidator.validatePeriodicLimits(controlParametersObject); - - Assert.assertFalse(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.INVALID_PARAMETER_PERIODIC_LIMITS, - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidatePeriodicLimits_MissingPeriodLimits() { - - JSONObject controlParametersObject = new JSONObject(); - - JSONObject validationResult = VRPConsentRequestValidator.validatePeriodicLimits(controlParametersObject); - - Assert.assertFalse(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.MISSING_PERIOD_LIMITS, - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateAmountCurrencyPeriodicLimit_Valid() { - - JSONObject controlParametersObject = new JSONObject(); - JSONArray periodicLimitsArray = new JSONArray(); - - JSONObject periodicLimit1 = new JSONObject(); - periodicLimit1.put(ConsentExtensionConstants.CURRENCY, "USD"); - - JSONObject periodicLimit2 = new JSONObject(); - periodicLimit2.put(ConsentExtensionConstants.CURRENCY, "EUR"); - - periodicLimitsArray.add(periodicLimit1); - periodicLimitsArray.add(periodicLimit2); - - controlParametersObject.put(ConsentExtensionConstants.PERIODIC_LIMITS, periodicLimitsArray); - - JSONObject validationResult = VRPConsentRequestValidator. - validateAmountPeriodicLimit(controlParametersObject); - - assertFalse(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals("Mandatory parameter 'Amount' is not present in payload", - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateAmountCurrencyPeriodicLimit_MissingCurrency() { - JSONObject controlParametersObject = new JSONObject(); - JSONArray periodicLimitsArray = new JSONArray(); - - JSONObject periodicLimit1 = new JSONObject(); - periodicLimitsArray.add(periodicLimit1); - - controlParametersObject.put(ConsentExtensionConstants.PERIODIC_LIMITS, periodicLimitsArray); - - JSONObject validationResult = VRPConsentRequestValidator. - validateAmountPeriodicLimit(controlParametersObject); - - Assert.assertFalse(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals("Mandatory parameter 'Amount' is not present in payload", - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidatePeriodicType_Valid() { - JSONObject periodicLimitObject = new JSONObject(); - periodicLimitObject.put(ConsentExtensionConstants.PERIOD_TYPE, ConsentExtensionConstants.MONTH); - - JSONObject validationResult = VRPConsentRequestValidator.validatePeriodType(periodicLimitObject); - - Assert.assertTrue(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - } - - @Test - public void testValidatePeriodicType_InvalidType() { - JSONObject periodicLimitObject = new JSONObject(); - periodicLimitObject.put(ConsentExtensionConstants.PERIOD_TYPE, "InvalidType"); - - JSONObject validationResult = VRPConsentRequestValidator.validatePeriodType(periodicLimitObject); - - Assert.assertFalse(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.INVALID_PERIOD_TYPE, - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidatePeriodicType_MissingType() { - - JSONObject periodicLimitObject = new JSONObject(); - - JSONObject validationResult = VRPConsentRequestValidator.validatePeriodType(periodicLimitObject); - - Assert.assertFalse(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.MISSING_PERIOD_TYPE, - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidatePeriodicType_EmptyType() { - - JSONObject periodicLimitObject = new JSONObject(); - periodicLimitObject.put(ConsentExtensionConstants.PERIOD_TYPE, ""); - - JSONObject validationResult = VRPConsentRequestValidator.validatePeriodType(periodicLimitObject); - - Assert.assertFalse(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals("Value of period type is empty or the value passed in is not a string", - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidatePeriodicType_NullType() { - JSONObject periodicLimitObject = new JSONObject(); - periodicLimitObject.put(ConsentExtensionConstants.PERIOD_TYPE, null); - - JSONObject validationResult = VRPConsentRequestValidator.validatePeriodType(periodicLimitObject); - - Assert.assertFalse(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals("Value of period type is empty or the value passed in is not a string", - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testYourMethod_ValidPeriodicType() { - - JSONObject controlParameters = new JSONObject(); - JSONArray periodicLimits = new JSONArray(); - - JSONObject periodicLimit = new JSONObject(); - periodicLimit.put(ConsentExtensionConstants.PERIOD_TYPE, ConsentExtensionConstants.MONTH); - periodicLimits.add(periodicLimit); - - controlParameters.put(ConsentExtensionConstants.PERIODIC_LIMITS, periodicLimits); - - JSONObject result = VRPConsentRequestValidator.validateAmountPeriodicLimit(controlParameters); - - Assert.assertFalse(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals("Mandatory parameter 'Amount' is not present in payload", - result.get(ConsentExtensionConstants.ERRORS)); - - } - - @Test - public void testYourMethod_InvalidPeriodicType() { - - JSONObject controlParameters = new JSONObject(); - JSONArray periodicLimits = new JSONArray(); - - JSONObject periodicLimit = new JSONObject(); - periodicLimit.put(ConsentExtensionConstants.PERIOD_TYPE, "InvalidType"); - periodicLimits.add(periodicLimit); - - controlParameters.put(ConsentExtensionConstants.PERIODIC_LIMITS, periodicLimits); - - JSONObject result = VRPConsentRequestValidator.validateCurrencyPeriodicLimit(controlParameters); - - Assert.assertFalse(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals("Mandatory parameter 'Currency' is not present in payload", - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testYourMethod_MissingPeriodicType() { - - JSONObject controlParameters = new JSONObject(); - - JSONObject result = VRPConsentRequestValidator.validatePeriodicLimits(controlParameters); - - Assert.assertFalse(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.MISSING_PERIOD_LIMITS, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadWithoutDebtorAccs() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_DEBTOR_ACCOUNT; - JSONObject result = VRPConsentRequestValidator.validateConsentInitiation((JSONObject) JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_DEBTOR_ACC, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateConsentRisk_InvalidRequest() { - JSONObject requestBody = new JSONObject(); - JSONObject data = new JSONObject(); - data.put("key1", "value1"); - requestBody.put("data", data); - - JSONObject validationResult = VRPConsentRequestValidator.validateVRPPayload(requestBody); - JSONObject validationResults = VRPConsentRequestValidator.validateConsentRisk(requestBody); - - Assert.assertFalse(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertFalse(Boolean.parseBoolean(validationResults.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_RISK, - validationResults.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadWithoutDeAcc() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_DEBTOR_ACCOUNT; - JSONObject result = VRPConsentRequestValidator.validateVRPInitiationPayload((JSONObject) JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_DEBTOR_ACC, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadDAccWithoutSchemeName() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_DEBTOR_ACCOUNT_SCHEME_NAME; - JSONObject result = VRPConsentRequestValidator.validateVRPPayload((JSONValue. - parse(initiationPayloads))); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.MISSING_DEBTOR_ACC_SCHEME_NAME, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadDAccWithoutIdentification() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_DEBTOR_ACCOUNT_IDENTIFICATION; - JSONObject result = VRPConsentRequestValidator.validateVRPPayload(JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.MISSING_DEBTOR_ACC_IDENTIFICATION, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadDAcc() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_DEBTOR_ACC; - JSONObject result = VRPConsentRequestValidator.validateVRPInitiationPayload((JSONObject) JSONValue. - parse(initiationPayloads)); - boolean isValidNonJSONObject = VRPConsentRequestValidator.isValidJSONObject(initiationPayloads); - Assert.assertFalse(isValidNonJSONObject); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_DEBTOR_ACC, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadWithoutDAcc() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_DEBTOR_ACCOUNT; - JSONObject result = VRPConsentRequestValidator.validateConsentInitiation((JSONObject) JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_DEBTOR_ACC, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateMaximumIndividualAmountCurrency_InvalidAmountCurrency() { - JSONObject controlParameters = new JSONObject(); - JSONObject maximumIndividualAmount = new JSONObject(); - maximumIndividualAmount.put("InvalidKey", "USD"); - controlParameters.put(ConsentExtensionConstants.MAXIMUM_INDIVIDUAL_AMOUNT, maximumIndividualAmount); - - JSONObject result = VRPConsentRequestValidator.validateMaximumIndividualAmountCurrency(controlParameters); - - Assert.assertFalse(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals("Mandatory parameter 'Currency' is not present in payload", - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateMaximumIndividualAmountCurrency_MissingCurrency() { - - JSONObject controlParameters = new JSONObject(); - JSONObject maximumIndividualAmount = new JSONObject(); - - controlParameters.put(ConsentExtensionConstants.MAXIMUM_INDIVIDUAL_AMOUNT, maximumIndividualAmount); - - JSONObject result = VRPConsentRequestValidator.validateMaximumIndividualAmountCurrency(controlParameters); - - Assert.assertFalse(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals("Mandatory parameter 'Currency' is not present in payload", - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidatePeriodAlignmentInvalidValue() { - - JSONObject limit = new JSONObject(); - limit.put(ConsentExtensionConstants.PERIOD_ALIGNMENT, "InvalidValue"); - - JSONObject result = VRPConsentRequestValidator.validatePeriodAlignment(limit); - - Assert.assertFalse(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(result.get(ConsentExtensionConstants.ERRORS), - ErrorConstants.INVALID_PERIOD_ALIGNMENT); - } - - @Test - public void testValidatePeriodAlignmentMissingKey() { - - JSONObject limit = new JSONObject(); - - JSONObject result = VRPConsentRequestValidator.validatePeriodAlignment(limit); - - Assert.assertFalse(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(result.get(ConsentExtensionConstants.ERRORS), - ErrorConstants.MISSING_PERIOD_ALIGNMENT); - } - - - @Test - public void testValidatePeriodicAlignment_EmptyType() { - - JSONObject periodicLimitObject = new JSONObject(); - periodicLimitObject.put(ConsentExtensionConstants.PERIOD_ALIGNMENT, ""); - - JSONObject result = VRPConsentRequestValidator.validatePeriodType(periodicLimitObject); - Assert.assertFalse(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.MISSING_PERIOD_TYPE, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidatePeriodicAlignment() { - // Test case 1: Valid periodic type - JSONObject validLimitObject = new JSONObject(); - validLimitObject.put(ConsentExtensionConstants.PERIOD_ALIGNMENT, ConsentExtensionConstants.DAY); - JSONObject result1 = VRPConsentRequestValidator.validatePeriodAlignment(validLimitObject); - Assert.assertFalse((boolean) result1.get(ConsentExtensionConstants.IS_VALID)); - - // Test case 2: Missing period type key - JSONObject missingKeyObject = new JSONObject(); - JSONObject result2 = VRPConsentRequestValidator.validatePeriodAlignment(missingKeyObject); - Assert.assertFalse((boolean) result2.get(ConsentExtensionConstants.IS_VALID)); - - // Test case 3: Null period type - JSONObject nullPeriodTypeObject = new JSONObject(); - nullPeriodTypeObject.put(ConsentExtensionConstants.PERIOD_ALIGNMENT, null); - JSONObject result3 = VRPConsentRequestValidator.validatePeriodAlignment(nullPeriodTypeObject); - Assert.assertFalse((boolean) result3.get(ConsentExtensionConstants.IS_VALID)); - - // Test case 4: Empty period type - JSONObject emptyPeriodTypeObject = new JSONObject(); - emptyPeriodTypeObject.put(ConsentExtensionConstants.PERIOD_ALIGNMENT, ""); - JSONObject result4 = VRPConsentRequestValidator.validatePeriodAlignment(emptyPeriodTypeObject); - Assert.assertFalse((boolean) result4.get(ConsentExtensionConstants.IS_VALID)); - - // Test case 5: Invalid period type - JSONObject invalidPeriodTypeObject = new JSONObject(); - invalidPeriodTypeObject.put(ConsentExtensionConstants.PERIOD_ALIGNMENT, "InvalidType"); - JSONObject result5 = VRPConsentRequestValidator.validatePeriodAlignment(invalidPeriodTypeObject); - Assert.assertFalse((boolean) result5.get(ConsentExtensionConstants.IS_VALID)); - - Assert.assertEquals("Invalid value for period alignment in PeriodicLimits", - result5.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidatePeriodicAlignment_Valid() { - JSONObject periodicLimitObject = new JSONObject(); - periodicLimitObject.put(ConsentExtensionConstants.PERIOD_ALIGNMENT, ConsentExtensionConstants.CONSENT); - - JSONObject validationResult = VRPConsentRequestValidator.validatePeriodAlignment(periodicLimitObject); - - Assert.assertTrue(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - - } - - @Test - public void testValidatePeriodicAlignment_InvalidType() { - JSONObject periodicLimitObject = new JSONObject(); - periodicLimitObject.put(ConsentExtensionConstants.PERIOD_ALIGNMENT, "InvalidType"); - - JSONObject validationResult = VRPConsentRequestValidator.validatePeriodAlignment(periodicLimitObject); - - Assert.assertFalse(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals("Invalid value for period alignment in PeriodicLimits", - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidatePeriodicAlignment_MissingType() { - - JSONObject periodicLimitObject = new JSONObject(); - - JSONObject validationResult = VRPConsentRequestValidator.validatePeriodAlignment(periodicLimitObject); - - Assert.assertFalse(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.MISSING_PERIOD_ALIGNMENT, - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidatePeriodicAlignments_EmptyType() { - - JSONObject periodicLimitObject = new JSONObject(); - periodicLimitObject.put(ConsentExtensionConstants.PERIOD_ALIGNMENT, ""); - - JSONObject validationResult = VRPConsentRequestValidator.validatePeriodAlignment(periodicLimitObject); - - Assert.assertFalse(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals("Value of periodic alignment is empty or the value passed in is not a string", - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidatePeriodicAlignment_NullType() { - JSONObject periodicLimitObject = new JSONObject(); - periodicLimitObject.put(ConsentExtensionConstants.PERIOD_ALIGNMENT, null); - - JSONObject validationResult = VRPConsentRequestValidator.validatePeriodAlignment(periodicLimitObject); - - Assert.assertFalse(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals("Value of periodic alignment is empty or the value passed in is not a string", - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateAmountCurrencyPeriodicLimits_Valid() { - - JSONObject controlParametersObject = new JSONObject(); - JSONArray periodicLimitsArray = new JSONArray(); - - JSONObject periodicLimit1 = new JSONObject(); - periodicLimit1.put(ConsentExtensionConstants.CURRENCY, "USD"); - - JSONObject periodicLimit2 = new JSONObject(); - periodicLimit2.put(ConsentExtensionConstants.CURRENCY, "EUR"); - - periodicLimitsArray.add(periodicLimit1); - periodicLimitsArray.add(periodicLimit2); - - controlParametersObject.put(ConsentExtensionConstants.PERIODIC_LIMITS, periodicLimitsArray); - - JSONObject validationResult = VRPConsentRequestValidator. - validateCurrencyPeriodicLimit(controlParametersObject); - - - } - - - @Test - public void testValidateAmountCurrencyPeriodicLimitS_MissingCurrency() { - JSONObject controlParametersObject = new JSONObject(); - JSONArray periodicLimitsArray = new JSONArray(); - - JSONObject periodicLimit1 = new JSONObject(); - periodicLimitsArray.add(periodicLimit1); - - controlParametersObject.put(ConsentExtensionConstants.PERIODIC_LIMITS, periodicLimitsArray); - - JSONObject validationResult = VRPConsentRequestValidator. - validateCurrencyPeriodicLimit(controlParametersObject); - - Assert.assertFalse(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals("Mandatory parameter 'Currency' is not present in payload", - validationResult.get(ConsentExtensionConstants.ERRORS)); - - } - - @Test - public void testYourMethod_ValidPeriodicTypes() { - - JSONObject controlParameters = new JSONObject(); - JSONArray periodicLimits = new JSONArray(); - - JSONObject periodicLimit = new JSONObject(); - periodicLimit.put(ConsentExtensionConstants.PERIOD_TYPE, ConsentExtensionConstants.MONTH); - periodicLimits.add(periodicLimit); - - controlParameters.put(ConsentExtensionConstants.PERIODIC_LIMITS, periodicLimits); - - JSONObject result = VRPConsentRequestValidator.validateCurrencyPeriodicLimit(controlParameters); - - Assert.assertFalse(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - } - - @Test - public void testValidateAmountCurrencyPeriodicLimits_MissingCurrency() { - JSONObject controlParametersObject = new JSONObject(); - JSONArray periodicLimitsArray = new JSONArray(); - - JSONObject periodicLimit1 = new JSONObject(); - periodicLimitsArray.add(periodicLimit1); - - controlParametersObject.put(ConsentExtensionConstants.PERIODIC_LIMITS, periodicLimitsArray); - - JSONObject validationResult = VRPConsentRequestValidator. - validateCurrencyPeriodicLimit(controlParametersObject); - - Assert.assertFalse(Boolean.parseBoolean(validationResult.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals("Mandatory parameter 'Currency' is not present in payload", - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadMaximumIndividualAmountCurrencyNotJsonObject() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITH_EMPTY_MAX_INDIVIDUAL_AMOUNT; - - JSONObject results = VRPConsentRequestValidator.validateMaximumIndividualAmountCurrency((JSONObject) JSONValue. - parse(initiationPayloads)); - boolean isValidNonJSONObject = VRPConsentRequestValidator.isValidJSONObject(initiationPayloads); - Assert.assertFalse(isValidNonJSONObject, (ConsentExtensionConstants.IS_VALID)); - - boolean isValidNonJSONObjects = VRPConsentRequestValidator.isValidJSONObject(initiationPayloads); - Assert.assertFalse(isValidNonJSONObjects, (ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) results.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("parameter passed in is null", - results.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadMaximumIndividualCurrencyNotJsonObject() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITH_INVALID_MAX_INDIVIDUAL_AMOUNT; - JSONObject result = VRPConsentRequestValidator.validateMaximumIndividualAmountCurrency((JSONObject) JSONValue. - parse(initiationPayloads)); - boolean isValidNonJSONObject = VRPConsentRequestValidator.isValidJSONObject(initiationPayloads); - Assert.assertFalse(isValidNonJSONObject, (ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("parameter passed in is null", - result.get(ConsentExtensionConstants.ERRORS)); - } - - - @Test - public void testInvalidMaxAmountCurrencyFormatPeriodicLimit() { - - JSONObject controlParameters = new JSONObject(); - controlParameters.put(ConsentExtensionConstants.PERIODIC_LIMITS, "invalid-format"); - JSONObject validationResults = VRPConsentRequestValidator. - validateControlParameters(controlParameters); - JSONObject validationResult = VRPConsentRequestValidator. - validateMaximumIndividualAmountCurrency(controlParameters); - - Assert.assertFalse((boolean) validationResults.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertFalse((boolean) validationResult.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("parameter passed in is null", - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testInvalidCurrencyKey_MissingKeys() { - - JSONObject maximumIndividualAmount = new JSONObject(); - - JSONObject validationResults = VRPConsentRequestValidator. - validateJsonObjectKey(maximumIndividualAmount, ConsentExtensionConstants.CURRENCY, String.class); - - Assert.assertFalse((Boolean) validationResults.get(ConsentExtensionConstants.IS_VALID)); - - JSONObject parentObj = new JSONObject(); - JSONObject validationResult = VRPConsentRequestValidator.validateMaximumIndividualAmountCurrency(parentObj); - - Assert.assertFalse((Boolean) validationResult.get(ConsentExtensionConstants.IS_VALID)); - JSONObject isValid = VRPConsentRequestValidator.validateJsonObjectKey(parentObj, "Currency", String.class); - - Assert.assertFalse((Boolean) isValid.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("parameter passed in is null", - validationResult.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadWithoutControlParameterCurrency() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_CURRENCY; - JSONObject result = VRPConsentRequestValidator.validateMaximumIndividualAmountCurrency((JSONObject) JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("parameter passed in is null", - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testVrpInitiationPayloadWithoutControlParameter() { - - String initiationPayloads = VRPTestConstants.METADATA_VRP_WITHOUT_CURRENCY; - JSONObject result = VRPConsentRequestValidator.validateMaximumIndividualAmountCurrency((JSONObject) JSONValue. - parse(initiationPayloads)); - - Assert.assertFalse((boolean) result.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("parameter passed in is null", - result.get(ConsentExtensionConstants.ERRORS)); - } - - - @Test - public void testWithEmptyDate() { - - String initiationPayloads = VRPTestConstants.vrpInitiationPayloadWithoutDate; - JSONObject response = VRPConsentRequestValidator.validateParameterDateTime((JSONObject) JSONValue. - parse(initiationPayloads)); - Assert.assertFalse((boolean) response.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals(ErrorConstants.MISSING_VALID_TO_DATE_TIME, response.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateConsentRisk() { - - JSONObject requestBody = new JSONObject(); - JSONObject data = new JSONObject(); - JSONObject risk = new JSONObject(); - - data.put(ConsentExtensionConstants.RISK, risk); - requestBody.put(ConsentExtensionConstants.DATA, data); - - JSONObject result = VRPConsentRequestValidator.validateConsentRisk(requestBody); - - Assert.assertFalse(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_RISK, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateConsentRiskInvalidFormat() { - JSONObject requestBody = new JSONObject(); - requestBody.put("invalidKey", "invalidValue"); - - JSONObject result = VRPConsentRequestValidator.validateConsentRisk(requestBody); - - Assert.assertFalse(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_RISK, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateConsentRiskMissingRiskKey() { - - JSONObject requestBody = new JSONObject(); - JSONObject data = new JSONObject(); - requestBody.put(ConsentExtensionConstants.DATA, data); - - JSONObject result = VRPConsentRequestValidator.validateConsentRisk(requestBody); - - Assert.assertFalse(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_RISK, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateConsentRiskWithDataEmpty() { - - JSONObject requestBody = new JSONObject(); - requestBody.put(ConsentExtensionConstants.DATA, new JSONObject()); - - JSONObject result = VRPConsentRequestValidator.validateConsentRisk(requestBody); - - Assert.assertFalse(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_RISK, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateConsentRiskWithDataNotPresent() { - - JSONObject requestBody = new JSONObject(); - - JSONObject result = VRPConsentRequestValidator.validateConsentRisk(requestBody); - - Assert.assertFalse(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_RISK, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateConsentRiskWithDataNotEmpty() { - JSONObject requestBody = new JSONObject(); - JSONObject data = new JSONObject(); - data.put("someKey", "someValue"); - requestBody.put(ConsentExtensionConstants.DATA, data); - - JSONObject result = VRPConsentRequestValidator.validateConsentRisk(requestBody); - Assert.assertFalse(Boolean.parseBoolean(result.getAsString(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals(ErrorConstants.PAYLOAD_FORMAT_ERROR_RISK, - result.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateCurrencyWithoutAmountKeyAndEmptyString() { - - // Test case 1: parentObj is null - JSONObject result1 = VRPConsentRequestValidator. - validateJsonObjectKey(null, "Currency", String.class); - Assert.assertFalse((boolean) result1.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("parameter passed in is null", - result1.get(ConsentExtensionConstants.ERRORS)); - - // Test case 2: Key is not present in parentObj - JSONObject result2 = VRPConsentRequestValidator. - validateJsonObjectKey(new JSONObject(), "nonExistentKey", String.class); - Assert.assertFalse((boolean) result2.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Mandatory parameter 'nonExistentKey' is not present in payload", - result2.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateAmountCurrencyWithCurrencyKey() { - - // Test case 3: Invalid currency key (missing key) - JSONObject testData3 = new JSONObject(); - - JSONObject result3 = VRPConsentRequestValidator. - validateJsonObjectKey(testData3, "currency", String.class); - Assert.assertFalse((boolean) result3.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Mandatory parameter 'currency' is not present in payload", - result3.get(ConsentExtensionConstants.ERRORS)); - - // Test case 4: Invalid currency key (null parentObj) - JSONObject result4 = VRPConsentRequestValidator. - validateJsonObjectKey(null, "currency", String.class); - Assert.assertFalse((boolean) result4.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Mandatory parameter 'currency' is not present in payload", - result3.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidatePeriodicType() { - // Test case 1: Valid periodic type - JSONObject validLimitObject = new JSONObject(); - validLimitObject.put(ConsentExtensionConstants.PERIOD_TYPE, ConsentExtensionConstants.DAY); - JSONObject result1 = VRPConsentRequestValidator.validatePeriodType(validLimitObject); - Assert.assertTrue((boolean) result1.get(ConsentExtensionConstants.IS_VALID)); - - - // Test case 2: Missing period type key - JSONObject missingKeyObject = new JSONObject(); - JSONObject result2 = VRPConsentRequestValidator.validatePeriodType(missingKeyObject); - Assert.assertFalse((boolean) result2.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Missing required parameter Period type", - result2.get(ConsentExtensionConstants.ERRORS)); - - // Test case 3: Null period type - JSONObject nullPeriodTypeObject = new JSONObject(); - nullPeriodTypeObject.put(ConsentExtensionConstants.PERIOD_TYPE, null); - JSONObject result3 = VRPConsentRequestValidator.validatePeriodType(nullPeriodTypeObject); - Assert.assertFalse((boolean) result3.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Missing required parameter Period type", - result2.get(ConsentExtensionConstants.ERRORS)); - - - // Test case 4: Empty period type - JSONObject emptyPeriodTypeObject = new JSONObject(); - emptyPeriodTypeObject.put(ConsentExtensionConstants.PERIOD_TYPE, ""); - JSONObject result4 = VRPConsentRequestValidator.validatePeriodType(emptyPeriodTypeObject); - Assert.assertFalse((boolean) result4.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Missing required parameter Period type", - result2.get(ConsentExtensionConstants.ERRORS)); - - - // Test case 5: Invalid period type - JSONObject invalidPeriodTypeObject = new JSONObject(); - invalidPeriodTypeObject.put(ConsentExtensionConstants.PERIOD_TYPE, "InvalidType"); - JSONObject result5 = VRPConsentRequestValidator.validatePeriodType(invalidPeriodTypeObject); - Assert.assertFalse((boolean) result5.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Missing required parameter Period type", - result2.get(ConsentExtensionConstants.ERRORS)); - } - - @Test - public void testValidateAmountCurrencyWithoutCurrentKeyAndEmptyString() { - // Test case 1: parentObj is null - JSONObject result1 = VRPConsentRequestValidator. - validateJsonObjectKey(null, "Currency", String.class); - Assert.assertFalse(((boolean) result1.get(ConsentExtensionConstants.IS_VALID))); - Assert.assertEquals("parameter passed in is null", - result1.get(ConsentExtensionConstants.ERRORS)); - - // Test case 2: Key is not present in parentObj - JSONObject result2 = VRPConsentRequestValidator. - validateJsonObjectKey(new JSONObject(), "nonExistentKey", String.class); - Assert.assertFalse((boolean) result2.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Mandatory parameter 'nonExistentKey' is not present in payload", - result2.get(ConsentExtensionConstants.ERRORS)); - - } - - @Test - public void testValidateAmountCurrencyWithoutAmountKeyAndEmptyString() { - - // Test case 1: parentObj is null - JSONObject result1 = VRPConsentRequestValidator. - validateJsonObjectKey(null, "Amount", String.class); - Assert.assertFalse((boolean) result1.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("parameter passed in is null", - result1.get(ConsentExtensionConstants.ERRORS)); - - // Test case 2: Key is not present in parentObj - JSONObject result2 = VRPConsentRequestValidator. - validateJsonObjectKey(new JSONObject(), "nonExistentKey", String.class); - Assert.assertFalse((boolean) result2.get(ConsentExtensionConstants.IS_VALID)); - Assert.assertEquals("Mandatory parameter 'nonExistentKey' is not present in payload", - result2.get(ConsentExtensionConstants.ERRORS)); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/manage/vrp/VRPTestConstants.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/manage/vrp/VRPTestConstants.java deleted file mode 100644 index f8c8535e..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/manage/vrp/VRPTestConstants.java +++ /dev/null @@ -1,1122 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.manage.vrp; - -/** - * Constant class for consent manage tests. - */ -public class VRPTestConstants { - - public static String vrpInitiationPayloadWithoutData = "{\n" + - " \"\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static String vrpInitiationPayloadWithoutDate = "{\n" + - " \"\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"\",\n" + - " \"ValidToDateTime\": null, // Set to null instead of an empty string\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - - public static String vrpInitiationPayloadWithStringData = "{\n" + - " \"\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static String vrpInitiationPayloadWithOutJsonObject = "{\n" + - " \"\": { }" + - ",\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String METADATA_VRP_CREDITOR_ACCOUNT = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - - public static final String METADATA_VRP_DEBTOR_ACCOUNT = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - ; - - - public static final String METADATA_VRP_DEBTOR_ACCOUNT_SCHEME_NAME = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String METADATA_VRP_CREDITOR_ACCOUNT_SCHEME_NAME = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - - public static final String METADATA_VRP_DEBTOR_ACCOUNT_IDENTIFICATION = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String METADATA_VRP_CREDITOR_ACCOUNT_IDENTIFICATION = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - - public static final String METADATA_VRP_WITHOUT_INITIATION = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - - public static final String METADATA_VRP_WITHOUT_CONTROL_PARAMETERS = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - - public static final String METADATA_VRP_WITHOUT_CURRENCY = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String METADATA_VRP_WITHOUT_PERIODIC_LIMIT_CURRENCY = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String METADATA_VRP_WITHOUT_PERIODIC_LIMIT_AMOUNT = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - - public static final String METADATA_VRP_WITHOUT_PERIODIC_TYPE = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - - public static final String METADATA_VRP_WITHOUT_PERIODIC_TYPE_CURRENCY = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"\": \"\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodicType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String METADATA_VRP_WITHOUT_VALID_TO_DATE = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String METADATA_VRP_WITH_INVALID_VALID_FROM_DATETIME = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"\": \"2023-09-12T\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String METADATA_VRP_WITH_INVALID_MAX_INDIVIDUAL_AMOUNT = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": \"\",\n" + // Empty string for MaximumIndividualAmount - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - - public static final String METADATA_VRP_WITHOUT_RISK = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String METADATA_VRP_WITH_EMPTY_CONTROL_PARAMETERS = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": \"\",\n" + // Empty string for ControlParameters - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - - public static final String METADATA_VRP_EMPTY_INITIATION = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": \"\",\n" + // Empty string for Initiation - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String METADATA_VRP_WITH_EMPTY_MAX_INDIVIDUAL_AMOUNT = "{\n" + - " \"\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": \"\",\n" + // Empty string for MaximumIndividualAmount - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String METADATA_VRP_WITHOUT_VALID_FROM_DATE = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String METADATA_VRP_WITHOUT_DEB_ACC = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": \"\",\n" + // Change DebtorAccount to an empty string - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - - public static final String METADATA_VRP_WITHOUT_DEBTOR_ACC = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"\": \"\",\n" + // Change DebtorAccount to an empty string - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - - public static final String METADATA_VRP_WITHOUT_CREDITOR_ACC = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": \"\", // Change CreditorAccount to an empty string\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - } diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/AuthServletTestConstants.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/AuthServletTestConstants.java deleted file mode 100644 index a2d462af..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/AuthServletTestConstants.java +++ /dev/null @@ -1,256 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.utils; - -/** - * Constant class for OB Auth Servlet tests. - */ -public class AuthServletTestConstants { - public static final String ACCOUNT_DATA = "{" + - " \"consentData\": [" + - " {" + - " \"data\":[" + - " \"ReadAccountsBasic\"," + - " \"ReadAccountsDetail\"," + - " \"ReadBalances\"," + - " ]," + - " \"title\":\"Permissions\"" + - " }," + - " {" + - " \"data\":[\"2021-07-19T13:51:43.347+05:30\"]," + - " \"title\":\"Expiration Date Time\"" + - " }," + - " {" + - " \"data\":[\"2021-07-14T13:51:43.397+05:30\"]," + - " \"title\":\"Transaction From Date Time\"" + - " }," + - " {" + - " \"data\":[\"2021-07-17T13:51:43.397+05:30\"]," + - " \"title\":\"Transaction To Date Time\"}," + - " ]," + - " \"application\":\"9b5usDpbNtmxDcTzs7GzKp\"," + - " \"accounts\":[" + - " {" + - " \"accountId\":\"30080012343456\"," + - " \"account_id\":\"30080012343456\"," + - " \"authorizationMethod\":\"single\"," + - " \"accountName\":\"account_1\"," + - " \"nickName\":\"not-working\"," + - " \"display_name\":\"account_1\"" + - " }," + - " {" + - " \"accountId\":\"30080098763459\"," + - " \"account_id\":\"30080098763459\"," + - " \"authorizationMethod\":\"single\"," + - " \"accountName\":\"account_2\"," + - " \"display_name\":\"account_2\"" + - " }" + - " ]," + - " \"type\":\"accounts\"" + - "}"; - - public static final String COF_DATA = "{" + - " \"consentData\":[" + - " {" + - " \"data\":[\"2021-07-19T20:14:11.069+05:30\"]," + - " \"title\":\"Expiration Date Time\"" + - " }," + - " {" + - " \"data\":[" + - " \"Scheme Name : OB.SortCodeAccountNumber\"," + - " \"Identification : 1234\"," + - " \"Name : Account1\"," + - " \"Secondary Identification : Account1\"" + - " ]," + - " \"title\":\"Debtor Account\"" + - " }," + - " ]," + - " \"application\":\"9b5usDpbNtmxDcTzs7GzKp\"," + - " \"type\":\"fundsconfirmations\"," + - " \"debtor_account\":\"1234\"" + - "}"; - - public static final String PAYMENT_DATA = "{" + - " \"consentData\":[" + - " {" + - " \"data\":[\"Domestic Payments\"]," + - " \"title\":\"Payment Type\"" + - " }," + - " {" + - " \"data\":[\"ACME412\"]," + - " \"title\":\"Instruction Identification\"" + - " }," + - " {" + - " \"data\":[\"FRESCO.21302.GFX.20\"]," + - " \"title\":\"End to End Identification\"" + - " }," + - " {" + - " \"data\":[\"Amount : 30.80\",\"Currency : GBP\"]," + - " \"title\":\"Instructed Amount\"" + - " }," + - " {" + - " \"data\":[" + - " \"Scheme Name : OB.SortCodeAccountNumber\"," + - " \"Identification : 30080012343456\"," + - " \"Name : Andrea Smith\"," + - " \"Secondary Identification : 30080012343456\"" + - " ]," + - " \"title\":\"Debtor Account\"" + - " }," + - " {" + - " \"data\":[" + - " \"Scheme Name : OB.SortCodeAccountNumber\"," + - " \"Identification : 08080021325698\"," + - " \"Name : ACME Inc\"," + - " \"Secondary Identification : 0002\"" + - " ]," + - " \"title\":\"Creditor Account\"" + - " }," + - " ]," + - " \"application\":\"9b5usDpbNtmxDcTzs7GzKp\"," + - " \"type\":\"payments\"," + - " \"debtor_account\":\"30080012343456\"" + - "}"; - - public static final String PAYMENT_DATA_WITHOUT_DEBTOR_ACC = "{\n" + - " \"consentData\":[\n" + - " {\n" + - " \"data\":[\n" + - " \"Domestic Payments\"\n" + - " ],\n" + - " \"title\":\"Payment Type\"\n" + - " },\n" + - " {\n" + - " \"data\":[\n" + - " \"ACME412\"\n" + - " ],\n" + - " \"title\":\"Instruction Identification\"\n" + - " },\n" + - " {\n" + - " \"data\":[\n" + - " \"FRESCO.21302.GFX.20\"\n" + - " ],\n" + - " \"title\":\"End to End Identification\"\n" + - " },\n" + - " {\n" + - " \"data\":[\n" + - " \"Amount : 30.80\",\n" + - " \"Currency : GBP\"\n" + - " ],\n" + - " \"title\":\"Instructed Amount\"\n" + - " },\n" + - " {\n" + - " \"data\":[\n" + - " \"Scheme Name : OB.SortCodeAccountNumber\",\n" + - " \"Identification : 08080021325698\",\n" + - " \"Name : ACME Inc\"\n" + - " ],\n" + - " \"title\":\"Creditor Account\"\n" + - " },\n" + - " ]," + - " \"type\":\"Payments\"\n" + - "}"; - - - public static final String VRP_DATA = "{" + - " \"consentData\":[" + - " {" + - " \"data\":[\"Domestic VRP\"]," + - " \"title\":\"Payment Type\"" + - " }," + - " {" + - " \"data\":[" + - " \"Scheme Name : OB.SortCodeAccountNumber\"," + - " \"Identification : 30080012343456\"," + - " \"Name : Andrea Smith\"," + - " \"Secondary Identification : 30080012343456\"" + - " ]," + - " \"title\":\"Debtor Account\"" + - " }," + - " {" + - " \"data\":[" + - " \"Scheme Name : OB.SortCodeAccountNumber\"," + - " \"Identification : 08080021325698\"," + - " \"Name : ACME Inc\"," + - " \"Secondary Identification : 0002\"" + - " ]," + - " \"title\":\"Creditor Account\"" + - " }," + - " {" + - " \"data\":[\"100\"]," + - " \"title\":\"Maximum amount per payment\"" + - " }," + - " {" + - " \"data\":[\"Consent\"]," + - " \"title\":\"Period Alignment\"" + - " }," + - " {" + - " \"data\":[\"200\"]," + - " \"title\":\"Maximum payment amount per Week\"" + - " }," + - " ]," + - " \"application\":\"9b5usDpbNtmxDcTzs7GzKp\"," + - " \"type\":\"vrp\"," + - " \"debtor_account\":\"30080012343456\"" + - "}"; - - - public static final String VRP_DATA_WITHOUT_DEBTOR_ACC = "{" + - " \"consentData\":[" + - " {" + - " \"data\":[\"Domestic VRP\"]," + - " \"title\":\"Payment Type\"" + - " }," + - " {" + - " \"data\":[" + - " \"Scheme Name : OB.SortCodeAccountNumber\"," + - " \"Identification : 30080012343456\"," + - " \"Name : Andrea Smith\"," + - " \"Secondary Identification : 30080012343456\"" + - " ]," + - " \"title\":\"Debtor Account\"" + - " }," + - " {" + - " \"data\":[" + - " \"Scheme Name : OB.SortCodeAccountNumber\"," + - " \"Identification : 08080021325698\"," + - " \"Name : ACME Inc\"," + - " \"Secondary Identification : 0002\"" + - " ]," + - " \"title\":\"Creditor Account\"" + - " }," + - " {" + - " \"data\":[\"100\"]," + - " \"title\":\"Maximum amount per payment\"" + - " }," + - " {" + - " \"data\":[\"Consent\"]," + - " \"title\":\"Period Alignment\"" + - " }," + - " {" + - " \"data\":[\"200\"]," + - " \"title\":\"Maximum payment amount per Week\"" + - " }," + - " ]," + - " \"type\":\"vrp\"," + - "}"; - - public static final String JSON_WITH_TYPE = "{" + - " \"type\":\"test\"" + - "}"; -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentAuthorizeTestConstants.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentAuthorizeTestConstants.java deleted file mode 100644 index 71bac8ce..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentAuthorizeTestConstants.java +++ /dev/null @@ -1,405 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.utils; - -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; - -import java.time.Instant; -import java.time.OffsetDateTime; - -/** - * Constant class for consent authorize tests. - */ -public class ConsentAuthorizeTestConstants { - public static final String INVALID_REQUEST_OBJECT = "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.aWF0.TIygRaBn7MUFR9Zzy3" + - "yu9K8uKVe8KXdAty0Ckrg2vFI"; - public static final String VALID_REQUEST_OBJECT = "eyJhbGciOiJQUzI1NiIsInR5cCI6IkpXVCIsImtpZCI6IkR3TUtkV01tajdQV2" + - "ludm9xZlF5WFZ6eVo2USJ9.eyJtYXhfYWdlIjo4NjQwMCwiYXVkIjoiaHR0cHM6Ly9sb2NhbGhvc3Q6OTQ0Ni9vYXV0aDIvdG9rZW4iL" + - "CJzY29wZSI6Im9wZW5pZCBhY2NvdW50cyIsImlzcyI6InF3ZGZnaGpwbG1nZmRhYWhrZ2pvcGhuayIsImNsYWltcyI6eyJpZF90b2tlb" + - "iI6eyJhY3IiOnsidmFsdWVzIjpbInVybjpvcGVuYmFua2luZzpwc2QyOnNjYSIsInVybjpvcGVuYmFua2luZzpwc2QyOmNhIl0sImVzc" + - "2VudGlhbCI6dHJ1ZX0sIm9wZW5iYW5raW5nX2ludGVudF9pZCI6eyJ2YWx1ZSI6IjEyMzQ1Njc3NjU0MzIxMjM0MjM0IiwiZXNzZW50a" + - "WFsIjp0cnVlfX0sInVzZXJpbmZvIjp7Im9wZW5iYW5raW5nX2ludGVudF9pZCI6eyJ2YWx1ZSI6IjEyMzQ1Njc3NjU0MzIxMjM0MjM0I" + - "iwiZXNzZW50aWFsIjp0cnVlfX19LCJyZXNwb25zZV90eXBlIjoiY29kZSBpZF90b2tlbiIsInJlZGlyZWN0X3VyaSI6Imh0dHBzOi8vd" + - "3NvMi5jb20iLCJzdGF0ZSI6IllXbHpjRG96TVRRMiIsImV4cCI6MTY1MzcxNzQ3OCwibm9uY2UiOiJuLTBTNl9XekEyTSIsImNsaWVud" + - "F9pZCI6InF3ZGZnaGpwbG1nZmRhYWhrZ2pvcGhuayJ9.lOvcc81dqjqdv4dslB_Kg4K3TKd13UQWaUKl3dBiPPlnu9y-R84Xfx-bMMnH" + - "atYyW9hYWJcUlprIm_dqgFXauCSTgBz6-vacrXLzuaGtj07d-8bL_qta45qbpbKPTY2pnM_PXe7fzs4RMCGEoiRLRs7lJUBfIbV9GzlS" + - "pHkOZiOjiFxxeYm0cNpZRvXkZNd59_GLdW2kKmWaGQHpQ9Ci_QpQENRzF8KEV1QtNd3cK2DjL5tKSw824C6AmXp-PKfvhurqPaVkz5p-" + - "iPA6bRaNBPY4hj_nsZpfuCnE8-V7YXWXXzWbK3gWo_dMOV1CZcHS6KqP7DANqDEEP4LoN081uQ"; - - public static final OffsetDateTime EXP_DATE = OffsetDateTime.now().plusDays(50); - - public static final OffsetDateTime INVALID_EXP_DATE = OffsetDateTime.now().plusDays(0); - - public static final OffsetDateTime NULL_EXP_DATE = null; - public static final String VALID_INITIATION_OBJECT = "{\"Data\": {\"Permissions\": [\"ReadAccountsDetail\"," + - "\"ReadBalances\",\"ReadBeneficiariesDetail\",\"ReadDirectDebits\",\"ReadProducts\"," + - "\"ReadStandingOrdersDetail\",\"ReadTransactionsCredits\",\"ReadTransactionsDebits\"," + - "\"ReadTransactionsDetail\",\"ReadOffers\",\"ReadPAN\",\"ReadParty\",\"ReadPartyPSU\"," + - " \"ReadScheduledPaymentsDetail\",\"ReadStatementsDetail\"],\"ExpirationDateTime\": " + - "\"" + EXP_DATE + "\",\"TransactionFromDateTime\": \"2021-05-03T00:00:00+00:00\"," + - "\"TransactionToDateTime\": \"2021-12-03T00:00:00+00:00\"},\"Risk\": {}}"; - - public static final String INVALID_INITIATION_OBJECT = "{\"Data\": {\"Permissions\": [\"ReadAccountsDetail\"," + - "\"ReadBalances\",\"ReadBeneficiariesDetail\",\"ReadDirectDebits\",\"ReadProducts\"," + - "\"ReadStandingOrdersDetail\",\"ReadTransactionsCredits\",\"ReadTransactionsDebits\"," + - "\"ReadTransactionsDetail\",\"ReadOffers\",\"ReadPAN\",\"ReadParty\",\"ReadPartyPSU\"," + - " \"ReadScheduledPaymentsDetail\",\"ReadStatementsDetail\"],\"ExpirationDateTime\": " + - "\"" + INVALID_EXP_DATE + "\",\"TransactionFromDateTime\": \"2021-05-03T00:00:00+00:00\"," + - "\"TransactionToDateTime\": \"2021-12-03T00:00:00+00:00\"},\"Risk\": {}}"; - - - public static final String AWAITING_AUTH_STATUS = "awaitingAuthorisation"; - public static final long CREATED_TIME = Instant.now().toEpochMilli(); - public static final String ACCOUNTS = "accounts"; - public static final String PAYMENTS = "payments"; - public static final String FUNDS_CONFIRMATIONS = "fundsconfirmations"; - public static final String VRP = "vrp"; - public static final String PAYLOAD_WITH_NON_STRING_ACCOUNTID = "{\"accountIds\": [1234, 2345]}"; - public static final String CONSENT_ID = "4ae1e012-eaa7-4994-a055-c6454f0aeeb4"; - public static final String USER_ID = "admin@wso2.com"; - public static final String CLIENT_ID = "9vblw2uUr7FOfQzI0_XGzM7IRxAa"; - public static final String COF_RECEIPT = "{" + - " \"Data\": {" + - " \"DebtorAccount\": {" + - " \"SchemeName\": \"OB.IBAN\"," + - " \"Identification\": \"GB76LOYD30949301273801\"," + - " \"Name\": \"Andrea Smith\"," + - " \"SecondaryIdentification\": \"Roll 56988\"" + - " }," + - " \"ExpirationDateTime\": \"" + EXP_DATE + "\"" + - " }" + - "}"; - - public static final String INVALID_COF_RECEIPT = "{" + - " \"Data\": {" + - " \"DebtorAccount\": {" + - " \"SchemeName\": \"OB.IBAN\"," + - " \"Identification\": \"GB76LOYD30949301273801\"," + - " \"Name\": \"Andrea Smith\"," + - " \"SecondaryIdentification\": \"Roll 56988\"" + - " }," + - " \"ExpirationDateTime\": \"" + INVALID_EXP_DATE + "\"" + - " }" + - "}"; - - public static final String NULL_COF_RECEIPT = "{" + - " \"Data\": {" + - " \"DebtorAccount\": {" + - " \"SchemeName\": \"OB.IBAN\"," + - " \"Identification\": \"GB76LOYD30949301273801\"," + - " \"Name\": \"Andrea Smith\"," + - " \"SecondaryIdentification\": \"Roll 56988\"" + - " }," + - " \"ExpirationDateTime\": \"" + NULL_EXP_DATE + "\"" + - " }" + - "}"; - public static final String VRP_INITIATION = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"Yes\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2017-06-05T15:15:13+00:00\",\n" + - " \"ValidToDateTime\": \"2022-07-05T15:15:13+00:00\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"100.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"200.00\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Week\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {" + - " \"Name\": \"Andrea Smith\", " + - " \"SchemeName\": \"OB.SortCodeAccountNumber\", " + - " \"Identification\": \"30080012343456\", " + - " \"SecondaryIdentification\": \"30080012343456\"" + - " }," + - " \"CreditorAccount\": {" + - " \"Name\": \"Andrea Smith\", " + - " \"SchemeName\": \"OB.SortCodeAccountNumber\", " + - " \"Identification\": \"30080012343456\", " + - " \"SecondaryIdentification\": \"30080012343456\"" + - " }," + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_WITHOUT_CONTROLPARAMETERS = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"Yes\",\n" + - " \"\": {\n" + - " \"ValidFromDateTime\": \"2017-06-05T15:15:13+00:00\",\n" + - " \"ValidToDateTime\": \"2022-07-05T15:15:13+00:00\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"100.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"200.00\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Week\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {" + - " \"Name\": \"Andrea Smith\", " + - " \"SchemeName\": \"OB.SortCodeAccountNumber\", " + - " \"Identification\": \"30080012343456\", " + - " \"SecondaryIdentification\": \"30080012343456\"" + - " }," + - " \"CreditorAccount\": {" + - " \"Name\": \"Andrea Smith\", " + - " \"SchemeName\": \"OB.SortCodeAccountNumber\", " + - " \"Identification\": \"30080012343456\", " + - " \"SecondaryIdentification\": \"30080012343456\"" + - " }," + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_WITHOUT_DATA = "{\n" + - " \"\": {\n" + - " \"ReadRefundAccount\": \"Yes\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2017-06-05T15:15:13+00:00\",\n" + - " \"ValidToDateTime\": \"2022-07-05T15:15:13+00:00\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"100.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"200.00\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Week\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {" + - " \"Name\": \"Andrea Smith\", " + - " \"SchemeName\": \"OB.SortCodeAccountNumber\", " + - " \"Identification\": \"30080012343456\", " + - " \"SecondaryIdentification\": \"30080012343456\"" + - " }," + - " \"CreditorAccount\": {" + - " \"Name\": \"Andrea Smith\", " + - " \"SchemeName\": \"OB.SortCodeAccountNumber\", " + - " \"Identification\": \"30080012343456\", " + - " \"SecondaryIdentification\": \"30080012343456\"" + - " }," + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - static OffsetDateTime expirationInstant = OffsetDateTime.now().plusDays(50); - public static final String PAYMENT_INITIATION = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"Yes\",\n" + - " \"Authorisation\": {\n" + - " \"AuthorisationType\": \"Any\",\n" + - " \"CompletionDateTime\": \"" + expirationInstant + "\"\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"165\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"DebtorAccount\": {\n" + - "\"SchemeName\": \"OB.SortCodeAccountNumber\",\n" + - "\"Identification\": \"30080012343456\",\n" + - "\"Name\": \"Andrea Smith\",\n" + - "\"SecondaryIdentification\": \"30080012343456\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.SortCodeAccountNumber\",\n" + - " \"Identification\": \"08080021325698\",\n" + - " \"Name\": \"ACME Inc\",\n" + - " \"SecondaryIdentification\": \"0002\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"FRESCO-101\",\n" + - " \"Unstructured\": \"Internal ops code 5120101\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"EcommerceGoods\",\n" + - " \"MerchantCategoryCode\": \"5967\",\n" + - " \"MerchantCustomerIdentification\": \"053598653254\",\n" + - " \"DeliveryAddress\": {\n" + - " \"AddressLine\": [\n" + - " \"Flat 7\",\n" + - " \"Acacia Lodge\"\n" + - " ],\n" + - " \"StreetName\": \"Acacia Avenue\",\n" + - " \"BuildingNumber\": \"27\",\n" + - " \"PostCode\": \"GU31 2ZZ\",\n" + - " \"TownName\": \"Sparsholt\",\n" + - " \"CountySubDivision\": [\n" + - " \"Wessex\"\n" + - " ],\n" + - " \"Country\": \"UK\"\n" + - " }\n" + - " }\n" + - "}"; - public static final String INTERNATIONAL_PAYMENT_INITIATION = "" + - "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"Yes\",\n" + - " \"Initiation\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"InstructionPriority\": \"Normal\",\n" + - " \"CurrencyOfTransfer\": \"USD\",\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"165.88\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.SortCodeAccountNumber\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Andrea Smith\",\n" + - " \"SecondaryIdentification\": \"30080012343456\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.SortCodeAccountNumber\",\n" + - " \"Identification\": \"08080021325698\",\n" + - " \"Name\": \"ACME Inc\",\n" + - " \"SecondaryIdentification\": \"0002\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"FRESCO-101\",\n" + - " \"Unstructured\": \"Internal ops code 5120101\"\n" + - " },\n" + - " \"ExchangeRateInformation\": {\n" + - " \"UnitCurrency\": \"GBP\",\n" + - " \"RateType\": \"Actual\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"TransferToThirdParty\"\n" + - " }\n" + - "}"; - - public static final String ACCOUNT_PERSIST_PAYLOAD_WITHOUT_ACCOUNT_ID = " " + - "{" + - " \"metadata\": {" + - " \"commonAuthId\":\"b37b9c9b-b5ce-4889-966e-9cb30f70cc78\"" + - " }," + - " \"cofAccount\":\"\"," + - " \"approval\":\"true\"," + - " \"accountIds\": \"\"," + - " \"isReauthorization\":\"false\"," + - " \"type\":\"accounts\"," + - " \"paymentAccount\":\"\"" + - "}"; - - public static final String COF_PERSIST_PAYLOAD = " " + - "{" + - " \"metadata\": {" + - " \"commonAuthId\":\"b37b9c9b-b5ce-4889-966e-9cb30f70cc78\"" + - " }," + - " \"approval\":\"true\"," + - " \"cofAccount\":\"1234\"," + - " \"accountIds\": \"\"," + - " \"type\":\"accounts\"," + - "}"; - - public static final String COF_PERSIST_PAYLOAD_WITHOUT_COF_ACC = " " + - "{" + - " \"metadata\": {" + - " \"commonAuthId\":\"b37b9c9b-b5ce-4889-966e-9cb30f70cc78\"" + - " }," + - " \"approval\":\"true\"," + - " \"accountIds\": \"\"," + - " \"isReauthorization\":\"false\"," + - " \"type\":\"accounts\"," + - " \"paymentAccount\":\"\"" + - "}"; - - public static final String COF_PERSIST_PAYLOAD_WITH_NON_STRING_COF_ACC = " " + - "{" + - " \"metadata\": {" + - " \"commonAuthId\":\"b37b9c9b-b5ce-4889-966e-9cb30f70cc78\"" + - " }," + - " \"cofAccount\":1234," + - " \"approval\":\"true\"," + - " \"accountIds\": \"\"," + - " \"isReauthorization\":\"false\"," + - " \"type\":\"accounts\"," + - " \"paymentAccount\":\"\"" + - "}"; - - public static AuthorizationResource getAuthResource() { - - AuthorizationResource authorizationResource = new AuthorizationResource(); - authorizationResource.setAuthorizationID("1234"); - authorizationResource.setConsentID(ConsentAuthorizeTestConstants.CONSENT_ID); - authorizationResource.setAuthorizationStatus("created"); - authorizationResource.setAuthorizationType("authorization"); - - return authorizationResource; - } - - public static final String ACCOUNT_PERSIST_PAYLOAD = " " + - "{" + - " \"metadata\": {" + - " \"commonAuthId\":\"b37b9c9b-b5ce-4889-966e-9cb30f70cc78\"" + - " }," + - " \"cofAccount\":\"\"," + - " \"approval\":\"true\"," + - " \"accountIds\":[" + - " \"30080012343456\"" + - " ]," + - " \"type\":\"accounts\"," + - " \"paymentAccount\":\"\"" + - "}"; - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentExtensionDataProvider.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentExtensionDataProvider.java deleted file mode 100644 index 19ea53c5..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentExtensionDataProvider.java +++ /dev/null @@ -1,49 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.utils; - -import org.testng.annotations.DataProvider; - -/** - * Data Provider for Consent Executor Tests. - */ -public class ConsentExtensionDataProvider { - - @DataProvider(name = "VRPInvalidSubmissionPayloadsDataProvider") - Object[][] getVRPInvalidSubmissionPayloadsDataProvider() { - - return new Object[][]{ - {ConsentValidateTestConstants.VRP_SUBMISSION_WITHOUT_INSTRUCTION_IDENTIFICATION}, - {ConsentValidateTestConstants.VRP_SUBMISSION_WITHOUT_END_TO_IDENTIFICATION}, - {ConsentValidateTestConstants.VRP_SUBMISSION_WITHOUT_INSTRUCTED_AMOUNT}, - {ConsentValidateTestConstants.VRP_SUBMISSION_WITHOUT_INSTRUCTION_CREDITOR_ACC}, - {ConsentValidateTestConstants.VRP_SUBMISSION_WITHOUT_INSTRUCTION_REMITTANCE_INFO}, - }; - } - - @DataProvider(name = "VRPInvalidInitiationSubmissionPayloadsDataProvider") - Object[][] getVRPInvalidInitiationSubmissionPayloadsDataProvider() { - - return new Object[][]{ - {ConsentValidateTestConstants.VRP_SUBMISSION_WITHOUT_CREDITOR_ACC}, - {ConsentValidateTestConstants.VRP_SUBMISSION_WITHOUT_REMITTANCE_INFO}, - {ConsentValidateTestConstants.VRP_SUBMISSION_WITHOUT_DEBTOR_ACC}, - }; - } - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentExtensionTestConstants.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentExtensionTestConstants.java deleted file mode 100644 index d7ccaa71..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentExtensionTestConstants.java +++ /dev/null @@ -1,65 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.utils; - -/** -comment. - */ -public class ConsentExtensionTestConstants { - - public static final String VALID_INITIATION_OBJECT = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentExtensionTestUtils.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentExtensionTestUtils.java deleted file mode 100644 index b945da8d..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentExtensionTestUtils.java +++ /dev/null @@ -1,98 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.utils; - -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentAttributes; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; - -import java.lang.reflect.Field; -import java.time.OffsetDateTime; -import java.util.HashMap; -import java.util.Map; - - -/** - * Utils class for consent executor tests. - */ -public class ConsentExtensionTestUtils { - - static JSONParser parser = new JSONParser(JSONParser.MODE_PERMISSIVE); - - public static void injectEnvironmentVariable(String key, String value) - throws ReflectiveOperationException { - - Class processEnvironment = Class.forName("java.lang.ProcessEnvironment"); - - Field unmodifiableMapField = getAccessibleField(processEnvironment, "theUnmodifiableEnvironment"); - Object unmodifiableMap = unmodifiableMapField.get(null); - injectIntoUnmodifiableMap(key, value, unmodifiableMap); - - Field mapField = getAccessibleField(processEnvironment, "theEnvironment"); - Map map = (Map) mapField.get(null); - map.put(key, value); - } - - private static Field getAccessibleField(Class clazz, String fieldName) - throws NoSuchFieldException { - - Field field = clazz.getDeclaredField(fieldName); - field.setAccessible(true); - return field; - } - - private static void injectIntoUnmodifiableMap(String key, String value, Object map) - throws ReflectiveOperationException { - - Class unmodifiableMap = Class.forName("java.util.Collections$UnmodifiableMap"); - Field field = getAccessibleField(unmodifiableMap, "m"); - Object obj = field.get(map); - ((Map) obj).put(key, value); - } - - public static JSONObject getInitiationPayload(JSONObject payload) { - return (JSONObject) ((JSONObject) payload.get(ConsentExtensionConstants.DATA)) - .get(ConsentExtensionConstants.INITIATION); - } - - public static JSONObject getJsonPayload(String payload) throws ParseException { - return (JSONObject) parser.parse(payload); - } - - public static ConsentAttributes getConsentAttributes(String paymentType) { - - Map consentAttributesMap = new HashMap(); - consentAttributesMap.put(ConsentExtensionConstants.MAXIMUM_INDIVIDUAL_AMOUNT, "100.00"); - consentAttributesMap.put(ConsentExtensionConstants.PAYMENT_TYPE, paymentType); - consentAttributesMap.put(ConsentExtensionConstants.PAID_AMOUNT, "20.00"); - consentAttributesMap.put(ConsentExtensionConstants.LAST_PAYMENT_DATE, - OffsetDateTime.now().minusDays(50).toString()); - consentAttributesMap.put(ConsentExtensionConstants.PREVIOUS_PAID_AMOUNT, "20.00"); - consentAttributesMap.put(ConsentExtensionConstants.PREVIOUS_LAST_PAYMENT_DATE, - OffsetDateTime.now().minusDays(50).toString()); - - ConsentAttributes consentAttributes = new ConsentAttributes(); - consentAttributes.setConsentAttributes(consentAttributesMap); - - return consentAttributes; - } - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentValidateTestConstants.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentValidateTestConstants.java deleted file mode 100644 index 0411286d..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/utils/ConsentValidateTestConstants.java +++ /dev/null @@ -1,1091 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.utils; - -import java.time.OffsetDateTime; - -/** - * comment. - */ -public class ConsentValidateTestConstants { - public static final OffsetDateTime EXPIRATION_DATE = OffsetDateTime.now().plusDays(50); - public static final String CONSENT_ID = "0ba972a9-08cd-4cad-b7e2-20655bcbd9e0"; - public static final String INVALID_CONSENT_ID = "0ba972a9-08cd-4cad-b7e2-20655bcbd9e0"; - public static final String INVALID_CONSENT_TYPE = "InvalidConsentType"; - public static final String VRP_PATH = "/domestic-vrps"; - public static final String PAYMENT_PATH = "/domestic-payments"; - public static final String USER_ID = "admin@wso2.com"; - public static final String CLIENT_ID = "xzX8t9fx6VxYMx_B6Lgpd5_yyUEa"; - public static final String SAMPLE_AUTHORIZATION_TYPE = "authorizationType"; - public static final String VRP_INITIATION = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_INITIATION_WITHOUT_DEBTOR_ACC = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_INITIATION_WITHOUT_CREDITOR_ACC = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - public static final String VRP_SUBMISSION = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_SUBMISSION_WITHOUT_RISK = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - "}"; - - public static final String VRP_SUBMISSION_WITH_INVALID_INSTRUCTION = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"UK.OBIE.IBAN\",\n" + - " \"Identification\": \"GB76LOYD30949301273801\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"SortCodeAccountNumber\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_SUBMISSION_WITH_INVALID_RISK = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"CreditToThirdParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_SUBMISSION_WITHOUT_INSTRUCTION = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - " \"PSUInteractionType\": \"OffSession\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"GB76LOYD30949301273801\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"SortCodeAccountNumber\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_SUBMISSION_WITHOUT_CREDITOR_ACC = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_SUBMISSION_WITHOUT_INSTRUCTION_REMITTANCE_INFO = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_SUBMISSION_DEBTOR_ACC_MISMATCH = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_SUBMISSION_WITHOUT_REMITTANCE_INFO = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_SUBMISSION_WITHOUT_REMITTANCE_INFO_MISMATCH = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"ThirdParty\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_SUBMISSION_WITHOUT_INSTRUCTED_AMOUNT = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_SUBMISSION_WITHOUT_INSTRUCTION_IDENTIFICATION = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_SUBMISSION_WITHOUT_END_TO_IDENTIFICATION = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"\": \"FRESCO.21302.GFX.20\",\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_SUBMISSION_WITHOUT_DEBTOR_ACC = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_SUBMISSION_WITH_INTEGER_INSTRUCTION_IDENTIFICATION = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"InstructionIdentification\": 788,\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_SUBMISSION_WITH_INTEGER_END_TO_IDENTIFICATION = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": 5666,\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_SUBMISSION_WITHOUT_INSTRUCTION_REMITTANCE_INFO_MISMATCH = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"ThirdParty\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_SUBMISSION_WITH_DEBTOR_ACC = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_SUBMISSION_WITH_INSTRUCTION_CREDITOR_ACC = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_SUBMISSION_WITHOUT_INSTRUCTION_CREDITOR_ACC = "{\n" + - " \"Data\": {\n" + - " \"ConsentId\": \"" + CONSENT_ID + "\",\n" + - " \"PSUAuthenticationMethod\": \"OB.SCA\",\n" + - "\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " },\n" + - "\n" + - " \"Instruction\": {\n" + - " \"InstructionIdentification\": \"ACME412\",\n" + - " \"EndToEndIdentification\": \"FRESCO.21302.GFX.20\",\n" + - " \"\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - "\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; - - public static final String VRP_INSTRUCTION = "{\n" + - " \"Data\": {\n" + - " \"ReadRefundAccount\": \"true\",\n" + - " \"ControlParameters\": {\n" + - " \"ValidFromDateTime\": \"2023-09-12T12:43:07.956Z\",\n" + - " \"ValidToDateTime\": \"2024-05-12T12:43:07.956Z\",\n" + - " \"MaximumIndividualAmount\": {\n" + - " \"Amount\": \"9\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"PeriodicLimits\": [\n" + - " {\n" + - " \"Amount\": \"1000\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"PeriodAlignment\": \"Consent\",\n" + - " \"PeriodType\": \"Half-year\"\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Initiation\": {\n" + - " \"DebtorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30080012343456\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"CreditorAccount\": {\n" + - " \"SchemeName\": \"OB.IBAN\",\n" + - " \"Identification\": \"30949330000010\",\n" + - " \"SecondaryIdentification\": \"Roll 90210\",\n" + - " \"Name\": \"Marcus Sweepimus\"\n" + - " },\n" + - " \"RemittanceInformation\": {\n" + - " \"Reference\": \"Sweepco\"\n" + - " }\n" + - " }\n" + - " },\n" + - " \"Risk\": {\n" + - " \"PaymentContextCode\": \"PartyToParty\"\n" + - " }\n" + - "}"; -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/validate/VRPSubmissionTest.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/validate/VRPSubmissionTest.java deleted file mode 100644 index 70c17895..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/java/com/wso2/openbanking/accelerator/consent/extensions/validate/VRPSubmissionTest.java +++ /dev/null @@ -1,887 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.extensions.validate; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.util.CarbonUtils; -import com.wso2.openbanking.accelerator.common.util.ErrorConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentServiceUtil; -import com.wso2.openbanking.accelerator.consent.extensions.utils.ConsentExtensionDataProvider; -import com.wso2.openbanking.accelerator.consent.extensions.utils.ConsentExtensionTestConstants; -import com.wso2.openbanking.accelerator.consent.extensions.utils.ConsentExtensionTestUtils; -import com.wso2.openbanking.accelerator.consent.extensions.utils.ConsentValidateTestConstants; -import com.wso2.openbanking.accelerator.consent.extensions.validate.impl.DefaultConsentValidator; -import com.wso2.openbanking.accelerator.consent.extensions.validate.impl.VRPSubmissionPayloadValidator; -import com.wso2.openbanking.accelerator.consent.extensions.validate.model.ConsentValidateData; -import com.wso2.openbanking.accelerator.consent.extensions.validate.model.ConsentValidationResult; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.joda.time.Instant; -import org.mockito.Mock; -import org.mockito.Mockito; -import org.mockito.MockitoAnnotations; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Map; - -import static org.mockito.Mockito.doReturn; -import static org.mockito.Mockito.mock; - -/** - * Test class for validating Variable Recurring Payment submission requests. - */ -@PrepareForTest({OpenBankingConfigParser.class, OpenBankingConfigParser.class, ConsentServiceUtil.class}) -@PowerMockIgnore({"com.wso2.openbanking.accelerator.consent.extensions.common.*", "net.minidev.*", - "jdk.internal.reflect.*"}) -public class VRPSubmissionTest { - VRPSubmissionPayloadValidator validator = new VRPSubmissionPayloadValidator(); - DefaultConsentValidator consentValidator; - @Mock - ConsentValidateData consentValidateDataMock; - @Mock - DetailedConsentResource detailedConsentResourceMock; - @Mock - ConsentCoreServiceImpl consentCoreServiceMock; - @Mock - ConsentValidationResult consentValidationResultMock; - Map resourceParams = new HashMap<>(); - JSONObject headers = new JSONObject(); - private static Map configMap; - Map consentAttributes = new HashMap<>(); - ArrayList authorizationResources = new ArrayList(); - - @BeforeClass - public void initClass() throws ReflectiveOperationException { - MockitoAnnotations.initMocks(this); - - //to execute util class initialization - new CarbonUtils(); - System.setProperty("some.property", "property.value"); - System.setProperty("carbon.home", "."); - ConsentExtensionTestUtils.injectEnvironmentVariable("CARBON_HOME", "."); - - configMap = new HashMap<>(); - configMap.put("ErrorURL", "https://localhost:8243/error"); - - consentValidator = new DefaultConsentValidator(); - consentValidateDataMock = mock(ConsentValidateData.class); - authorizationResources.add(getAuthorizationResource()); - detailedConsentResourceMock = mock(DetailedConsentResource.class); - consentCoreServiceMock = mock(ConsentCoreServiceImpl.class); - } - - @BeforeMethod - public void initMethod() { - - OpenBankingConfigParser openBankingConfigParserMock = mock(OpenBankingConfigParser.class); - doReturn(configMap).when(openBankingConfigParserMock).getConfiguration(); - - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - @Test - public void testValidateInitiation() throws ParseException { - - JSONObject initPayload = ConsentExtensionTestUtils.getJsonPayload( - ConsentValidateTestConstants.VRP_INITIATION); - JSONObject subPayload = ConsentExtensionTestUtils.getJsonPayload( - ConsentValidateTestConstants.VRP_SUBMISSION); - - JSONObject validationResult = validator.validateInitiation( - ConsentExtensionTestUtils.getInitiationPayload(subPayload), - ConsentExtensionTestUtils.getInitiationPayload(initPayload)); - - Assert.assertTrue((Boolean) validationResult.get(ConsentExtensionConstants.IS_VALID_PAYLOAD)); - } - - @Test - public void testCreditorAccInInstruction() throws ParseException { - - JSONObject initPayload = ConsentExtensionTestUtils.getJsonPayload( - ConsentValidateTestConstants.VRP_INITIATION_WITHOUT_CREDITOR_ACC); - JSONObject subPayload = ConsentExtensionTestUtils.getJsonPayload( - ConsentValidateTestConstants.VRP_SUBMISSION); - - JSONObject validationResult = validator.validateCreditorAcc( - ConsentExtensionTestUtils.getInitiationPayload(subPayload), - ConsentExtensionTestUtils.getInitiationPayload(initPayload)); - - Assert.assertTrue((Boolean) validationResult.get(ConsentExtensionConstants.IS_VALID_PAYLOAD)); - } - - @Test - public void testValidateVRPSubmission() throws ParseException, ConsentManagementException { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(detailedConsentResourceMock).getClientID(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionConstants.VRP).when(detailedConsentResourceMock).getConsentType(); - doReturn(ConsentValidateTestConstants.VRP_INITIATION).when(detailedConsentResourceMock).getReceipt(); - doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(detailedConsentResourceMock).getCurrentStatus(); - - doReturn(getVRPConsentAttributes()).when(detailedConsentResourceMock).getConsentAttributes(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(detailedConsentResourceMock).getConsentID(); - doReturn(ConsentValidateTestConstants.USER_ID).when(consentValidateDataMock).getUserId(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(consentValidateDataMock).getClientId(); - - doReturn(ConsentValidateTestConstants.VRP_PATH).when(consentValidateDataMock).getRequestPath(); - doReturn(resourceParams).when(consentValidateDataMock).getResourceParams(); - doReturn(headers).when(consentValidateDataMock).getHeaders(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(consentValidateDataMock).getConsentId(); - JSONObject submissionPayload = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE) - .parse(ConsentValidateTestConstants.VRP_SUBMISSION); - doReturn(submissionPayload).when(consentValidateDataMock).getPayload(); - - doReturn(ConsentExtensionTestUtils.getConsentAttributes("vrp")) - .when(consentCoreServiceMock).getConsentAttributes(Mockito.anyString()); - doReturn(true).when(consentCoreServiceMock).deleteConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - doReturn(true).when(consentCoreServiceMock).storeConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - PowerMockito.when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceMock); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertTrue(consentValidationResult.isValid()); - } - @Test - public void testValidateVRPSubmissionWithoutRisk() throws ParseException, ConsentManagementException { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(detailedConsentResourceMock).getClientID(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionConstants.VRP).when(detailedConsentResourceMock).getConsentType(); - doReturn(ConsentValidateTestConstants.VRP_INITIATION).when(detailedConsentResourceMock).getReceipt(); - doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(detailedConsentResourceMock).getCurrentStatus(); - - doReturn(getVRPConsentAttributes()).when(detailedConsentResourceMock).getConsentAttributes(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(detailedConsentResourceMock).getConsentID(); - doReturn(ConsentValidateTestConstants.USER_ID).when(consentValidateDataMock).getUserId(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(consentValidateDataMock).getClientId(); - - doReturn(ConsentValidateTestConstants.VRP_PATH).when(consentValidateDataMock).getRequestPath(); - doReturn(resourceParams).when(consentValidateDataMock).getResourceParams(); - doReturn(headers).when(consentValidateDataMock).getHeaders(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(consentValidateDataMock).getConsentId(); - JSONObject submissionPayload = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE) - .parse(ConsentValidateTestConstants.VRP_SUBMISSION_WITHOUT_RISK); - doReturn(submissionPayload).when(consentValidateDataMock).getPayload(); - - doReturn(ConsentExtensionTestUtils.getConsentAttributes("vrp")) - .when(consentCoreServiceMock).getConsentAttributes(Mockito.anyString()); - doReturn(true).when(consentCoreServiceMock).deleteConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - doReturn(true).when(consentCoreServiceMock).storeConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - PowerMockito.when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceMock); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertFalse(consentValidationResult.isValid()); - Assert.assertEquals(consentValidationResult.getErrorMessage(), ErrorConstants.RISK_NOT_FOUND); - Assert.assertEquals(consentValidationResult.getErrorCode(), ErrorConstants.FIELD_MISSING); - Assert.assertEquals(consentValidationResult.getHttpCode(), 400); - } - - @Test - public void testConsentValidateWithUserIdMismatch() { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionTestConstants.VALID_INITIATION_OBJECT).when(detailedConsentResourceMock) - .getReceipt(); - doReturn(resourceParams).when(consentValidateDataMock).getResourceParams(); - doReturn(headers).when(consentValidateDataMock).getHeaders(); - doReturn("psu1@wso2.com").when(consentValidateDataMock).getUserId(); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertFalse(consentValidationResult.isValid()); - Assert.assertEquals(consentValidationResult.getErrorMessage(), ErrorConstants.INVALID_USER_ID);; - Assert.assertEquals(consentValidationResult.getErrorCode(), ErrorConstants.RESOURCE_CONSENT_MISMATCH); - Assert.assertEquals(consentValidationResult.getHttpCode(), 400); - } - - @Test - public void testValidateVRPSubmissionWithInvalidStatus() { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(detailedConsentResourceMock).getClientID(); - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionConstants.VRP).when(detailedConsentResourceMock).getConsentType(); - doReturn(ConsentExtensionTestConstants.VALID_INITIATION_OBJECT).when(detailedConsentResourceMock) - .getReceipt(); - doReturn(ConsentExtensionConstants.AWAITING_AUTH_STATUS).when(detailedConsentResourceMock).getCurrentStatus(); - doReturn(ConsentValidateTestConstants.VRP_PATH).when(consentValidateDataMock).getRequestPath(); - doReturn(ConsentValidateTestConstants.USER_ID).when(consentValidateDataMock).getUserId(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(consentValidateDataMock).getClientId(); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertFalse(consentValidationResult.isValid()); - Assert.assertEquals(consentValidationResult.getErrorMessage(), ErrorConstants.VRP_CONSENT_STATUS_INVALID); - Assert.assertEquals(consentValidationResult.getErrorCode(), ErrorConstants.RESOURCE_INVALID_CONSENT_STATUS); - Assert.assertEquals(consentValidationResult.getHttpCode(), 400); - } - - @Test - public void testValidateVRPSubmissionWithInvalidInstruction() throws ParseException, ConsentManagementException { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(detailedConsentResourceMock).getClientID(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionConstants.VRP).when(detailedConsentResourceMock).getConsentType(); - doReturn(ConsentValidateTestConstants.VRP_INITIATION).when(detailedConsentResourceMock).getReceipt(); - doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(detailedConsentResourceMock).getCurrentStatus(); - - doReturn(getVRPConsentAttributes()).when(detailedConsentResourceMock).getConsentAttributes(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(detailedConsentResourceMock).getConsentID(); - doReturn(ConsentValidateTestConstants.USER_ID).when(consentValidateDataMock).getUserId(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(consentValidateDataMock).getClientId(); - - doReturn(ConsentValidateTestConstants.VRP_PATH).when(consentValidateDataMock).getRequestPath(); - doReturn(resourceParams).when(consentValidateDataMock).getResourceParams(); - doReturn(headers).when(consentValidateDataMock).getHeaders(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(consentValidateDataMock).getConsentId(); - JSONObject submissionPayload = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE) - .parse(ConsentValidateTestConstants.VRP_SUBMISSION_WITH_INVALID_INSTRUCTION); - doReturn(submissionPayload).when(consentValidateDataMock).getPayload(); - - doReturn(ConsentExtensionTestUtils.getConsentAttributes("vrp")) - .when(consentCoreServiceMock).getConsentAttributes(Mockito.anyString()); - doReturn(true).when(consentCoreServiceMock).deleteConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - doReturn(true).when(consentCoreServiceMock).storeConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - PowerMockito.when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceMock); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertFalse(consentValidationResult.isValid()); - Assert.assertEquals(consentValidationResult.getErrorMessage(), - ErrorConstants.CREDITOR_ACC_SCHEME_NAME_MISMATCH); - Assert.assertEquals(consentValidationResult.getErrorCode(), ErrorConstants.RESOURCE_CONSENT_MISMATCH); - Assert.assertEquals(consentValidationResult.getHttpCode(), 400); - } - - @Test - public void testValidateVRPSubmissionWithInvalidRisk() throws ParseException, ConsentManagementException { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(detailedConsentResourceMock).getClientID(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionConstants.VRP).when(detailedConsentResourceMock).getConsentType(); - doReturn(ConsentValidateTestConstants.VRP_INITIATION).when(detailedConsentResourceMock).getReceipt(); - doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(detailedConsentResourceMock).getCurrentStatus(); - - doReturn(getVRPConsentAttributes()).when(detailedConsentResourceMock).getConsentAttributes(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(detailedConsentResourceMock).getConsentID(); - doReturn(ConsentValidateTestConstants.USER_ID).when(consentValidateDataMock).getUserId(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(consentValidateDataMock).getClientId(); - - doReturn(ConsentValidateTestConstants.VRP_PATH).when(consentValidateDataMock).getRequestPath(); - doReturn(resourceParams).when(consentValidateDataMock).getResourceParams(); - doReturn(headers).when(consentValidateDataMock).getHeaders(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(consentValidateDataMock).getConsentId(); - JSONObject submissionPayload = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE) - .parse(ConsentValidateTestConstants.VRP_SUBMISSION_WITH_INVALID_RISK); - doReturn(submissionPayload).when(consentValidateDataMock).getPayload(); - - doReturn(ConsentExtensionTestUtils.getConsentAttributes("vrp")) - .when(consentCoreServiceMock).getConsentAttributes(Mockito.anyString()); - doReturn(true).when(consentCoreServiceMock).deleteConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - doReturn(true).when(consentCoreServiceMock).storeConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - PowerMockito.when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceMock); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertFalse(consentValidationResult.isValid()); - Assert.assertEquals(consentValidationResult.getErrorMessage(), ErrorConstants.RISK_PARAMETER_MISMATCH); - Assert.assertEquals(consentValidationResult.getErrorCode(), ErrorConstants.RESOURCE_CONSENT_MISMATCH); - Assert.assertEquals(consentValidationResult.getHttpCode(), 400); - } - - @Test - public void testValidateVRPSubmissionWithoutInstruction() throws ParseException { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(detailedConsentResourceMock).getClientID(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionConstants.VRP).when(detailedConsentResourceMock).getConsentType(); - doReturn(ConsentValidateTestConstants.VRP_INITIATION).when(detailedConsentResourceMock).getReceipt(); - doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(detailedConsentResourceMock).getCurrentStatus(); - - doReturn(getVRPConsentAttributes()).when(detailedConsentResourceMock).getConsentAttributes(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(detailedConsentResourceMock).getConsentID(); - doReturn(ConsentValidateTestConstants.USER_ID).when(consentValidateDataMock).getUserId(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(consentValidateDataMock).getClientId(); - - doReturn(ConsentValidateTestConstants.VRP_PATH).when(consentValidateDataMock).getRequestPath(); - doReturn(resourceParams).when(consentValidateDataMock).getResourceParams(); - doReturn(headers).when(consentValidateDataMock).getHeaders(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(consentValidateDataMock).getConsentId(); - JSONObject submissionPayload = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE) - .parse(ConsentValidateTestConstants.VRP_SUBMISSION_WITHOUT_INSTRUCTION); - doReturn(submissionPayload).when(consentValidateDataMock).getPayload(); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertFalse(consentValidationResult.isValid()); - Assert.assertEquals(consentValidationResult.getErrorMessage(), ErrorConstants.INSTRUCTION_NOT_FOUND); - Assert.assertEquals(consentValidationResult.getErrorCode(), ErrorConstants.FIELD_MISSING); - Assert.assertEquals(consentValidationResult.getHttpCode(), 400); - } - - private Map getVRPConsentAttributes() { - consentAttributes.put(ConsentExtensionConstants.PAYMENT_TYPE, "domestic-vrp-consents"); - consentAttributes.put(ConsentExtensionConstants.MAXIMUM_INDIVIDUAL_AMOUNT, "100.00"); - consentAttributes.put(ConsentExtensionConstants.PERIOD_ALIGNMENT, "Consent"); - consentAttributes.put(ConsentExtensionConstants.PERIOD_TYPE, "Week"); - consentAttributes.put(ConsentExtensionConstants.LAST_PAYMENT_DATE, - Long.toString(ConsentValidateTestConstants.EXPIRATION_DATE.toEpochSecond())); - consentAttributes.put(ConsentExtensionConstants.AMOUNT, "100.00"); - - return consentAttributes; - } - - private AuthorizationResource getAuthorizationResource() { - return new AuthorizationResource(ConsentValidateTestConstants.CONSENT_ID, - ConsentValidateTestConstants.USER_ID, "awaitingAuthorization", - ConsentValidateTestConstants.SAMPLE_AUTHORIZATION_TYPE, Instant.now().getMillis()); - } - - @Test - public void testValidateVRPSubmissionWithoutCreditorAccount() throws ParseException, ConsentManagementException { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(detailedConsentResourceMock).getClientID(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionConstants.VRP).when(detailedConsentResourceMock).getConsentType(); - doReturn(ConsentValidateTestConstants.VRP_INITIATION).when(detailedConsentResourceMock).getReceipt(); - doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(detailedConsentResourceMock).getCurrentStatus(); - - doReturn(getVRPConsentAttributes()).when(detailedConsentResourceMock).getConsentAttributes(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(detailedConsentResourceMock).getConsentID(); - doReturn(ConsentValidateTestConstants.USER_ID).when(consentValidateDataMock).getUserId(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(consentValidateDataMock).getClientId(); - - doReturn(ConsentValidateTestConstants.VRP_PATH).when(consentValidateDataMock).getRequestPath(); - doReturn(resourceParams).when(consentValidateDataMock).getResourceParams(); - doReturn(headers).when(consentValidateDataMock).getHeaders(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(consentValidateDataMock).getConsentId(); - JSONObject submissionPayload = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE) - .parse(ConsentValidateTestConstants.VRP_SUBMISSION_WITHOUT_CREDITOR_ACC); - doReturn(submissionPayload).when(consentValidateDataMock).getPayload(); - - doReturn(ConsentExtensionTestUtils.getConsentAttributes("vrp")) - .when(consentCoreServiceMock).getConsentAttributes(Mockito.anyString()); - doReturn(true).when(consentCoreServiceMock).deleteConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - doReturn(true).when(consentCoreServiceMock).storeConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - PowerMockito.when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceMock); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertFalse(consentValidationResult.isValid()); - Assert.assertEquals(consentValidationResult.getErrorMessage(), ErrorConstants.CREDITOR_ACC_NOT_FOUND); - Assert.assertEquals(consentValidationResult.getErrorCode(), ErrorConstants.FIELD_MISSING); - Assert.assertEquals(consentValidationResult.getHttpCode(), 400); - } - - @Test - public void testValidateVRPSubmissionWithDebtorAccountMisMatch() throws ParseException, - ConsentManagementException { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(detailedConsentResourceMock).getClientID(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionConstants.VRP).when(detailedConsentResourceMock).getConsentType(); - doReturn(ConsentValidateTestConstants.VRP_INITIATION).when(detailedConsentResourceMock).getReceipt(); - doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(detailedConsentResourceMock).getCurrentStatus(); - - doReturn(getVRPConsentAttributes()).when(detailedConsentResourceMock).getConsentAttributes(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(detailedConsentResourceMock).getConsentID(); - doReturn(ConsentValidateTestConstants.USER_ID).when(consentValidateDataMock).getUserId(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(consentValidateDataMock).getClientId(); - - doReturn(ConsentValidateTestConstants.VRP_PATH).when(consentValidateDataMock).getRequestPath(); - doReturn(resourceParams).when(consentValidateDataMock).getResourceParams(); - doReturn(headers).when(consentValidateDataMock).getHeaders(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(consentValidateDataMock).getConsentId(); - JSONObject submissionPayload = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE) - .parse(ConsentValidateTestConstants.VRP_SUBMISSION_DEBTOR_ACC_MISMATCH); - doReturn(submissionPayload).when(consentValidateDataMock).getPayload(); - - doReturn(ConsentExtensionTestUtils.getConsentAttributes("vrp")) - .when(consentCoreServiceMock).getConsentAttributes(Mockito.anyString()); - doReturn(true).when(consentCoreServiceMock).deleteConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - doReturn(true).when(consentCoreServiceMock).storeConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - PowerMockito.when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceMock); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertFalse(consentValidationResult.isValid()); - Assert.assertEquals(consentValidationResult.getErrorMessage(), ErrorConstants.DEBTOR_ACC_NOT_FOUND); - Assert.assertEquals(consentValidationResult.getErrorCode(), ErrorConstants.FIELD_MISSING); - Assert.assertEquals(consentValidationResult.getHttpCode(), 400); - } - - @Test - public void testValidateVRPSubmissionWithoutRemittanceInfo() throws ParseException, - ConsentManagementException { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(detailedConsentResourceMock).getClientID(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionConstants.VRP).when(detailedConsentResourceMock).getConsentType(); - doReturn(ConsentValidateTestConstants.VRP_INITIATION).when(detailedConsentResourceMock).getReceipt(); - doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(detailedConsentResourceMock).getCurrentStatus(); - - doReturn(getVRPConsentAttributes()).when(detailedConsentResourceMock).getConsentAttributes(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(detailedConsentResourceMock).getConsentID(); - doReturn(ConsentValidateTestConstants.USER_ID).when(consentValidateDataMock).getUserId(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(consentValidateDataMock).getClientId(); - - doReturn(ConsentValidateTestConstants.VRP_PATH).when(consentValidateDataMock).getRequestPath(); - doReturn(resourceParams).when(consentValidateDataMock).getResourceParams(); - doReturn(headers).when(consentValidateDataMock).getHeaders(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(consentValidateDataMock).getConsentId(); - JSONObject submissionPayload = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE) - .parse(ConsentValidateTestConstants.VRP_SUBMISSION_WITHOUT_REMITTANCE_INFO); - doReturn(submissionPayload).when(consentValidateDataMock).getPayload(); - - doReturn(ConsentExtensionTestUtils.getConsentAttributes("vrp")) - .when(consentCoreServiceMock).getConsentAttributes(Mockito.anyString()); - doReturn(true).when(consentCoreServiceMock).deleteConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - doReturn(true).when(consentCoreServiceMock).storeConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - PowerMockito.when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceMock); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertFalse(consentValidationResult.isValid()); - Assert.assertEquals(consentValidationResult.getErrorMessage(), - ErrorConstants.REMITTANCE_INFO_NOT_FOUND); - Assert.assertEquals(consentValidationResult.getErrorCode(), ErrorConstants.FIELD_MISSING); - Assert.assertEquals(consentValidationResult.getHttpCode(), 400); - } - - @Test - public void testValidateVRPSubmissionWithRemittanceInfoMisMatch() throws ParseException, - ConsentManagementException { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(detailedConsentResourceMock).getClientID(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionConstants.VRP).when(detailedConsentResourceMock).getConsentType(); - doReturn(ConsentValidateTestConstants.VRP_INITIATION).when(detailedConsentResourceMock).getReceipt(); - doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(detailedConsentResourceMock).getCurrentStatus(); - - doReturn(getVRPConsentAttributes()).when(detailedConsentResourceMock).getConsentAttributes(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(detailedConsentResourceMock).getConsentID(); - doReturn(ConsentValidateTestConstants.USER_ID).when(consentValidateDataMock).getUserId(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(consentValidateDataMock).getClientId(); - - doReturn(ConsentValidateTestConstants.VRP_PATH).when(consentValidateDataMock).getRequestPath(); - doReturn(resourceParams).when(consentValidateDataMock).getResourceParams(); - doReturn(headers).when(consentValidateDataMock).getHeaders(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(consentValidateDataMock).getConsentId(); - JSONObject submissionPayload = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE) - .parse(ConsentValidateTestConstants.VRP_SUBMISSION_WITHOUT_REMITTANCE_INFO_MISMATCH); - doReturn(submissionPayload).when(consentValidateDataMock).getPayload(); - - doReturn(ConsentExtensionTestUtils.getConsentAttributes("vrp")) - .when(consentCoreServiceMock).getConsentAttributes(Mockito.anyString()); - doReturn(true).when(consentCoreServiceMock).deleteConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - doReturn(true).when(consentCoreServiceMock).storeConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - PowerMockito.when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceMock); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertFalse(consentValidationResult.isValid()); - Assert.assertEquals(consentValidationResult.getErrorMessage(), - ErrorConstants.REMITTANCE_INFO_MISMATCH); - Assert.assertEquals(consentValidationResult.getErrorCode(), ErrorConstants.RESOURCE_CONSENT_MISMATCH); - Assert.assertEquals(consentValidationResult.getHttpCode(), 400); - } - - @Test(dataProvider = "VRPInvalidInitiationSubmissionPayloadsDataProvider", - dataProviderClass = ConsentExtensionDataProvider.class) - public void testValidateVRPSubmissionForInvalidInitiation(String payload) throws ParseException { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(detailedConsentResourceMock).getClientID(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionConstants.VRP).when(detailedConsentResourceMock).getConsentType(); - doReturn(ConsentValidateTestConstants.VRP_INITIATION).when(detailedConsentResourceMock).getReceipt(); - doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(detailedConsentResourceMock).getCurrentStatus(); - - doReturn(getVRPConsentAttributes()).when(detailedConsentResourceMock).getConsentAttributes(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(detailedConsentResourceMock).getConsentID(); - doReturn(ConsentValidateTestConstants.USER_ID).when(consentValidateDataMock).getUserId(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(consentValidateDataMock).getClientId(); - - doReturn(ConsentValidateTestConstants.VRP_PATH).when(consentValidateDataMock).getRequestPath(); - doReturn(resourceParams).when(consentValidateDataMock).getResourceParams(); - doReturn(headers).when(consentValidateDataMock).getHeaders(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(consentValidateDataMock).getConsentId(); - JSONObject submissionPayload = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE).parse(payload); - doReturn(submissionPayload).when(consentValidateDataMock).getPayload(); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertFalse(consentValidationResult.isValid()); - // Using VRPInvalidInitiationSubmissionPayloadsDataProvider dataProvider three test scenarios are been tested. - // Relevant error messages will be returned respectively. - } - - @Test - public void testValidateVRPSubmissionWithIntegerInstructionIdentification() throws ParseException, - ConsentManagementException { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(detailedConsentResourceMock).getClientID(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionConstants.VRP).when(detailedConsentResourceMock).getConsentType(); - doReturn(ConsentValidateTestConstants.VRP_INITIATION).when(detailedConsentResourceMock).getReceipt(); - doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(detailedConsentResourceMock).getCurrentStatus(); - - doReturn(getVRPConsentAttributes()).when(detailedConsentResourceMock).getConsentAttributes(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(detailedConsentResourceMock).getConsentID(); - doReturn(ConsentValidateTestConstants.USER_ID).when(consentValidateDataMock).getUserId(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(consentValidateDataMock).getClientId(); - - doReturn(ConsentValidateTestConstants.VRP_PATH).when(consentValidateDataMock).getRequestPath(); - doReturn(resourceParams).when(consentValidateDataMock).getResourceParams(); - doReturn(headers).when(consentValidateDataMock).getHeaders(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(consentValidateDataMock).getConsentId(); - JSONObject submissionPayload = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE) - .parse(ConsentValidateTestConstants.VRP_SUBMISSION_WITH_INTEGER_INSTRUCTION_IDENTIFICATION); - doReturn(submissionPayload).when(consentValidateDataMock).getPayload(); - - doReturn(ConsentExtensionTestUtils.getConsentAttributes("vrp")) - .when(consentCoreServiceMock).getConsentAttributes(Mockito.anyString()); - doReturn(true).when(consentCoreServiceMock).deleteConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - doReturn(true).when(consentCoreServiceMock).storeConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - PowerMockito.when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceMock); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertFalse(consentValidationResult.isValid()); - Assert.assertEquals(consentValidationResult.getErrorMessage(), ErrorConstants.INVALID_SUBMISSION_TYPE); - Assert.assertEquals(consentValidationResult.getErrorCode(), ErrorConstants.FIELD_INVALID); - Assert.assertEquals(consentValidationResult.getHttpCode(), 400); - } - - @Test - public void testValidateVRPSubmissionWithIntegerEndToEndIdentification() throws ParseException, - ConsentManagementException { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(detailedConsentResourceMock).getClientID(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionConstants.VRP).when(detailedConsentResourceMock).getConsentType(); - doReturn(ConsentValidateTestConstants.VRP_INITIATION).when(detailedConsentResourceMock).getReceipt(); - doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(detailedConsentResourceMock).getCurrentStatus(); - - doReturn(getVRPConsentAttributes()).when(detailedConsentResourceMock).getConsentAttributes(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(detailedConsentResourceMock).getConsentID(); - doReturn(ConsentValidateTestConstants.USER_ID).when(consentValidateDataMock).getUserId(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(consentValidateDataMock).getClientId(); - - doReturn(ConsentValidateTestConstants.VRP_PATH).when(consentValidateDataMock).getRequestPath(); - doReturn(resourceParams).when(consentValidateDataMock).getResourceParams(); - doReturn(headers).when(consentValidateDataMock).getHeaders(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(consentValidateDataMock).getConsentId(); - JSONObject submissionPayload = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE) - .parse(ConsentValidateTestConstants.VRP_SUBMISSION_WITH_INTEGER_END_TO_IDENTIFICATION); - doReturn(submissionPayload).when(consentValidateDataMock).getPayload(); - - doReturn(ConsentExtensionTestUtils.getConsentAttributes("vrp")) - .when(consentCoreServiceMock).getConsentAttributes(Mockito.anyString()); - doReturn(true).when(consentCoreServiceMock).deleteConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - doReturn(true).when(consentCoreServiceMock).storeConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - PowerMockito.when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceMock); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertFalse(consentValidationResult.isValid()); - Assert.assertEquals(consentValidationResult.getErrorMessage(), - ErrorConstants.INVALID_END_TO_END_IDENTIFICATION_TYPE); - Assert.assertEquals(consentValidationResult.getErrorCode(), ErrorConstants.FIELD_INVALID); - Assert.assertEquals(consentValidationResult.getHttpCode(), 400); - } - - @Test - public void testValidateVRPSubmissionWithoutDebtorAccInSubmission() throws ParseException, - ConsentManagementException { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(detailedConsentResourceMock).getClientID(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionConstants.VRP).when(detailedConsentResourceMock).getConsentType(); - doReturn(ConsentValidateTestConstants.VRP_INITIATION_WITHOUT_DEBTOR_ACC).when(detailedConsentResourceMock) - .getReceipt(); - doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(detailedConsentResourceMock).getCurrentStatus(); - - doReturn(getVRPConsentAttributes()).when(detailedConsentResourceMock).getConsentAttributes(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(detailedConsentResourceMock).getConsentID(); - doReturn(ConsentValidateTestConstants.USER_ID).when(consentValidateDataMock).getUserId(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(consentValidateDataMock).getClientId(); - - doReturn(ConsentValidateTestConstants.VRP_PATH).when(consentValidateDataMock).getRequestPath(); - doReturn(resourceParams).when(consentValidateDataMock).getResourceParams(); - doReturn(headers).when(consentValidateDataMock).getHeaders(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(consentValidateDataMock).getConsentId(); - JSONObject submissionPayload = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE) - .parse(ConsentValidateTestConstants.VRP_SUBMISSION_WITH_DEBTOR_ACC); - doReturn(submissionPayload).when(consentValidateDataMock).getPayload(); - - doReturn(ConsentExtensionTestUtils.getConsentAttributes("vrp")) - .when(consentCoreServiceMock).getConsentAttributes(Mockito.anyString()); - doReturn(true).when(consentCoreServiceMock).deleteConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - doReturn(true).when(consentCoreServiceMock).storeConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - PowerMockito.when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceMock); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertFalse(consentValidationResult.isValid()); - Assert.assertEquals(consentValidationResult.getErrorMessage(), ErrorConstants.DEBTOR_ACC_NOT_FOUND); - Assert.assertEquals(consentValidationResult.getErrorCode(), ErrorConstants.FIELD_MISSING); - Assert.assertEquals(consentValidationResult.getHttpCode(), 400); - } - - @Test - public void testValidateVRPSubmissionWithoutCreditorAccInInitiation() throws ParseException, - ConsentManagementException { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(detailedConsentResourceMock).getClientID(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionConstants.VRP).when(detailedConsentResourceMock).getConsentType(); - doReturn(ConsentValidateTestConstants.VRP_INITIATION_WITHOUT_CREDITOR_ACC).when(detailedConsentResourceMock). - getReceipt(); - doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(detailedConsentResourceMock).getCurrentStatus(); - - doReturn(getVRPConsentAttributes()).when(detailedConsentResourceMock).getConsentAttributes(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(detailedConsentResourceMock).getConsentID(); - doReturn(ConsentValidateTestConstants.USER_ID).when(consentValidateDataMock).getUserId(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(consentValidateDataMock).getClientId(); - - doReturn(ConsentValidateTestConstants.VRP_PATH).when(consentValidateDataMock).getRequestPath(); - doReturn(resourceParams).when(consentValidateDataMock).getResourceParams(); - doReturn(headers).when(consentValidateDataMock).getHeaders(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(consentValidateDataMock).getConsentId(); - JSONObject submissionPayload = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE) - .parse(ConsentValidateTestConstants.VRP_SUBMISSION_WITH_INSTRUCTION_CREDITOR_ACC); - doReturn(submissionPayload).when(consentValidateDataMock).getPayload(); - - doReturn(ConsentExtensionTestUtils.getConsentAttributes("vrp")) - .when(consentCoreServiceMock).getConsentAttributes(Mockito.anyString()); - doReturn(true).when(consentCoreServiceMock).deleteConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - doReturn(true).when(consentCoreServiceMock).storeConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - PowerMockito.when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceMock); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertFalse(consentValidationResult.isValid()); - Assert.assertEquals(consentValidationResult.getErrorMessage(), ErrorConstants.CREDITOR_ACC_NOT_FOUND); - Assert.assertEquals(consentValidationResult.getErrorCode(), ErrorConstants.FIELD_MISSING); - Assert.assertEquals(consentValidationResult.getHttpCode(), 400); - } - - @Test(dataProvider = "VRPInvalidSubmissionPayloadsDataProvider", - dataProviderClass = ConsentExtensionDataProvider.class) - public void testValidateVRPSubmissionForInvalidInstruction(String payload) throws ParseException { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(detailedConsentResourceMock).getClientID(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionConstants.VRP).when(detailedConsentResourceMock).getConsentType(); - doReturn(ConsentValidateTestConstants.VRP_INITIATION).when(detailedConsentResourceMock).getReceipt(); - doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(detailedConsentResourceMock).getCurrentStatus(); - - doReturn(getVRPConsentAttributes()).when(detailedConsentResourceMock).getConsentAttributes(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(detailedConsentResourceMock).getConsentID(); - doReturn(ConsentValidateTestConstants.USER_ID).when(consentValidateDataMock).getUserId(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(consentValidateDataMock).getClientId(); - - doReturn(ConsentValidateTestConstants.VRP_PATH).when(consentValidateDataMock).getRequestPath(); - doReturn(resourceParams).when(consentValidateDataMock).getResourceParams(); - doReturn(headers).when(consentValidateDataMock).getHeaders(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(consentValidateDataMock).getConsentId(); - JSONObject submissionPayload = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE).parse(payload); - doReturn(submissionPayload).when(consentValidateDataMock).getPayload(); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertFalse(consentValidationResult.isValid()); - // Using the VRPInvalidSubmissionPayloadsDataProvider dataProvider five test scenarios are been tested. - // Relevant error messages will be returned respectively. - } - - @Test - public void testValidateVRPSubmissionWithInstructionRemittanceMismatch() throws ParseException, - ConsentManagementException { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(detailedConsentResourceMock).getClientID(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionConstants.VRP).when(detailedConsentResourceMock).getConsentType(); - doReturn(ConsentValidateTestConstants.VRP_INITIATION).when(detailedConsentResourceMock).getReceipt(); - doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(detailedConsentResourceMock).getCurrentStatus(); - - doReturn(getVRPConsentAttributes()).when(detailedConsentResourceMock).getConsentAttributes(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(detailedConsentResourceMock).getConsentID(); - doReturn(ConsentValidateTestConstants.USER_ID).when(consentValidateDataMock).getUserId(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(consentValidateDataMock).getClientId(); - - doReturn(ConsentValidateTestConstants.VRP_PATH).when(consentValidateDataMock).getRequestPath(); - doReturn(resourceParams).when(consentValidateDataMock).getResourceParams(); - doReturn(headers).when(consentValidateDataMock).getHeaders(); - doReturn(ConsentValidateTestConstants.CONSENT_ID).when(consentValidateDataMock).getConsentId(); - JSONObject submissionPayload = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE) - .parse(ConsentValidateTestConstants.VRP_SUBMISSION_WITHOUT_REMITTANCE_INFO_MISMATCH); - doReturn(submissionPayload).when(consentValidateDataMock).getPayload(); - - doReturn(ConsentExtensionTestUtils.getConsentAttributes("vrp")) - .when(consentCoreServiceMock).getConsentAttributes(Mockito.anyString()); - doReturn(true).when(consentCoreServiceMock).deleteConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - doReturn(true).when(consentCoreServiceMock).storeConsentAttributes(Mockito.anyString(), - Mockito.>anyObject()); - - PowerMockito.mockStatic(ConsentServiceUtil.class); - PowerMockito.when(ConsentServiceUtil.getConsentService()).thenReturn(consentCoreServiceMock); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertFalse(consentValidationResult.isValid()); - Assert.assertEquals(consentValidationResult.getErrorMessage(), ErrorConstants.REMITTANCE_INFO_MISMATCH); - Assert.assertEquals(consentValidationResult.getErrorCode(), ErrorConstants.RESOURCE_CONSENT_MISMATCH); - Assert.assertEquals(consentValidationResult.getHttpCode(), 400); - } - - @Test - public void testConsentValidateVRPvWithInvalidConsentId() { - - doReturn(authorizationResources).when(detailedConsentResourceMock).getAuthorizationResources(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(detailedConsentResourceMock).getClientID(); - doReturn(detailedConsentResourceMock).when(consentValidateDataMock).getComprehensiveConsent(); - doReturn(ConsentExtensionConstants.VRP).when(detailedConsentResourceMock).getConsentType(); - doReturn(ConsentExtensionConstants.AUTHORIZED_STATUS).when(detailedConsentResourceMock).getCurrentStatus(); - doReturn(ConsentValidateTestConstants.INVALID_CONSENT_ID).when(detailedConsentResourceMock).getConsentID(); - doReturn(ConsentExtensionTestConstants.VALID_INITIATION_OBJECT).when(detailedConsentResourceMock) - .getReceipt(); - doReturn(resourceParams).when(consentValidateDataMock).getResourceParams(); - doReturn(headers).when(consentValidateDataMock).getHeaders(); - doReturn(ConsentValidateTestConstants.USER_ID).when(consentValidateDataMock).getUserId(); - doReturn(ConsentValidateTestConstants.CLIENT_ID).when(consentValidateDataMock).getClientId(); - - ConsentValidationResult consentValidationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateDataMock, consentValidationResult); - - Assert.assertFalse(consentValidationResult.isValid()); - Assert.assertEquals(consentValidationResult.getErrorMessage(), ErrorConstants.MSG_INVALID_CONSENT_ID);; - Assert.assertEquals(consentValidationResult.getErrorCode(), ErrorConstants.RESOURCE_CONSENT_MISMATCH); - Assert.assertEquals(consentValidationResult.getHttpCode(), 400); - } - -} - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/resources/testng.xml b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/resources/testng.xml deleted file mode 100644 index ddc3905f..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.extensions/src/test/resources/testng.xml +++ /dev/null @@ -1,37 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/pom.xml b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/pom.xml deleted file mode 100644 index bae5d5ee..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/pom.xml +++ /dev/null @@ -1,207 +0,0 @@ - - - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../../pom.xml - - 4.0.0 - - com.wso2.openbanking.accelerator.consent.dao - WSO2 Open Banking - Consent DAO - WSO2 Open Banking - Consent DAO Module - jar - - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - provided - - - org.apache.commons - commons-lang3 - - - commons-dbcp - commons-dbcp - - - commons-logging - commons-logging - - - org.testng - testng - - - com.h2database - h2 - test - - - org.jacoco - org.jacoco.agent - runtime - test - - - org.mockito - mockito-all - test - - - org.powermock - powermock-api-mockito - test - - - org.powermock - powermock-module-testng - test - - - - - - - org.apache.maven.plugins - maven-surefire-plugin - - - src/test/resources/testng.xml - - - target/jacoco.exec - - true - - - - - org.jacoco - jacoco-maven-plugin - ${jacoco.version} - - - - **/*Constants.class - **/*Component.class - **/*Exception.class - - **/ConsentStoreInitializer.class - **/ConsentRetentionDataStoreInitializer.class - **/MssqlConsentCoreDAOImpl.class - **/ConsentMgtMssqlDBQueries.class - **/OracleConsentCoreDAOImpl.class - **/ConsentMgtOracleDBQueries.class - **/ConsentMgtPostgresDBQueries.class - - - - - default-prepare-agent - - prepare-agent - - - - default-prepare-agent-integration - - prepare-agent-integration - - - - default-report - - report - - - - default-report-integration - - report-integration - - - - default-check - - check - - - - - BUNDLE - - - COMPLEXITY - COVEREDRATIO - 0.79 - - - - - - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - false - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-exclude.xml - ${project.basedir}/src/main/resources/findbugs-include.xml - false - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - org.apache.maven.plugins - maven-compiler-plugin - - 1.8 - 1.8 - UTF-8 - - - - - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/ConsentCoreDAO.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/ConsentCoreDAO.java deleted file mode 100644 index bc6b66ed..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/ConsentCoreDAO.java +++ /dev/null @@ -1,506 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao; - - -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataDeletionException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataInsertionException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataRetrievalException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataUpdationException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentAttributes; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentFile; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentHistoryResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentMappingResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentStatusAuditRecord; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; - -import java.sql.Connection; -import java.util.ArrayList; -import java.util.List; -import java.util.Map; - -/** - * This interface access the data storage layer to retrieve, store, delete and update consent management related. - * resources. - */ -public interface ConsentCoreDAO { - - /** - * This method is used to store the consent resource in the database. The request consent resource object must - * contain all data in it without the consent ID. A consent ID will be generated and set to the response object - * if the insertion is successful. - * - * @param connection connection object - * @param consentResource consent resource with all required data - * @return returns the consent resource with the consent ID and created time if insertion is successful - * @throws OBConsentDataInsertionException thrown if a database error occur or an insertion failure - */ - ConsentResource storeConsentResource(Connection connection, ConsentResource consentResource) - throws OBConsentDataInsertionException; - - /** - * This method is used to store the authorization resource in the database. The request authorization resource - * object must contain data all in it without the authorization ID and the updated time. Both of them will be - * generated and set to the response object if the insertion is successful. - * - * @param connection connection object - * @param authorizationResource authorization resource with all required data - * @return returns the authorization resource with the updated time if insertion is successful - * @throws OBConsentDataInsertionException thrown if a database error occur or an insertion failure - */ - AuthorizationResource storeAuthorizationResource(Connection connection, AuthorizationResource authorizationResource) - throws OBConsentDataInsertionException; - - /** - * This method is used to store the consent mapping resource in the database. The request consent mapping object - * must contain all the data in it without the consent mapping ID. It will be generated and set to the response - * object if the insertion is successful. - * - * @param connection connection object - * @param consentMappingResource consent mapping resource with all required data - * @return returns the consent mapping resource if the insertion is successful - * @throws OBConsentDataInsertionException thrown if a database error occur or an insertion failure - */ - ConsentMappingResource storeConsentMappingResource(Connection connection, - ConsentMappingResource consentMappingResource) - throws OBConsentDataInsertionException; - - /** - * This method is used to store the consent status audit record in the database. The request consent status audit - * record object must contain all the data in it without the status audit ID and actionTime. They will be generated - * and set to the response object if the insertion is successful. - * - * @param connection connection object - * @param consentStatusAuditRecord consent status audit record with all required data - * @return returns the consent status audit record if the insertion is successful - * @throws OBConsentDataInsertionException thrown if a database error occur or an insertion failure - */ - ConsentStatusAuditRecord storeConsentStatusAuditRecord(Connection connection, - ConsentStatusAuditRecord consentStatusAuditRecord) - throws OBConsentDataInsertionException; - - /** - * This method is used to store the consent attributes in the database. The request consent attributes object - * must be set with a consent ID and consent attribute map. - * - * @param connection connection object - * @param consentAttributes consent attributes object with consent ID and attributes map - * @return returns true if insertion is successful - * @throws OBConsentDataInsertionException thrown if a database error occur or an insertion failure - */ - boolean storeConsentAttributes(Connection connection, ConsentAttributes consentAttributes) - throws OBConsentDataInsertionException; - - /** - * This method is used to store the consent file in the database. The request consent file object must be set - * with a consent ID and the file that needed to be stored. - * - * @param connection connection object - * @param consentFileResource consent file resource with consent ID and the file content - * @return returns true if insertion is successful - * @throws OBConsentDataInsertionException thrown if a database error occur or an insertion failure - */ - boolean storeConsentFile(Connection connection, ConsentFile consentFileResource) - throws OBConsentDataInsertionException; - - /** - * This method is used to update the status of a consent resource. The request consent resource object must be - * set with a consent ID and the new consent status. - * - * @param connection connection object - * @param consentID consent ID of the consent needed to be updated - * @param consentStatus the new status that should be updated with - * @return returns the consent resource with the consent ID and the new status - * @throws OBConsentDataUpdationException thrown if a database error occur or an update failure - */ - ConsentResource updateConsentStatus(Connection connection, String consentID, String consentStatus) - throws OBConsentDataUpdationException; - - /** - * This method is used to update given consent mapping resources. All the mapping resources of provided mapping - * IDs will be updated with the new mapping status provided. - * - * @param connection connection object - * @param mappingIDs a list of mapping IDs that needed to be updated - * @param mappingStatus the new mapping status that should be updated with - * @return returns true if the update is successful - * @throws OBConsentDataUpdationException thrown if a database error occur or an update failure - */ - boolean updateConsentMappingStatus(Connection connection, ArrayList mappingIDs, - String mappingStatus) - throws OBConsentDataUpdationException; - - /** - * This method is used to update given consent mapping resource permissions. All the mapping resources of provided - * map will be updated with the new mapping permission provided. - * - * @param connection - Connection object - * @param mappingIDPermissionMap - A map of mapping IDs against new permissions - * @return - true if the update is successful - * @throws OBConsentDataUpdationException - Thrown of a database level error occurs - */ - boolean updateConsentMappingPermission(Connection connection, Map mappingIDPermissionMap) - throws OBConsentDataUpdationException; - - /** - * This method is used to update a given authorization object. The status of the authorization resource provided - * will be updated with the new status. - * - * @param connection connection object - * @param authorizationID authorization ID of the resource needed to be updated - * @param newAuthorizationStatus the new authorization status that should be updated with - * @return returns authorization resource with authorizationID, new status and updated time - * @throws OBConsentDataUpdationException thrown if a database error occur or an update failure - */ - AuthorizationResource updateAuthorizationStatus(Connection connection, String authorizationID, - String newAuthorizationStatus) - throws OBConsentDataUpdationException; - - /** - * This method is used for updating the user of a given authorization resource. The user ID of the authorization - * resource provided will be updated with the new user ID. - * - * @param connection connection object - * @param authorizationID authorization ID of the resource needed to be updated - * @param userID the new user ID that should be updated with - * @return returns authorization resource with authorization ID, user ID and updated time - * @throws OBConsentDataUpdationException thrown if a database error occur or an update failure - */ - AuthorizationResource updateAuthorizationUser(Connection connection, String authorizationID, String userID) - throws OBConsentDataUpdationException; - - /** - * This method is used to retrieve the consent file from the database. - * - * @param connection connection object - * @param consentID consent ID of the file needed to be retrieved - * @param fetchFromRetentionTables boolean value to fetch from retention tables (temporary purged data) - * @return returns the requested consent file resource - * @throws OBConsentDataRetrievalException thrown if a database error occur or an retrieval failure - */ - ConsentFile getConsentFile(Connection connection, String consentID, boolean fetchFromRetentionTables) - throws OBConsentDataRetrievalException; - - /** - * This method is used to retrieve the consent attributes from the database for given attribute keys. - * - * @param connection connection object - * @param consentID consent ID - * @param consentAttributeKeys the keys of the consent attributes that need to be retrieved - * @return returns the consent attributes that matches the provided consentID and consent attribute keys - * @throws OBConsentDataRetrievalException thrown if a database error occurs - */ - ConsentAttributes getConsentAttributes(Connection connection, String consentID, - ArrayList consentAttributeKeys) - throws OBConsentDataRetrievalException; - - /** - * This method is used to retrieve all the consent attributes from the database for the given consent ID. - * - * @param connection connection object - * @param consentID consent ID - * @return returns the consent attributes that matches the provided consentID and consent attribute keys - * @throws OBConsentDataRetrievalException thrown if a database error occurs - */ - ConsentAttributes getConsentAttributes(Connection connection, String consentID) - throws OBConsentDataRetrievalException; - - /** - * This method is used to retrieve consent attribute using the attribute name. - * - * @param connection connection object - * @param attributeName attribute name - * @return a map with the conesnt ID and the related attribute value - * @throws OBConsentDataRetrievalException thrown if a database error occurs - */ - Map getConsentAttributesByName(Connection connection, String attributeName) - throws OBConsentDataRetrievalException; - - /** - * This method is used to retrieve consent id using the attribute name and value. - * - * @param connection connection object - * @param attributeName attribute name - * @param attributeValue attribute value - * @return Consent ID related to the given attribute key and value - * @throws OBConsentDataRetrievalException `thrown if a database error occurs - */ - ArrayList getConsentIdByConsentAttributeNameAndValue(Connection connection, String attributeName, - String attributeValue) - throws OBConsentDataRetrievalException; - - /** - * This method is used to retrieve a consent resource for the provided consent ID (without associated consent - * attributes). - * - * @param connection connection object - * @param consentID consent ID - * @return returns the consent resource related to the provided consent ID with additional consent attributes or - * not - * @throws OBConsentDataRetrievalException thrown if a database error occurs - */ - ConsentResource getConsentResource(Connection connection, String consentID) - throws OBConsentDataRetrievalException; - - /** - * This method is used to retrieve a detailed consent resource for the provided consent ID (includes - * authorization resources, account mapping resources and consent attributes). - * - * @param connection connection object - * @param consentID consent ID - * @param fetchFromRetentionTables boolean value to fetch from retention tables (temporary purged data) - * @return returns a detailed consent resource related to the provided consent ID - * @throws OBConsentDataRetrievalException thrown if a database error occurs - */ - DetailedConsentResource getDetailedConsentResource(Connection connection, String consentID, - boolean fetchFromRetentionTables) - throws OBConsentDataRetrievalException; - - /** - * This method is used to retrieve a consent resource for the provided consent ID with associated attributes. - * - * @param connection connection object - * @param consentID consent ID - * @return returns the consent resource related to the provided consent ID with additional consent attributes or - * not - * @throws OBConsentDataRetrievalException thrown if a database error occurs - */ - ConsentResource getConsentResourceWithAttributes(Connection connection, String consentID) - throws OBConsentDataRetrievalException; - - /** - * This method is used to retrieve an authorization resource for the provided authorization ID. - * - * @param connection connection object - * @param authorizationID authorization ID - * @return the relevant authorization resource - * @throws OBConsentDataRetrievalException thrown if a database error occurs - */ - AuthorizationResource getAuthorizationResource(Connection connection, String authorizationID) - throws OBConsentDataRetrievalException; - - /** - * This method is used to retrieve consent status audit records. It queries the consent audit records by the - * parameter that is provided. All parameters are optional. If no parameters are provided, all the records will - * be queried. - * - * @param connection connection object - * @param consentID consent ID - * @param currentStatus current status of the consent - * @param actionBy the user who performed the status update - * @param fromTime lower bound of the time of the needed records - * @param toTime upper bound of the time of the needed records - * @param statusAuditID status audit ID - * @param fetchFromRetentionTables boolean value to fetch from retention tables (temporary purged data) - * @return a list of retrieved audit records that matches the provided parameters - * @throws OBConsentDataRetrievalException thrown if a database error occurs - */ - ArrayList getConsentStatusAuditRecords(Connection connection, String consentID, - String currentStatus, String actionBy, - Long fromTime, Long toTime, - String statusAuditID, - boolean fetchFromRetentionTables) - throws OBConsentDataRetrievalException; - - /** - * This method is used to retrieve consent mapping resources for a given authorization ID. - * - * @param connection connection object - * @param authorizationID authorization ID - * @return a list of all consent mapping resources for the given authorization ID - * @throws OBConsentDataRetrievalException thrown if a database error occurs - */ - ArrayList getConsentMappingResources(Connection connection, String authorizationID) - throws OBConsentDataRetrievalException; - - /** - * This method is used to retrieve consent mapping resources for a given authorization ID and mapping status. - * - * @param connection connection object - * @param authorizationID authorization ID - * @param mappingStatus mapping status - * @return a list of all consent mapping resources for the given authorization ID - * @throws OBConsentDataRetrievalException thrown if a database error occurs - */ - ArrayList getConsentMappingResources(Connection connection, String authorizationID, - String mappingStatus) - throws OBConsentDataRetrievalException; - - /** - * This method is used to delete a given list of consent attributes. - * - * @param connection connection object - * @param consentID consent ID - * @param consentAttributeKeys a list of attribute keys that should be deleted - * @return true if the deletion is successful - * @throws OBConsentDataDeletionException thrown if a database error occurs - */ - boolean deleteConsentAttributes(Connection connection, String consentID, ArrayList consentAttributeKeys) - throws OBConsentDataDeletionException; - - /** - * This method is used to search detailed consents for the given lists of parameters. The search will be - * performed according to the provided input. Any list can contain any number of elements. The conjunctive result - * will be returned. If all lists are passed as null, all the consents related to other search parameters will be - * returned. "fromTime" and "toTime" are also optional. "limit" and "offset" are optional combined. If all - * parameters are null, all the consents will be returned. - * - * @param connection connection object - * @param consentIDs consent IDs optional list - * @param clientIDs client IDs optional list - * @param consentTypes consent types optional list - * @param consentStatuses consent statuses optional list - * @param userIDs user IDs optional list - * @param fromTime from time - * @param toTime to time - * @param limit limit - * @param offset offset - * @return a list of detailed consent resources according to the provided parameters or the list of all consents - * if all parameters are null - * @throws OBConsentDataRetrievalException thrown if any error occur - */ - ArrayList searchConsents(Connection connection, ArrayList consentIDs, - ArrayList clientIDs, ArrayList consentTypes, - ArrayList consentStatuses, ArrayList userIDs, - Long fromTime, Long toTime, Integer limit, Integer offset) - throws OBConsentDataRetrievalException; - - /** - * This method is used to search authorization resources using following optional parameters. If all the input - * parameters are null, all the relevant authorization resources will be returned. - * - * 1. Consent ID - * 2. User ID - * - * @param connection connection object - * @param consentID consent ID (optional) - * @param userID user ID (optional) - * @return a list of authorization resources - * @throws OBConsentDataRetrievalException thrown if an error occurs in the process - */ - ArrayList searchConsentAuthorizations(Connection connection, String consentID, String userID) - throws OBConsentDataRetrievalException; - - /** - * This method is used to update consent receipt. - * - * @param connection connection object - * @param consentID ID of the consent to be amended - * @param consentReceipt new consent receipt - * @throws OBConsentDataUpdationException thrown if an error occur in the process - */ - boolean updateConsentReceipt(Connection connection, String consentID, String consentReceipt) - throws OBConsentDataUpdationException; - - /** - * This method is used to update consent validity time. - * - * @param connection connection object - * @param consentID consent ID - * @param validityTime new validity time - * @return true if update successful - * @throws OBConsentDataUpdationException thrown if any error occurs in the process - */ - boolean updateConsentValidityTime(Connection connection, String consentID, long validityTime) - throws OBConsentDataUpdationException; - - /** - * This method is used to store the changed attribute values of the consent into consent history when an - * amendment happens to the consent. - * - * @param connection connection object - * @param historyID An identifier for consent history uniquely assigned per consent amendment - * @param timestamp The timestamp at which the consent amendment happened - * @param recordID Identifier for each history record (can be ConsentID or MappingID) - * @param consentDataType The consent data type stored in each history record (can be ConsentData, - * ConsentAttributesData or ConsentMappingData) - * @param changedAttributesJsonString The key-value pair json string that represents the changes - * relevant to each history record - * @param amendmentReason A string that indicates the reason that caused the amendment of the consent - * @return true if insertion successful - * @throws OBConsentDataInsertionException thrown if any error occurs in the process - */ - boolean storeConsentAmendmentHistory(Connection connection, String historyID, long timestamp, String recordID, - String consentDataType, String changedAttributesJsonString, String amendmentReason) - throws OBConsentDataInsertionException; - - /** - * This method is used to retrieve consent amendment history for a given consentID provided with its mappingIDs, - * AuthorizationIDs. - * - * @param connection connection object - * @param recordIDsList the list of recordIDs relevant to the consent (includes consentID, MappingIDs, AuthIDs) - * @return a comprehensive map of consent history data - * @throws OBConsentDataRetrievalException thrown if any error occurs in the process - */ - Map retrieveConsentAmendmentHistory(Connection connection, - List recordIDsList) throws OBConsentDataRetrievalException; - - /** - * This method is used to fetch consents which has a expiring time as a consent attribute - * (eligible for expiration). - * @throws OBConsentDataRetrievalException thrown if any error occurs in the process - */ - ArrayList getExpiringConsents(Connection connection, - String statusesEligibleForExpiration) - throws OBConsentDataRetrievalException; - - /** - * This method is used to delete the consent details completely from consent database. - * This include deletion of consent attributes, auth resources, consent mappings, audit records and consent file. - * - * @param connection connection object - * @param consentID consent ID - * @param executeOnRetentionTables boolean value to execute query on retention tables (temporary purged data) - * @return true if the deletion is successful - * @throws OBConsentDataDeletionException thrown if a database error occurs - */ - boolean deleteConsentData(Connection connection, String consentID, boolean executeOnRetentionTables) - throws OBConsentDataDeletionException; - - - /** - * This method is used to retrieve a list of consent_ids in consent table. - * - * @param connection connection object - * @param fetchFromRetentionTable boolean value to fetch from retention tables (temporary purged data) - * @return returns a list of consent_ids in consent table. - * @throws OBConsentDataRetrievalException thrown if a database error occurs - */ - ArrayList getListOfConsentIds(Connection connection, boolean fetchFromRetentionTable) - throws OBConsentDataRetrievalException; - - - /** - * This method is used to retrieve a list of consent status audit records by consent_ids. - * - * @param connection connection object - * @param consentIDs consentIDs - * @param fetchFromRetentionTable boolean value to fetch from retention tables (temporary purged data) - * @return returns a list of consent status audit records. - * @throws OBConsentDataRetrievalException thrown if a database error occurs - */ - ArrayList getConsentStatusAuditRecordsByConsentId(Connection connection, - ArrayList consentIDs, - Integer limit, Integer offset, - boolean fetchFromRetentionTable) - throws OBConsentDataRetrievalException; -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/constants/ConsentMgtDAOConstants.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/constants/ConsentMgtDAOConstants.java deleted file mode 100644 index 171e4910..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/constants/ConsentMgtDAOConstants.java +++ /dev/null @@ -1,139 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.constants; - -/** - * This class contains all the constants needed for the consent management DAO layer. - */ -public class ConsentMgtDAOConstants { - - public static final String CONSENT_ID = "CONSENT_ID"; - public static final String CONSENT_FILE = "CONSENT_FILE"; - public static final String ATT_KEY = "ATT_KEY"; - public static final String ATT_VALUE = "ATT_VALUE"; - public static final String RECEIPT = "RECEIPT"; - public static final String CREATED_TIME = "CREATED_TIME"; - public static final String CONSENT_CREATED_TIME = "CONSENT_CREATED_TIME"; - public static final String CONSENT_UPDATED_TIME = "CONSENT_UPDATED_TIME"; - public static final String CLIENT_ID = "CLIENT_ID"; - public static final String CONSENT_TYPE = "CONSENT_TYPE"; - public static final String CURRENT_STATUS = "CURRENT_STATUS"; - public static final String CONSENT_FREQUENCY = "CONSENT_FREQUENCY"; - public static final String VALIDITY_TIME = "VALIDITY_TIME"; - public static final String RECURRING_INDICATOR = "RECURRING_INDICATOR"; - public static final String AUTH_ID = "AUTH_ID"; - public static final String AUTH_TYPE = "AUTH_TYPE"; - public static final String USER_ID = "USER_ID"; - public static final String AUTH_STATUS = "AUTH_STATUS"; - public static final String UPDATED_TIME = "UPDATED_TIME"; - public static final String AUTH_UPDATED_TIME = "AUTH_UPDATED_TIME"; - public static final String MAPPING_ID = "MAPPING_ID"; - public static final String ACCOUNT_ID = "ACCOUNT_ID"; - public static final String PERMISSION = "PERMISSION"; - public static final String MAPPING_STATUS = "MAPPING_STATUS"; - public static final String ACTION_BY = "ACTION_BY"; - public static final String ACTION_TIME = "ACTION_TIME"; - public static final String STATUS_AUDIT_ID = "STATUS_AUDIT_ID"; - public static final String PREVIOUS_STATUS = "PREVIOUS_STATUS"; - public static final String REASON = "REASON"; - public static final String EFFECTIVE_TIMESTAMP = "EFFECTIVE_TIMESTAMP"; - public static final String CONSENT_IDS = "consentIDs"; - public static final String CLIENT_IDS = "clientIDs"; - public static final String CONSENT_TYPES = "consentTypes"; - public static final String CONSENT_STATUSES = "consentStatuses"; - public static final String USER_IDS = "userIDs"; - public static final String IN = "inOperator"; - public static final String AND = "andOperator"; - public static final String OR = "orOperator"; - public static final String WHERE = "where"; - public static final String PLACEHOLDER = "placeholder"; - public static final String PLAIN_PLACEHOLDER = "plainPlaceholder"; - public static final String EQUALS = "equals"; - public static final String CONSENT_EXPIRY_TIME_ATTRIBUTE = "ExpirationDateTime"; - public static final String SESSION_DATA_KEY = "sessionDataKey"; - public static final String AUTH_MAPPING_ID = "AUTH_MAPPING_ID"; - - public static final String RETENTION_TABLE_NAME_PREFIX = "RET_"; - - public static final String CONSENT_MAPPING_RETRIEVE_ERROR_MSG = "Error occurred while retrieving consent mapping " + - "resources from the database"; - public static final String CONSENT_ATTRIBUTES_RETRIEVE_ERROR_MSG = "Error occurred while retrieving consent " + - "attributes from the database for the given consent ID and attribute keys"; - public static final String NO_RECORDS_FOUND_ERROR_MSG = "No records are found for the given inputs"; - public static final String CONSENT_RESOURCE_STORE_ERROR_MSG = "Error occurred while storing consent resource in " + - "the database"; - public static final String AUTHORIZATION_RESOURCE_STORE_ERROR_MSG = "Error occurred while storing authorization " + - "resource in the database"; - public static final String CONSENT_MAPPING_RESOURCE_STORE_ERROR_MSG = "Error occurred while storing consent " + - "mapping resource in the database"; - public static final String AUDIT_RECORD_STORE_ERROR_MSG = "Error occurred while storing consent status audit " + - "record in the database"; - public static final String CONSENT_ATTRIBUTES_STORE_ERROR_MSG = "Error occurred while storing consent attributes " + - "in the database"; - public static final String CONSENT_FILE_STORE_ERROR_MSG = "Error occurred while storing consent file resource in " + - "the database"; - public static final String CONSENT_STATUS_UPDATE_ERROR_MSG = "Error occurred while updating consent status in the" + - " database"; - public static final String CONSENT_MAPPING_STATUS_UPDATE_ERROR_MSG = "Error occurred while updating consent " + - "mapping status in the database"; - public static final String CONSENT_MAPPING_PERMISSION_UPDATE_ERROR_MSG = "Error occurred while updating consent " + - "mapping permission in the database"; - public static final String CONSENT_AUTHORIZATION_STATUS_UPDATE_ERROR_MSG = "Error occurred while updating " + - "authorization status in the database"; - public static final String CONSENT_AUTHORIZATION_USER_UPDATE_ERROR_MSG = "Error occurred while updating " + - "authorization user in the database"; - public static final String CONSENT_FILE_RETRIEVE_ERROR_MSG = "Error occurred while retrieving consent file " + - "resource from the database"; - public static final String CONSENT_RESOURCE_RETRIEVE_ERROR_MSG = "Error occurred while retrieving consent " + - "resource from the database"; - public static final String DETAILED_CONSENT_RESOURCE_RETRIEVE_ERROR_MSG = "Error occurred while retrieving " + - "detailed consent resource from the database"; - public static final String CONSENT_AUTHORIZATION_RESOURCE_RETRIEVE_ERROR_MSG = "Error occurred while retrieving " + - "consent authorization resource from the database"; - public static final String AUDIT_RECORDS_RETRIEVE_ERROR_MSG = "Error occurred while retrieving consent status " + - "audit records"; - public static final String CONSENT_ATTRIBUTES_DELETE_ERROR_MSG = "Error occurred while deleting consent " + - "attributes in the database"; - public static final String CONSENT_DATA_DELETE_ERROR_MSG = "Error occurred while deleting the consent " + - "data in the database"; - public static final String CONSENT_SEARCH_ERROR_MSG = "Error occurred while searching consents"; - public static final String CONSENT_ID_RETRIEVE_ERROR_MSG = "Error occurred while retrieving consent id from the " + - "database for the given attribute key and attribute value"; - public static final String CONSENT_AMENDMENT_HISTORY_RETRIEVE_ERROR_MSG = "Error occurred while retrieving " + - "consent amendment history records from the database for the given consent ID"; - - // Consent Database Table Identifiers - public static final String TABLE_OB_CONSENT = "OB_CONSENT"; - public static final String TABLE_OB_CONSENT_AUTH_RESOURCE = "OB_CONSENT_AUTH_RESOURCE"; - public static final String TABLE_OB_CONSENT_MAPPING = "OB_CONSENT_MAPPING"; - public static final String TABLE_OB_CONSENT_ATTRIBUTE = "OB_CONSENT_ATTRIBUTE"; - public static final String TABLE_OB_CONSENT_FILE = "OB_CONSENT_FILE"; - - // Categorizations of the consent data according to the consent db tables to be used in CA history processing - public static final String TYPE_CONSENT_BASIC_DATA = "ConsentData"; - public static final String TYPE_CONSENT_AUTH_RESOURCE_DATA = "ConsentAuthResourceData"; - public static final String TYPE_CONSENT_ATTRIBUTES_DATA = "ConsentAttributesData"; - public static final String TYPE_CONSENT_MAPPING_DATA = "ConsentMappingData"; - - public static final String TABLE_ID = "TABLE_ID"; - public static final String RECORD_ID = "RECORD_ID"; - public static final String HISTORY_ID = "HISTORY_ID"; - public static final String CHANGED_VALUES = "CHANGED_VALUES"; - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/exceptions/OBConsentDataDeletionException.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/exceptions/OBConsentDataDeletionException.java deleted file mode 100644 index d4f7a08a..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/exceptions/OBConsentDataDeletionException.java +++ /dev/null @@ -1,34 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; - -/** - * OBConsentDataDeletionException. - */ -public class OBConsentDataDeletionException extends OpenBankingException { - - public OBConsentDataDeletionException(String message) { - super(message); - } - - public OBConsentDataDeletionException(String message, Throwable e) { - super(message, e); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/exceptions/OBConsentDataInsertionException.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/exceptions/OBConsentDataInsertionException.java deleted file mode 100644 index 2ca03daf..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/exceptions/OBConsentDataInsertionException.java +++ /dev/null @@ -1,34 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; - -/** - * OBConsentDataInsertionException. - */ -public class OBConsentDataInsertionException extends OpenBankingException { - - public OBConsentDataInsertionException(String message) { - super(message); - } - - public OBConsentDataInsertionException(String message, Throwable e) { - super(message, e); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/exceptions/OBConsentDataRetrievalException.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/exceptions/OBConsentDataRetrievalException.java deleted file mode 100644 index 5db1399f..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/exceptions/OBConsentDataRetrievalException.java +++ /dev/null @@ -1,34 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; - -/** - * OBConsentDataRetrievalException. - */ -public class OBConsentDataRetrievalException extends OpenBankingException { - - public OBConsentDataRetrievalException(String message) { - super(message); - } - - public OBConsentDataRetrievalException(String message, Throwable e) { - super(message, e); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/exceptions/OBConsentDataUpdationException.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/exceptions/OBConsentDataUpdationException.java deleted file mode 100644 index 80d31e73..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/exceptions/OBConsentDataUpdationException.java +++ /dev/null @@ -1,34 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; - -/** - * OBConsentDataUpdationException. - */ -public class OBConsentDataUpdationException extends OpenBankingException { - - public OBConsentDataUpdationException(String message) { - super(message); - } - - public OBConsentDataUpdationException(String message, Throwable e) { - super(message, e); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/impl/ConsentCoreDAOImpl.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/impl/ConsentCoreDAOImpl.java deleted file mode 100644 index 9d225758..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/impl/ConsentCoreDAOImpl.java +++ /dev/null @@ -1,2151 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.impl; - -import com.wso2.openbanking.accelerator.consent.mgt.dao.ConsentCoreDAO; -import com.wso2.openbanking.accelerator.consent.mgt.dao.constants.ConsentMgtDAOConstants; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataDeletionException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataInsertionException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataRetrievalException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataUpdationException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentAttributes; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentFile; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentHistoryResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentMappingResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentStatusAuditRecord; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.queries.ConsentMgtCommonDBQueries; -import com.wso2.openbanking.accelerator.consent.mgt.dao.utils.ConsentDAOUtils; -import net.minidev.json.JSONValue; -import org.apache.commons.collections.CollectionUtils; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.sql.Types; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.HashSet; -import java.util.Iterator; -import java.util.LinkedHashMap; -import java.util.List; -import java.util.Map; -import java.util.Set; -import java.util.UUID; -import java.util.stream.Collectors; -import java.util.stream.IntStream; - -import static com.wso2.openbanking.accelerator.consent.mgt.dao.constants.ConsentMgtDAOConstants.SESSION_DATA_KEY; - -/** - * This class only implements the data access methods for the consent management accelerator. It implements all the - * methods defined in the ConsentCoreDAO interface and is only responsible for reading and writing data from/to the - * database. The incoming data are pre-validated in the upper service layer. Therefore, no validations are done in - * this layer. - */ -public class ConsentCoreDAOImpl implements ConsentCoreDAO { - - private static Log log = LogFactory.getLog(ConsentCoreDAOImpl.class); - private static final String GROUP_BY_SEPARATOR = "\\|\\|"; - ConsentMgtCommonDBQueries sqlStatements; - //Numbers are assigned to each consent DB table & used as the reference for each table when storing CA history - static final Map TABLES_MAP = new HashMap() { - { - put(ConsentMgtDAOConstants.TABLE_OB_CONSENT, "01"); - put(ConsentMgtDAOConstants.TABLE_OB_CONSENT_AUTH_RESOURCE, "02"); - put(ConsentMgtDAOConstants.TABLE_OB_CONSENT_MAPPING, "03"); - put(ConsentMgtDAOConstants.TABLE_OB_CONSENT_ATTRIBUTE, "04"); - put(ConsentMgtDAOConstants.TABLE_OB_CONSENT_FILE, "05"); - } - }; - static final Map COLUMNS_MAP = new HashMap() { - { - put(ConsentMgtDAOConstants.CONSENT_IDS, "CONSENT_ID"); - put(ConsentMgtDAOConstants.CLIENT_IDS, "CLIENT_ID"); - put(ConsentMgtDAOConstants.CONSENT_TYPES, "CONSENT_TYPE"); - put(ConsentMgtDAOConstants.CONSENT_STATUSES, "CURRENT_STATUS"); - put(ConsentMgtDAOConstants.USER_IDS, "OCAR.USER_ID"); - } - }; - - public ConsentCoreDAOImpl(ConsentMgtCommonDBQueries sqlStatements) { - - this.sqlStatements = sqlStatements; - } - - @Override - public ConsentResource storeConsentResource(Connection connection, ConsentResource consentResource) - throws OBConsentDataInsertionException { - - int result; - String consentID = ""; - if (StringUtils.isEmpty(consentResource.getConsentID())) { - consentID = UUID.randomUUID().toString(); - } else { - consentID = consentResource.getConsentID(); - } - // Unix time in seconds - long createdTime; - if (consentResource.getCreatedTime() == 0) { - createdTime = System.currentTimeMillis() / 1000; - } else { - createdTime = consentResource.getCreatedTime(); - } - long updatedTime; - if (consentResource.getUpdatedTime() == 0) { - updatedTime = System.currentTimeMillis() / 1000; - } else { - updatedTime = consentResource.getUpdatedTime(); - } - String storeConsentPrepStatement = sqlStatements.getStoreConsentPreparedStatement(); - - try (PreparedStatement storeConsentPreparedStmt = connection.prepareStatement(storeConsentPrepStatement)) { - - log.debug("Setting parameters to prepared statement to store consent resource"); - - storeConsentPreparedStmt.setString(1, consentID); - storeConsentPreparedStmt.setString(2, consentResource.getReceipt()); - storeConsentPreparedStmt.setLong(3, createdTime); - storeConsentPreparedStmt.setLong(4, updatedTime); - storeConsentPreparedStmt.setString(5, consentResource.getClientID()); - storeConsentPreparedStmt.setString(6, consentResource.getConsentType()); - storeConsentPreparedStmt.setString(7, consentResource.getCurrentStatus()); - storeConsentPreparedStmt.setLong(8, consentResource.getConsentFrequency()); - storeConsentPreparedStmt.setLong(9, consentResource.getValidityPeriod()); - storeConsentPreparedStmt.setBoolean(10, consentResource.isRecurringIndicator()); - - // with result, we can determine whether the insertion was successful or not - result = storeConsentPreparedStmt.executeUpdate(); - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_RESOURCE_STORE_ERROR_MSG, e); - throw new OBConsentDataInsertionException(ConsentMgtDAOConstants.CONSENT_RESOURCE_STORE_ERROR_MSG, e); - } - - // Confirm that the data are inserted successfully - if (result > 0) { - log.debug("Stored the consent resource successfully"); - consentResource.setConsentID(consentID); - consentResource.setCreatedTime(createdTime); - consentResource.setUpdatedTime(createdTime); - return consentResource; - } else { - throw new OBConsentDataInsertionException("Failed to store consent data properly."); - } - } - - @Override - public AuthorizationResource storeAuthorizationResource(Connection connection, - AuthorizationResource authorizationResource) - throws OBConsentDataInsertionException { - - int result; - if (authorizationResource == null) { - throw new OBConsentDataInsertionException("Failed to store authorization resource due to null value."); - } - String authorizationID = UUID.randomUUID().toString(); - if (!StringUtils.isEmpty(authorizationResource.getAuthorizationID())) { - authorizationID = authorizationResource.getAuthorizationID(); - } - // Unix time in seconds - long updatedTime = System.currentTimeMillis() / 1000; - if (authorizationResource.getUpdatedTime() != 0) { - updatedTime = authorizationResource.getUpdatedTime(); - } - String storeAuthorizationPrepStatement = sqlStatements.getStoreAuthorizationPreparedStatement(); - - try (PreparedStatement storeAuthorizationPreparedStmt = - connection.prepareStatement(storeAuthorizationPrepStatement)) { - - log.debug("Setting parameters to prepared statement to store authorization resource"); - - storeAuthorizationPreparedStmt.setString(1, authorizationID); - storeAuthorizationPreparedStmt.setString(2, authorizationResource.getConsentID()); - storeAuthorizationPreparedStmt.setString(3, authorizationResource.getAuthorizationType()); - storeAuthorizationPreparedStmt.setString(4, authorizationResource.getUserID()); - storeAuthorizationPreparedStmt.setString(5, authorizationResource.getAuthorizationStatus()); - storeAuthorizationPreparedStmt.setLong(6, updatedTime); - - // with result, we can determine whether the insertion was successful or not - result = storeAuthorizationPreparedStmt.executeUpdate(); - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.AUTHORIZATION_RESOURCE_STORE_ERROR_MSG, e); - throw new OBConsentDataInsertionException(ConsentMgtDAOConstants.AUTHORIZATION_RESOURCE_STORE_ERROR_MSG, e); - } - - // Confirm that the data are inserted successfully - if (result > 0) { - log.debug("Stored the authorization resource successfully"); - authorizationResource.setAuthorizationID(authorizationID); - authorizationResource.setUpdatedTime(updatedTime); - return authorizationResource; - } else { - throw new OBConsentDataInsertionException("Failed to store authorization resource data properly."); - } - } - - @Override - public ConsentMappingResource storeConsentMappingResource(Connection connection, - ConsentMappingResource consentMappingResource) - throws OBConsentDataInsertionException { - - int result; - if (consentMappingResource == null) { - throw new OBConsentDataInsertionException("Failed to store consent mapping resource due to null value."); - } - String consentMappingID = UUID.randomUUID().toString(); - if (!StringUtils.isEmpty(consentMappingResource.getMappingID())) { - consentMappingID = consentMappingResource.getMappingID(); - } - String storeConsentMappingPrepStatement = sqlStatements.getStoreConsentMappingPreparedStatement(); - - try (PreparedStatement storeConsentMappingPreparedStmt = - connection.prepareStatement(storeConsentMappingPrepStatement)) { - - log.debug("Setting parameters to prepared statement to store consent mapping resource"); - - storeConsentMappingPreparedStmt.setString(1, consentMappingID); - storeConsentMappingPreparedStmt.setString(2, consentMappingResource.getAuthorizationID()); - storeConsentMappingPreparedStmt.setString(3, consentMappingResource.getAccountID()); - storeConsentMappingPreparedStmt.setString(4, consentMappingResource.getPermission()); - storeConsentMappingPreparedStmt.setString(5, consentMappingResource.getMappingStatus()); - - // with result, we can determine whether the insertion was successful or not - result = storeConsentMappingPreparedStmt.executeUpdate(); - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_MAPPING_RESOURCE_STORE_ERROR_MSG, e); - throw new OBConsentDataInsertionException(ConsentMgtDAOConstants.CONSENT_MAPPING_RESOURCE_STORE_ERROR_MSG, - e); - } - - // Confirm that the data are inserted successfully - if (result > 0) { - log.debug("Stored the consent mapping resource successfully"); - consentMappingResource.setMappingID(consentMappingID); - return consentMappingResource; - } else { - throw new OBConsentDataInsertionException("Failed to store consent mapping resource data properly."); - } - } - - @Override - public ConsentStatusAuditRecord storeConsentStatusAuditRecord(Connection connection, - ConsentStatusAuditRecord consentStatusAuditRecord) - throws OBConsentDataInsertionException { - - int result; - if (consentStatusAuditRecord == null) { - throw new OBConsentDataInsertionException("Failed to store consent audit record due to null value."); - } - String statusAuditID = UUID.randomUUID().toString(); - if (!StringUtils.isEmpty(consentStatusAuditRecord.getStatusAuditID())) { - statusAuditID = consentStatusAuditRecord.getStatusAuditID(); - } - // Unix time in seconds - long actionTime = System.currentTimeMillis() / 1000; - if (consentStatusAuditRecord.getActionTime() != 0) { - actionTime = consentStatusAuditRecord.getActionTime(); - } - String storeConsentStatusAuditRecordPrepStatement = - sqlStatements.getStoreConsentStatusAuditRecordPreparedStatement(); - - try (PreparedStatement storeConsentStatusAuditRecordPreparedStmt = - connection.prepareStatement(storeConsentStatusAuditRecordPrepStatement)) { - - log.debug("Setting parameters to prepared statement to store consent audit record"); - - storeConsentStatusAuditRecordPreparedStmt.setString(1, statusAuditID); - storeConsentStatusAuditRecordPreparedStmt.setString(2, consentStatusAuditRecord - .getConsentID()); - storeConsentStatusAuditRecordPreparedStmt.setString(3, consentStatusAuditRecord - .getCurrentStatus()); - storeConsentStatusAuditRecordPreparedStmt.setLong(4, actionTime); - storeConsentStatusAuditRecordPreparedStmt.setString(5, consentStatusAuditRecord.getReason()); - storeConsentStatusAuditRecordPreparedStmt.setString(6, consentStatusAuditRecord - .getActionBy()); - storeConsentStatusAuditRecordPreparedStmt.setString(7, consentStatusAuditRecord - .getPreviousStatus()); - - // with result, we can determine whether the insertion was successful or not - result = storeConsentStatusAuditRecordPreparedStmt.executeUpdate(); - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.AUDIT_RECORD_STORE_ERROR_MSG, e); - throw new OBConsentDataInsertionException(ConsentMgtDAOConstants.AUDIT_RECORD_STORE_ERROR_MSG, e); - } - - // Confirm that the data are inserted successfully - if (result > 0) { - log.debug("Stored the consent status audit record successfully"); - consentStatusAuditRecord.setStatusAuditID(statusAuditID); - consentStatusAuditRecord.setActionTime(actionTime); - return consentStatusAuditRecord; - } else { - throw new OBConsentDataInsertionException("Failed to store consent status audit record data properly."); - } - } - - @Override - public boolean storeConsentAttributes(Connection connection, ConsentAttributes consentAttributes) - throws OBConsentDataInsertionException { - - int[] result; - String storeConsentAttributesPrepStatement = sqlStatements.getStoreConsentAttributesPreparedStatement(); - Map consentAttributesMap = consentAttributes.getConsentAttributes(); - - try (PreparedStatement storeConsentAttributesPreparedStmt = - connection.prepareStatement(storeConsentAttributesPrepStatement)) { - - for (Map.Entry entry : consentAttributesMap.entrySet()) { - storeConsentAttributesPreparedStmt.setString(1, consentAttributes.getConsentID()); - storeConsentAttributesPreparedStmt.setString(2, entry.getKey()); - storeConsentAttributesPreparedStmt.setString(3, entry.getValue()); - storeConsentAttributesPreparedStmt.addBatch(); - } - - // with result, we can determine whether the updating was successful or not - result = storeConsentAttributesPreparedStmt.executeBatch(); - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_ATTRIBUTES_STORE_ERROR_MSG, e); - throw new OBConsentDataInsertionException(ConsentMgtDAOConstants.CONSENT_ATTRIBUTES_STORE_ERROR_MSG, e); - } - - /* - An empty array or an array with value -3 means the batch execution is failed. - If an array contains value -2, it means the command completed successfully but the number of rows affected - are unknown. Therefore, only checking for the existence of -3. - */ - if (result.length != 0 && IntStream.of(result).noneMatch(value -> value == -3)) { - log.debug("Stored the consent attributes successfully"); - return true; - } else { - throw new OBConsentDataInsertionException("Failed to store consent attribute data properly."); - } - } - - @Override - public boolean storeConsentFile(Connection connection, ConsentFile consentFileResource) - throws OBConsentDataInsertionException { - - int result; - String storeConsentMappingPrepStatement = sqlStatements.getStoreConsentFilePreparedStatement(); - - try (PreparedStatement storeConsentFilePreparedStmt = - connection.prepareStatement(storeConsentMappingPrepStatement)) { - - log.debug("Setting parameters to prepared statement to store consent file resource"); - - storeConsentFilePreparedStmt.setString(1, consentFileResource.getConsentID()); - storeConsentFilePreparedStmt.setString(2, consentFileResource.getConsentFile()); - - // with result, we can determine whether the insertion was successful or not - result = storeConsentFilePreparedStmt.executeUpdate(); - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_FILE_STORE_ERROR_MSG, e); - throw new OBConsentDataInsertionException(ConsentMgtDAOConstants.CONSENT_FILE_STORE_ERROR_MSG, e); - } - - // Confirm that the data are inserted successfully - if (result > 0) { - log.debug("Stored the consent file resource successfully"); - return true; - } else { - throw new OBConsentDataInsertionException("Failed to store consent file resource data properly."); - } - } - - @Override - public ConsentResource updateConsentStatus(Connection connection, String consentID, String newConsentStatus) - throws OBConsentDataUpdationException { - - int result; - long updatedTime = System.currentTimeMillis() / 1000; - ConsentResource consentResource = new ConsentResource(); - String updateConsentStatusPrepStatement = sqlStatements.getUpdateConsentStatusPreparedStatement(); - - try (PreparedStatement updateConsentStatusPreparedStmt = - connection.prepareStatement(updateConsentStatusPrepStatement)) { - - log.debug("Setting parameters to prepared statement to update consent status"); - - updateConsentStatusPreparedStmt.setString(1, newConsentStatus); - updateConsentStatusPreparedStmt.setLong(2, updatedTime); - updateConsentStatusPreparedStmt.setString(3, consentID); - - // with result, we can determine whether the updating was successful or not - result = updateConsentStatusPreparedStmt.executeUpdate(); - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_STATUS_UPDATE_ERROR_MSG, e); - throw new OBConsentDataUpdationException(ConsentMgtDAOConstants.CONSENT_STATUS_UPDATE_ERROR_MSG, e); - } - - // Confirm that the data are updated successfully - if (result > 0) { - log.debug("Updated the consent status successfully"); - consentResource.setConsentID(consentID); - consentResource.setCurrentStatus(newConsentStatus); - return consentResource; - } else { - throw new OBConsentDataUpdationException("Failed to update consent status properly."); - } - } - - @Override - public boolean updateConsentMappingStatus(Connection connection, ArrayList mappingIDs, String mappingStatus) - throws OBConsentDataUpdationException { - - int[] result; - String updateConsentMappingStatusPrepStatement = sqlStatements.getUpdateConsentMappingStatusPreparedStatement(); - - try (PreparedStatement updateConsentMappingStatusPreparedStmt = - connection.prepareStatement(updateConsentMappingStatusPrepStatement)) { - - log.debug("Setting parameters to prepared statement to update consent mapping status"); - - for (String mappingID : mappingIDs) { - updateConsentMappingStatusPreparedStmt.setString(1, mappingStatus); - updateConsentMappingStatusPreparedStmt.setString(2, mappingID); - updateConsentMappingStatusPreparedStmt.addBatch(); - } - - // with result, we can determine whether the updating was successful or not - result = updateConsentMappingStatusPreparedStmt.executeBatch(); - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_MAPPING_STATUS_UPDATE_ERROR_MSG, e); - throw new OBConsentDataUpdationException(ConsentMgtDAOConstants.CONSENT_MAPPING_STATUS_UPDATE_ERROR_MSG, e); - } - - // An empty array or an array with value -3 means the batch execution is failed - if (result.length != 0 && IntStream.of(result).noneMatch(value -> value == -3)) { - log.debug("Updated the consent mapping statuses of matching records successfully"); - return true; - } else { - throw new OBConsentDataUpdationException("Failed to update consent mapping status properly."); - } - } - - @Override - public boolean updateConsentMappingPermission(Connection connection, Map mappingIDPermissionMap) - throws OBConsentDataUpdationException { - - int[] result; - String updateConsentMappingPermissionQuery = - sqlStatements.getUpdateConsentMappingPermissionPreparedStatement(); - - try (PreparedStatement updateConsentMappingPermissionPreparedStmt = - connection.prepareStatement(updateConsentMappingPermissionQuery)) { - - log.debug("Setting parameters to prepared statement to update consent mapping permissions"); - - for (String mappingID : mappingIDPermissionMap.keySet()) { - updateConsentMappingPermissionPreparedStmt.setString(1, mappingIDPermissionMap.get(mappingID)); - updateConsentMappingPermissionPreparedStmt.setString(2, mappingID); - updateConsentMappingPermissionPreparedStmt.addBatch(); - } - - // With result, we can determine whether the updating was successful or not - result = updateConsentMappingPermissionPreparedStmt.executeBatch(); - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_MAPPING_PERMISSION_UPDATE_ERROR_MSG, e); - throw new OBConsentDataUpdationException( - ConsentMgtDAOConstants.CONSENT_MAPPING_PERMISSION_UPDATE_ERROR_MSG, e); - } - - // An empty array or an array with value -3 means the batch execution is failed - if (result.length != 0 && IntStream.of(result).noneMatch(value -> value == -3)) { - log.debug("Updated the consent mapping permissions of matching records successfully"); - return true; - } else { - throw new OBConsentDataUpdationException("Failed to update consent mapping permissions properly."); - } - } - - @Override - public AuthorizationResource updateAuthorizationStatus(Connection connection, String authorizationID, - String newAuthorizationStatus) - throws OBConsentDataUpdationException { - - int result; - - // Unix time in seconds - long updatedTime = System.currentTimeMillis() / 1000; - String updateAuthorizationStatusPrepStatement = sqlStatements.getUpdateAuthorizationStatusPreparedStatement(); - - try (PreparedStatement updateAuthorizationStatusPreparedStmt = - connection.prepareStatement(updateAuthorizationStatusPrepStatement)) { - - log.debug("Setting parameters to prepared statement to update authorization status"); - - updateAuthorizationStatusPreparedStmt.setString(1, newAuthorizationStatus); - updateAuthorizationStatusPreparedStmt.setLong(2, updatedTime); - updateAuthorizationStatusPreparedStmt.setString(3, authorizationID); - - // with result, we can determine whether the updating was successful or not - result = updateAuthorizationStatusPreparedStmt.executeUpdate(); - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_AUTHORIZATION_STATUS_UPDATE_ERROR_MSG, e); - throw new OBConsentDataUpdationException(ConsentMgtDAOConstants - .CONSENT_AUTHORIZATION_STATUS_UPDATE_ERROR_MSG, e); - } - - // Confirm that the data are updated successfully - if (result > 0) { - log.debug("Updated the authorization status successfully"); - AuthorizationResource authorizationResource = new AuthorizationResource(); - - authorizationResource.setAuthorizationStatus(newAuthorizationStatus); - authorizationResource.setAuthorizationID(authorizationID); - authorizationResource.setUpdatedTime(updatedTime); - return authorizationResource; - } else { - throw new OBConsentDataUpdationException("Failed to update consent status properly."); - } - } - - @Override - public AuthorizationResource updateAuthorizationUser(Connection connection, String authorizationID, String userID) - throws OBConsentDataUpdationException { - - int result; - - // Unix time in seconds - long updatedTime = System.currentTimeMillis() / 1000; - String updateAuthorizationUserPrepStatement = sqlStatements.getUpdateAuthorizationUserPreparedStatement(); - - try (PreparedStatement updateAuthorizationUserPreparedStmt = - connection.prepareStatement(updateAuthorizationUserPrepStatement)) { - - log.debug("Setting parameters to prepared statement to update authorization user"); - - updateAuthorizationUserPreparedStmt.setString(1, userID); - updateAuthorizationUserPreparedStmt.setLong(2, updatedTime); - updateAuthorizationUserPreparedStmt.setString(3, authorizationID); - - // with result, we can determine whether the updating was successful or not - result = updateAuthorizationUserPreparedStmt.executeUpdate(); - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_AUTHORIZATION_USER_UPDATE_ERROR_MSG, e); - throw new OBConsentDataUpdationException(ConsentMgtDAOConstants.CONSENT_AUTHORIZATION_USER_UPDATE_ERROR_MSG, - e); - } - - // Confirm that the data are updated successfully - if (result > 0) { - log.debug("Updated the authorization user successfully"); - AuthorizationResource authorizationResource = new AuthorizationResource(); - - authorizationResource.setUserID(userID); - authorizationResource.setAuthorizationID(authorizationID); - authorizationResource.setUpdatedTime(updatedTime); - return authorizationResource; - } else { - throw new OBConsentDataUpdationException("Failed to update authorization user properly."); - } - } - - @Override - public ConsentFile getConsentFile(Connection connection, String consentID, boolean fetchFromRetentionTables) - throws OBConsentDataRetrievalException { - - ConsentFile receivedConsentFileResource = new ConsentFile(); - String getConsentFilePrepStatement = sqlStatements.getGetConsentFileResourcePreparedStatement( - fetchFromRetentionTables); - - try (PreparedStatement getConsentFileResourcePreparedStmt = - connection.prepareStatement(getConsentFilePrepStatement)) { - - log.debug("Setting parameters to prepared statement to retrieve consent file resource"); - - getConsentFileResourcePreparedStmt.setString(1, consentID); - - try (ResultSet resultSet = getConsentFileResourcePreparedStmt.executeQuery()) { - if (resultSet.next()) { - String storedConsentID = resultSet.getString(ConsentMgtDAOConstants.CONSENT_ID); - String consentFile = resultSet.getString(ConsentMgtDAOConstants.CONSENT_FILE); - - receivedConsentFileResource.setConsentID(storedConsentID); - receivedConsentFileResource.setConsentFile(consentFile); - } else { - log.error("No records are found for consent ID :" + consentID); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.NO_RECORDS_FOUND_ERROR_MSG); - } - } catch (SQLException e) { - log.error("Error occurred while reading consent file resource"); - throw new OBConsentDataRetrievalException(String.format("Error occurred while retrieving consent file" + - " resource for consent ID : %s", consentID), e); - } - - if (log.isDebugEnabled()) { - log.debug("Retrieved the consent file resource for consent ID : " + consentID); - } - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_FILE_RETRIEVE_ERROR_MSG, e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.CONSENT_FILE_RETRIEVE_ERROR_MSG, e); - } - return receivedConsentFileResource; - } - - @Override - public ConsentAttributes getConsentAttributes(Connection connection, String consentID, - ArrayList consentAttributeKeys) - throws OBConsentDataRetrievalException { - - Map retrievedConsentAttributesMap = new HashMap<>(); - ConsentAttributes retrievedConsentAttributesResource; - String getConsentAttributesPrepStatement = sqlStatements.getGetConsentAttributesPreparedStatement(); - - try (PreparedStatement getConsentAttributesPreparedStmt = - connection.prepareStatement(getConsentAttributesPrepStatement)) { - - log.debug("Setting parameters to prepared statement to retrieve consent attributes"); - - getConsentAttributesPreparedStmt.setString(1, consentID); - - try (ResultSet resultSet = getConsentAttributesPreparedStmt.executeQuery()) { - if (resultSet.isBeforeFirst()) { - while (resultSet.next()) { - String attributeKey = resultSet.getString(ConsentMgtDAOConstants.ATT_KEY); - String attributeValue = resultSet.getString(ConsentMgtDAOConstants.ATT_VALUE); - - // Filter the needed attributes - if (consentAttributeKeys.contains(attributeKey)) { - retrievedConsentAttributesMap.put(attributeKey, attributeValue); - if (retrievedConsentAttributesMap.size() == consentAttributeKeys.size()) { - break; - } - } - } - retrievedConsentAttributesResource = new ConsentAttributes(); - retrievedConsentAttributesResource.setConsentID(consentID); - retrievedConsentAttributesResource.setConsentAttributes(retrievedConsentAttributesMap); - } else { - log.error("No records are found for consent ID : " + consentID + " and consent attribute keys"); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.NO_RECORDS_FOUND_ERROR_MSG); - } - } catch (SQLException e) { - log.error("Error occurred while reading consent attributes", e); - throw new OBConsentDataRetrievalException(String.format("Error occurred while retrieving consent " + - "attributes for consent ID : %s and provided consent attributes", consentID), e); - } - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_ATTRIBUTES_RETRIEVE_ERROR_MSG, e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.CONSENT_ATTRIBUTES_RETRIEVE_ERROR_MSG, e); - } - return retrievedConsentAttributesResource; - } - - @Override - public ConsentAttributes getConsentAttributes(Connection connection, String consentID) - throws OBConsentDataRetrievalException { - - Map retrievedConsentAttributesMap = new HashMap<>(); - ConsentAttributes retrievedConsentAttributesResource; - String getConsentAttributesPrepStatement = sqlStatements.getGetConsentAttributesPreparedStatement(); - - try (PreparedStatement getConsentAttributesPreparedStmt = - connection.prepareStatement(getConsentAttributesPrepStatement)) { - - log.debug("Setting parameters to prepared statement to retrieve consent attributes"); - - getConsentAttributesPreparedStmt.setString(1, consentID); - - try (ResultSet resultSet = getConsentAttributesPreparedStmt.executeQuery()) { - if (resultSet.isBeforeFirst()) { - while (resultSet.next()) { - retrievedConsentAttributesMap.put(resultSet.getString(ConsentMgtDAOConstants.ATT_KEY), - resultSet.getString(ConsentMgtDAOConstants.ATT_VALUE)); - } - retrievedConsentAttributesResource = new ConsentAttributes(); - retrievedConsentAttributesResource.setConsentID(consentID); - retrievedConsentAttributesResource.setConsentAttributes(retrievedConsentAttributesMap); - } else { - log.error("No records are found for consent ID :" + consentID); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.NO_RECORDS_FOUND_ERROR_MSG); - } - } catch (SQLException e) { - log.error("Error occurred while reading consent attributes", e); - throw new OBConsentDataRetrievalException(String.format("Error occurred while retrieving consent " + - "attributes for consent ID : %s", consentID), e); - } - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_ATTRIBUTES_RETRIEVE_ERROR_MSG, e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.CONSENT_ATTRIBUTES_RETRIEVE_ERROR_MSG, e); - } - return retrievedConsentAttributesResource; - } - - @Override - public Map getConsentAttributesByName(Connection connection, String attributeName) - throws OBConsentDataRetrievalException { - - Map retrievedConsentAttributesMap = new HashMap<>(); - String getConsentAttributesByNamePrepStatement = sqlStatements.getGetConsentAttributesByNamePreparedStatement(); - - try (PreparedStatement getConsentAttributesByNamePreparedStmt = - connection.prepareStatement(getConsentAttributesByNamePrepStatement)) { - - if (log.isDebugEnabled()) { - log.debug("Setting parameters to prepared statement to retrieve consent attributes for the provided " + - "key: " + attributeName); - } - getConsentAttributesByNamePreparedStmt.setString(1, attributeName); - - try (ResultSet resultSet = getConsentAttributesByNamePreparedStmt.executeQuery()) { - if (resultSet.isBeforeFirst()) { - while (resultSet.next()) { - retrievedConsentAttributesMap.put(resultSet.getString(ConsentMgtDAOConstants.CONSENT_ID), - resultSet.getString(ConsentMgtDAOConstants.ATT_VALUE)); - } - } - } catch (SQLException e) { - log.error("Error occurred while reading consent attributes for the given key: " - + attributeName, e); - throw new OBConsentDataRetrievalException(String.format("Error occurred while retrieving consent " + - "attributes for attribute key: %s", attributeName), e); - } - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_ATTRIBUTES_RETRIEVE_ERROR_MSG, e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.CONSENT_ATTRIBUTES_RETRIEVE_ERROR_MSG, e); - } - return retrievedConsentAttributesMap; - } - - @Override - public ArrayList getConsentIdByConsentAttributeNameAndValue(Connection connection, String attributeName, - String attributeValue) - throws OBConsentDataRetrievalException { - - ArrayList retrievedConsentIdList = new ArrayList<>(); - String getConsentIdByConsentAttributeNameAndValuePrepStatement = sqlStatements - .getConsentIdByConsentAttributeNameAndValuePreparedStatement(); - - try (PreparedStatement getConsentAttributesByNamePreparedStmt = - connection.prepareStatement(getConsentIdByConsentAttributeNameAndValuePrepStatement)) { - - if (log.isDebugEnabled()) { - log.debug("Setting parameters to prepared statement to retrieve consent id for the provided " + - "key: " + attributeName + " and value: " + attributeValue); - } - getConsentAttributesByNamePreparedStmt.setString(1, attributeName); - getConsentAttributesByNamePreparedStmt.setString(2, attributeValue); - - try (ResultSet resultSet = getConsentAttributesByNamePreparedStmt.executeQuery()) { - if (resultSet.isBeforeFirst()) { - while (resultSet.next()) { - retrievedConsentIdList.add(resultSet.getString(ConsentMgtDAOConstants.CONSENT_ID)); - } - } else { - log.error("No records are found for the provided attribute key :" + attributeName + - " and value: " + attributeValue); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.NO_RECORDS_FOUND_ERROR_MSG); - } - } catch (SQLException e) { - log.error("Error occurred while reading consent attributes for the given key: " + attributeName + - " and value: " + attributeValue); - throw new OBConsentDataRetrievalException(String.format("Error occurred while retrieving consent " + - "attributes for attribute key: %s and value: %s", attributeName, attributeValue), e); - } - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_ID_RETRIEVE_ERROR_MSG); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.CONSENT_ID_RETRIEVE_ERROR_MSG, e); - } - return retrievedConsentIdList; - } - - @Override - public ConsentResource getConsentResource(Connection connection, String consentID) - throws OBConsentDataRetrievalException { - - ConsentResource retrievedConsentResource = new ConsentResource(); - - String getConsentResourcePrepStatement = sqlStatements.getGetConsentPreparedStatement(); - - try (PreparedStatement getConsentResourcePreparedStmt = - connection.prepareStatement(getConsentResourcePrepStatement)) { - - log.debug("Setting parameters to prepared statement to retrieve consent resource"); - - getConsentResourcePreparedStmt.setString(1, consentID); - - try (ResultSet resultSet = getConsentResourcePreparedStmt.executeQuery()) { - if (resultSet.next()) { - setDataToConsentResource(resultSet, retrievedConsentResource); - } else { - log.error("No records are found for consent ID :" + consentID); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.NO_RECORDS_FOUND_ERROR_MSG); - } - } catch (SQLException e) { - log.error("Error occurred while reading consent resource", e); - throw new OBConsentDataRetrievalException(String.format("Error occurred while retrieving consent " + - "resource for consent ID : %s", consentID), e); - } - - if (log.isDebugEnabled()) { - log.debug("Retrieved the consent resource from OB_CONSENT table for consent ID : " + consentID); - } - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_RESOURCE_RETRIEVE_ERROR_MSG, e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.CONSENT_RESOURCE_RETRIEVE_ERROR_MSG, e); - } - return retrievedConsentResource; - } - - @Override - public DetailedConsentResource getDetailedConsentResource(Connection connection, String consentID, - boolean fetchFromRetentionTables) - throws OBConsentDataRetrievalException { - - DetailedConsentResource retrievedDetailedConsentResource = new DetailedConsentResource(); - - String getDetailedConsentResourcePrepStatement = sqlStatements.getGetDetailedConsentPreparedStatement( - fetchFromRetentionTables); - - try (PreparedStatement getDetailedConsentResourcePreparedStmt = connection - .prepareStatement(getDetailedConsentResourcePrepStatement)) { - - log.debug("Setting parameters to prepared statement to retrieve detailed consent resource"); - - getDetailedConsentResourcePreparedStmt.setString(1, consentID); - - try (ResultSet resultSet = getDetailedConsentResourcePreparedStmt.executeQuery()) { - if (resultSet.isBeforeFirst()) { - setDataToDetailedConsentResource(resultSet, retrievedDetailedConsentResource); - } else { - log.error("No records are found for consent ID :" + consentID); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.NO_RECORDS_FOUND_ERROR_MSG); - } - } catch (SQLException e) { - log.error("Error occurred while reading detailed consent resource", e); - throw new OBConsentDataRetrievalException(String.format("Error occurred while retrieving " + - "detailed consent resource for consent ID : %s", consentID), e); - } - - if (log.isDebugEnabled()) { - log.debug("Retrieved the detailed consent resource for consent ID : " + - consentID); - } - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.DETAILED_CONSENT_RESOURCE_RETRIEVE_ERROR_MSG, e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants - .DETAILED_CONSENT_RESOURCE_RETRIEVE_ERROR_MSG, e); - } - return retrievedDetailedConsentResource; - } - - @Override - public ConsentResource getConsentResourceWithAttributes(Connection connection, String consentID) - throws OBConsentDataRetrievalException { - - Map retrievedConsentAttributeMap = new HashMap<>(); - ConsentResource retrievedConsentResource = new ConsentResource(); - - String getConsentResourcePrepStatement = sqlStatements.getGetConsentWithConsentAttributesPreparedStatement(); - - try (PreparedStatement getConsentResourcePreparedStmt = - connection.prepareStatement(getConsentResourcePrepStatement, ResultSet.TYPE_SCROLL_INSENSITIVE, - ResultSet.CONCUR_READ_ONLY)) { - - log.debug("Setting parameters to prepared statement to retrieve consent resource with consent attributes"); - - getConsentResourcePreparedStmt.setString(1, consentID); - - try (ResultSet resultSet = getConsentResourcePreparedStmt.executeQuery()) { - if (resultSet.next()) { - setDataToConsentResource(resultSet, retrievedConsentResource); - - // Point the cursor to the beginning of the result set to read attributes - resultSet.beforeFirst(); - while (resultSet.next()) { - retrievedConsentAttributeMap.put(resultSet.getString(ConsentMgtDAOConstants.ATT_KEY), - resultSet.getString(ConsentMgtDAOConstants.ATT_VALUE)); - } - retrievedConsentResource.setConsentAttributes(retrievedConsentAttributeMap); - } else { - log.error("No records are found for consent ID :" + consentID); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.NO_RECORDS_FOUND_ERROR_MSG); - } - } catch (SQLException e) { - log.error("Error occurred while reading consent resource with consent attributes", e); - throw new OBConsentDataRetrievalException(String.format("Error occurred while retrieving consent " + - "resource with consent attributes for consent ID : %s", consentID), e); - } - - if (log.isDebugEnabled()) { - log.debug("Retrieved the consent resource with consent attributes for consent ID : " + consentID); - } - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_ATTRIBUTES_RETRIEVE_ERROR_MSG, e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.CONSENT_ATTRIBUTES_RETRIEVE_ERROR_MSG, e); - } - return retrievedConsentResource; - } - - @Override - public AuthorizationResource getAuthorizationResource(Connection connection, String authorizationID) - throws OBConsentDataRetrievalException { - - AuthorizationResource retrievedAuthorizationResource = new AuthorizationResource(); - String getAuthorizationResourcePrepStatement = sqlStatements.getGetAuthorizationResourcePreparedStatement(); - - try (PreparedStatement getConsentResourcePreparedStmt = - connection.prepareStatement(getAuthorizationResourcePrepStatement)) { - - log.debug("Setting parameters to prepared statement to retrieve consent authorization resource"); - - getConsentResourcePreparedStmt.setString(1, authorizationID); - - try (ResultSet resultSet = getConsentResourcePreparedStmt.executeQuery()) { - if (resultSet.next()) { - setAuthorizationData(retrievedAuthorizationResource, resultSet); - } else { - log.error("No records are found for authorization ID :" + authorizationID); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.NO_RECORDS_FOUND_ERROR_MSG); - } - } catch (SQLException e) { - log.error("Error occurred while reading consent authorization resource", e); - throw new OBConsentDataRetrievalException(String.format("Error occurred while retrieving consent " + - "authorization resource for authorization ID : %s", authorizationID), e); - } - - if (log.isDebugEnabled()) { - log.debug("Retrieved the consent authorization resource for authorization ID : " + authorizationID); - } - - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_AUTHORIZATION_RESOURCE_RETRIEVE_ERROR_MSG, e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants. - CONSENT_AUTHORIZATION_RESOURCE_RETRIEVE_ERROR_MSG, e); - } - return retrievedAuthorizationResource; - } - - @Override - public ArrayList getConsentStatusAuditRecords(Connection connection, String consentID, - String currentStatus, String actionBy, - Long fromTime, Long toTime, - String statusAuditID, - boolean fetchFromRetentionTables) - throws OBConsentDataRetrievalException { - - ArrayList retrievedAuditRecords = new ArrayList<>(); - String getConsentStatusAuditRecordsPrepStatement = - sqlStatements.getGetConsentStatusAuditRecordsPreparedStatement(fetchFromRetentionTables); - - try (PreparedStatement getConsentStatusAuditRecordPreparedStmt = - connection.prepareStatement(getConsentStatusAuditRecordsPrepStatement)) { - - if (log.isDebugEnabled()) { - log.debug("Setting parameters to prepared statement to retrieve consent status audit records"); - } - - // consentID - if (StringUtils.trimToNull(consentID) != null) { - getConsentStatusAuditRecordPreparedStmt.setString(1, consentID); - } else { - getConsentStatusAuditRecordPreparedStmt.setNull(1, Types.VARCHAR); - } - - // currentStatus - if (StringUtils.trimToNull(currentStatus) != null) { - getConsentStatusAuditRecordPreparedStmt.setString(2, currentStatus); - } else { - getConsentStatusAuditRecordPreparedStmt.setNull(2, Types.VARCHAR); - } - - // actionBy - if (StringUtils.trimToNull(actionBy) != null) { - getConsentStatusAuditRecordPreparedStmt.setString(3, actionBy); - } else { - getConsentStatusAuditRecordPreparedStmt.setNull(3, Types.VARCHAR); - } - - // statusAuditID - if (StringUtils.trimToNull(statusAuditID) != null) { - getConsentStatusAuditRecordPreparedStmt.setString(4, statusAuditID); - } else { - getConsentStatusAuditRecordPreparedStmt.setNull(4, Types.VARCHAR); - } - - // fromTime - if (fromTime != null) { - getConsentStatusAuditRecordPreparedStmt.setLong(5, fromTime); - } else { - getConsentStatusAuditRecordPreparedStmt.setNull(5, Types.BIGINT); - } - - // toTime - if (toTime != null) { - getConsentStatusAuditRecordPreparedStmt.setLong(6, toTime); - } else { - getConsentStatusAuditRecordPreparedStmt.setNull(6, Types.BIGINT); - } - - try (ResultSet resultSet = getConsentStatusAuditRecordPreparedStmt.executeQuery()) { - if (resultSet.isBeforeFirst()) { - while (resultSet.next()) { - ConsentStatusAuditRecord consentStatusAuditRecord = new ConsentStatusAuditRecord(); - consentStatusAuditRecord - .setStatusAuditID(resultSet.getString(ConsentMgtDAOConstants.STATUS_AUDIT_ID)); - consentStatusAuditRecord.setConsentID(resultSet.getString(ConsentMgtDAOConstants.CONSENT_ID)); - consentStatusAuditRecord - .setCurrentStatus(resultSet.getString(ConsentMgtDAOConstants.CURRENT_STATUS)); - consentStatusAuditRecord.setActionBy(resultSet.getString(ConsentMgtDAOConstants.ACTION_BY)); - consentStatusAuditRecord.setActionTime(resultSet.getLong(ConsentMgtDAOConstants.ACTION_TIME)); - consentStatusAuditRecord.setReason(resultSet.getString(ConsentMgtDAOConstants.REASON)); - consentStatusAuditRecord - .setPreviousStatus(resultSet.getString(ConsentMgtDAOConstants.PREVIOUS_STATUS)); - retrievedAuditRecords.add(consentStatusAuditRecord); - } - } else { - log.error("No records are found for the provided inputs"); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.NO_RECORDS_FOUND_ERROR_MSG); - } - } catch (SQLException e) { - log.error("Error occurred while reading consent status audit records", e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.AUDIT_RECORDS_RETRIEVE_ERROR_MSG, e); - } - - log.debug("Retrieved the consent status audit records successfully"); - - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.AUDIT_RECORDS_RETRIEVE_ERROR_MSG, e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.AUDIT_RECORDS_RETRIEVE_ERROR_MSG, e); - } - return retrievedAuditRecords; - } - - @Override - public ArrayList getConsentMappingResources(Connection connection, String authorizationID) - throws OBConsentDataRetrievalException { - - ArrayList retrievedConsentMappingResources = new ArrayList<>(); - String getMappingResourcePrepStatement = sqlStatements.getGetConsentMappingResourcesPreparedStatement(); - - try (PreparedStatement getConsentMappingResourcePreparedStmt = - connection.prepareStatement(getMappingResourcePrepStatement)) { - - log.debug("Setting parameters to prepared statement to retrieve consent mapping resources"); - - getConsentMappingResourcePreparedStmt.setString(1, authorizationID); - - try (ResultSet resultSet = getConsentMappingResourcePreparedStmt.executeQuery()) { - if (resultSet.isBeforeFirst()) { - while (resultSet.next()) { - retrievedConsentMappingResources.add(getConsentMappingResourceWithData(resultSet)); - } - } else { - log.error("No records are found for authorization ID : " + authorizationID); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.NO_RECORDS_FOUND_ERROR_MSG); - } - } catch (SQLException e) { - log.error("Error occurred while reading consent mapping resources", e); - throw new OBConsentDataRetrievalException(String.format("Error occurred while retrieving consent " + - "mapping resources for authorization ID : %s", authorizationID), e); - } - - if (log.isDebugEnabled()) { - log.debug("Retrieved the consent mapping resources for authorization ID : " + authorizationID); - } - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_MAPPING_RETRIEVE_ERROR_MSG, e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.CONSENT_MAPPING_RETRIEVE_ERROR_MSG, e); - } - return retrievedConsentMappingResources; - } - - @Override - public ArrayList getConsentMappingResources(Connection connection, String authorizationID, - String mappingStatus) - throws OBConsentDataRetrievalException { - - ArrayList retrievedConsentMappingResources = new ArrayList<>(); - String getMappingResourcePrepStatement - = sqlStatements.getGetConsentMappingResourcesForStatusPreparedStatement(); - - try (PreparedStatement getConsentMappingResourcePreparedStmt = - connection.prepareStatement(getMappingResourcePrepStatement)) { - - log.debug("Setting parameters to prepared statement to retrieve consent mapping resources"); - - getConsentMappingResourcePreparedStmt.setString(1, authorizationID); - getConsentMappingResourcePreparedStmt.setString(2, mappingStatus); - - try (ResultSet resultSet = getConsentMappingResourcePreparedStmt.executeQuery()) { - if (resultSet.isBeforeFirst()) { - while (resultSet.next()) { - retrievedConsentMappingResources.add(getConsentMappingResourceWithData(resultSet)); - } - } else { - log.error("No records are found for authorization ID : " + authorizationID + " and mapping " + - "status " + mappingStatus); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.NO_RECORDS_FOUND_ERROR_MSG); - } - } catch (SQLException e) { - log.error("Error occurred while reading consent mapping resources", e); - throw new OBConsentDataRetrievalException(String.format("Error occurred while retrieving consent " + - "mapping resources for authorization ID : %s and mapping status : %s", authorizationID, - mappingStatus), e); - } - - if (log.isDebugEnabled()) { - log.debug("Retrieved the consent mapping resources for authorization ID : " + authorizationID + " and" + - " mapping status : " + mappingStatus); - } - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_MAPPING_RETRIEVE_ERROR_MSG, e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.CONSENT_MAPPING_RETRIEVE_ERROR_MSG, e); - } - return retrievedConsentMappingResources; - } - - @Override - public boolean deleteConsentAttributes(Connection connection, String consentID, - ArrayList consentAttributeKeys) - throws OBConsentDataDeletionException { - - int[] result; - String deleteConsentAttributePrepStatement = sqlStatements.getDeleteConsentAttributePreparedStatement(); - - try (PreparedStatement deleteConsentAttributesPreparedStmt = - connection.prepareStatement(deleteConsentAttributePrepStatement)) { - - if (log.isDebugEnabled()) { - log.debug("Setting parameters to prepared statement to delete the provided consent attributes"); - } - - for (String key : consentAttributeKeys) { - deleteConsentAttributesPreparedStmt.setString(1, consentID); - deleteConsentAttributesPreparedStmt.setString(2, key); - deleteConsentAttributesPreparedStmt.addBatch(); - } - - result = deleteConsentAttributesPreparedStmt.executeBatch(); - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_ATTRIBUTES_DELETE_ERROR_MSG, e); - throw new OBConsentDataDeletionException(ConsentMgtDAOConstants.CONSENT_ATTRIBUTES_DELETE_ERROR_MSG, e); - } - - if (result.length != 0 && IntStream.of(result).noneMatch(value -> value == -3)) { - if (log.isDebugEnabled()) { - log.debug("Deleted the consent attribute of key " + consentAttributeKeys); - } - return true; - } else { - throw new OBConsentDataDeletionException("Failed to delete consent attribute properly."); - } - } - - @Override - public ArrayList searchConsents(Connection connection, ArrayList consentIDs, - ArrayList clientIDs, - ArrayList consentTypes, - ArrayList consentStatuses, - ArrayList userIDs, Long fromTime, Long toTime, - Integer limit, Integer offset) - throws OBConsentDataRetrievalException { - - boolean shouldLimit = true; - boolean shouldOffset = true; - int parameterIndex = 0; - Map applicableConditionsMap = new HashMap<>(); - - validateAndSetSearchConditions(applicableConditionsMap, consentIDs, clientIDs, consentTypes, consentStatuses); - - // Don't limit if either of limit or offset is null - if (limit == null) { - shouldLimit = false; - } - if (offset == null) { - shouldOffset = false; - } - - // logic to set the prepared statement - log.debug("Constructing the prepared statement"); - String constructedConditions = - ConsentDAOUtils.constructConsentSearchPreparedStatement(applicableConditionsMap); - - String userIDFilterCondition = ""; - Map userIdMap = new HashMap<>(); - if (CollectionUtils.isNotEmpty(userIDs)) { - userIdMap.put(COLUMNS_MAP.get(ConsentMgtDAOConstants.USER_IDS), userIDs); - userIDFilterCondition = ConsentDAOUtils.constructUserIdListFilterCondition(userIdMap); - } - - String searchConsentsPreparedStatement = - sqlStatements.getSearchConsentsPreparedStatement(constructedConditions, shouldLimit, shouldOffset, - userIDFilterCondition); - - try (PreparedStatement searchConsentsPreparedStmt = - connection.prepareStatement(searchConsentsPreparedStatement, ResultSet.TYPE_SCROLL_INSENSITIVE, - ResultSet.CONCUR_UPDATABLE)) { - - /* Since we don't know the order of the set condition clauses, have to determine the order of them to set - the actual values to the prepared statement */ - Map orderedParamsMap = ConsentDAOUtils - .determineOrderOfParamsToSet(constructedConditions, applicableConditionsMap, COLUMNS_MAP); - - log.debug("Setting parameters to prepared statement to search consents"); - - parameterIndex = setDynamicConsentSearchParameters(searchConsentsPreparedStmt, orderedParamsMap, - ++parameterIndex); - parameterIndex = parameterIndex - 1; - - //determine order of user Ids to set - if (CollectionUtils.isNotEmpty(userIDs)) { - Map orderedUserIdsMap = ConsentDAOUtils - .determineOrderOfParamsToSet(userIDFilterCondition, userIdMap, COLUMNS_MAP); - parameterIndex = setDynamicConsentSearchParameters(searchConsentsPreparedStmt, orderedUserIdsMap, - ++parameterIndex); - parameterIndex = parameterIndex - 1; - } - - if (fromTime != null) { - searchConsentsPreparedStmt.setLong(++parameterIndex, fromTime); - } else { - searchConsentsPreparedStmt.setNull(++parameterIndex, Types.BIGINT); - } - - if (toTime != null) { - searchConsentsPreparedStmt.setLong(++parameterIndex, toTime); - } else { - searchConsentsPreparedStmt.setNull(++parameterIndex, Types.BIGINT); - } - - if (limit != null && offset != null) { - searchConsentsPreparedStmt.setInt(++parameterIndex, limit); - searchConsentsPreparedStmt.setInt(++parameterIndex, offset); - } else if (limit != null) { - searchConsentsPreparedStmt.setInt(++parameterIndex, limit); - } - ArrayList detailedConsentResources = new ArrayList<>(); - - try (ResultSet resultSet = searchConsentsPreparedStmt.executeQuery()) { - if (resultSet.isBeforeFirst()) { - int resultSetSize = getResultSetSize(resultSet); - detailedConsentResources = constructDetailedConsentsSearchResult(resultSet, resultSetSize); - } - return detailedConsentResources; - } catch (SQLException e) { - log.error("Error occurred while searching detailed consent resources", e); - throw new OBConsentDataRetrievalException("Error occurred while searching detailed " + - "consent resources", e); - } - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_SEARCH_ERROR_MSG, e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.CONSENT_SEARCH_ERROR_MSG); - } - } - - @Override - public ArrayList searchConsentAuthorizations(Connection connection, String consentID, - String userID) - throws OBConsentDataRetrievalException { - - ArrayList retrievedAuthorizationResources = new ArrayList<>(); - Map conditions = new HashMap<>(); - if (StringUtils.trimToNull(consentID) != null) { - conditions.put("CONSENT_ID", consentID); - } - if (StringUtils.trimToNull(userID) != null) { - conditions.put("USER_ID", userID); - } - String whereClause = ConsentDAOUtils.constructAuthSearchPreparedStatement(conditions); - String searchAuthorizationResourcesPrepStatement = - sqlStatements.getSearchAuthorizationResourcesPreparedStatement(whereClause); - - try (PreparedStatement getSearchAuthorizationResourcesPreparedStmt = - connection.prepareStatement(searchAuthorizationResourcesPrepStatement)) { - - if (log.isDebugEnabled()) { - log.debug("Setting parameters to prepared statement to search authorization resources"); - } - - Iterator> conditionIterator = conditions.entrySet().iterator(); - - for (int count = 1; count <= conditions.size(); count++) { - getSearchAuthorizationResourcesPreparedStmt.setString(count, conditionIterator.next().getValue()); - } - - try (ResultSet resultSet = getSearchAuthorizationResourcesPreparedStmt.executeQuery()) { - if (resultSet.isBeforeFirst()) { - while (resultSet.next()) { - AuthorizationResource authorizationResource = new AuthorizationResource(); - authorizationResource - .setAuthorizationID(resultSet.getString(ConsentMgtDAOConstants.AUTH_ID)); - authorizationResource.setConsentID(resultSet.getString(ConsentMgtDAOConstants.CONSENT_ID)); - authorizationResource - .setUserID(resultSet.getString(ConsentMgtDAOConstants.USER_ID)); - authorizationResource.setAuthorizationStatus(resultSet - .getString(ConsentMgtDAOConstants.AUTH_STATUS)); - authorizationResource - .setAuthorizationType(resultSet.getString(ConsentMgtDAOConstants.AUTH_TYPE)); - authorizationResource.setUpdatedTime(resultSet.getLong(ConsentMgtDAOConstants.UPDATED_TIME)); - retrievedAuthorizationResources.add(authorizationResource); - } - } else { - log.error("No records are found for the provided inputs"); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.NO_RECORDS_FOUND_ERROR_MSG); - } - } catch (SQLException e) { - log.error("Error occurred while searching authorization resources", e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants - .CONSENT_AUTHORIZATION_RESOURCE_RETRIEVE_ERROR_MSG, e); - } - log.debug("Retrieved the authorization resources successfully"); - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_AUTHORIZATION_RESOURCE_RETRIEVE_ERROR_MSG, e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants - .CONSENT_AUTHORIZATION_RESOURCE_RETRIEVE_ERROR_MSG, e); - } - return retrievedAuthorizationResources; - } - - /** - * Set data from the result set to ConsentResource object. - * - * @param resultSet result set - * @param consentResource consent resource - * @throws SQLException thrown if an error occurs when getting data from the result set - */ - private void setDataToConsentResource(ResultSet resultSet, ConsentResource consentResource) throws SQLException { - - consentResource.setConsentID(resultSet.getString(ConsentMgtDAOConstants.CONSENT_ID)); - consentResource.setReceipt(resultSet.getString(ConsentMgtDAOConstants.RECEIPT)); - consentResource.setCreatedTime(resultSet.getLong(ConsentMgtDAOConstants.CREATED_TIME)); - consentResource.setUpdatedTime(resultSet.getLong(ConsentMgtDAOConstants.UPDATED_TIME)); - consentResource.setClientID(resultSet.getString(ConsentMgtDAOConstants.CLIENT_ID)); - consentResource.setConsentType(resultSet.getString(ConsentMgtDAOConstants.CONSENT_TYPE)); - consentResource. - setCurrentStatus(resultSet.getString(ConsentMgtDAOConstants.CURRENT_STATUS)); - consentResource.setConsentFrequency(resultSet - .getInt(ConsentMgtDAOConstants.CONSENT_FREQUENCY)); - consentResource.setValidityPeriod(resultSet.getLong(ConsentMgtDAOConstants.VALIDITY_TIME)); - consentResource.setRecurringIndicator(resultSet.getBoolean( - ConsentMgtDAOConstants.RECURRING_INDICATOR)); - } - - /** - * Set data from the result set to DetaildConsentResource object. - * - * @param resultSet result set - * @param detailedConsentResource consent resource - * @throws SQLException thrown if an error occurs when getting data from the result set - */ - private void setDataToDetailedConsentResource(ResultSet resultSet, DetailedConsentResource detailedConsentResource) - throws SQLException { - - Map consentAttributesMap = new HashMap<>(); - ArrayList authorizationResources = new ArrayList<>(); - ArrayList consentMappingResources = new ArrayList<>(); - ArrayList authIds = new ArrayList<>(); - ArrayList consentMappingIds = new ArrayList<>(); - - while (resultSet.next()) { - // Set data related to the consent resource - setConsentDataToDetailedConsentResource(resultSet, detailedConsentResource); - - // Set data related to consent attributes - if (StringUtils.isNotBlank(resultSet.getString(ConsentMgtDAOConstants.ATT_KEY))) { - String attributeValue = resultSet.getString(ConsentMgtDAOConstants.ATT_VALUE); - - // skip adding all temporary session data to consent attributes - if (!(JSONValue.isValidJson(attributeValue) && attributeValue.contains(SESSION_DATA_KEY))) { - consentAttributesMap.put(resultSet.getString(ConsentMgtDAOConstants.ATT_KEY), - attributeValue); - } - } - - // Set data related to authorization resources - if (authIds.isEmpty()) { - AuthorizationResource authorizationResource = new AuthorizationResource(); - authorizationResource.setAuthorizationID(resultSet.getString(ConsentMgtDAOConstants.AUTH_ID)); - authorizationResource.setConsentID(resultSet.getString(ConsentMgtDAOConstants.CONSENT_ID)); - authorizationResource.setAuthorizationStatus(resultSet.getString(ConsentMgtDAOConstants.AUTH_STATUS)); - authorizationResource.setAuthorizationType(resultSet.getString(ConsentMgtDAOConstants.AUTH_TYPE)); - authorizationResource.setUserID(resultSet.getString(ConsentMgtDAOConstants.USER_ID)); - authorizationResource.setUpdatedTime(resultSet.getLong(ConsentMgtDAOConstants.AUTH_UPDATED_TIME)); - authorizationResources.add(authorizationResource); - authIds.add(authorizationResource.getAuthorizationID()); - } else { - if (!authIds.contains(resultSet.getString(ConsentMgtDAOConstants.AUTH_ID))) { - AuthorizationResource authorizationResource = new AuthorizationResource(); - authorizationResource.setAuthorizationID(resultSet.getString(ConsentMgtDAOConstants.AUTH_ID)); - authorizationResource.setConsentID(resultSet.getString(ConsentMgtDAOConstants.CONSENT_ID)); - authorizationResource - .setAuthorizationStatus(resultSet.getString(ConsentMgtDAOConstants.AUTH_STATUS)); - authorizationResource.setAuthorizationType(resultSet.getString(ConsentMgtDAOConstants.AUTH_TYPE)); - authorizationResource.setUserID(resultSet.getString(ConsentMgtDAOConstants.USER_ID)); - authorizationResource.setUpdatedTime(resultSet.getLong(ConsentMgtDAOConstants.AUTH_UPDATED_TIME)); - authorizationResources.add(authorizationResource); - authIds.add(authorizationResource.getAuthorizationID()); - } - } - - // Set data related to consent account mappings - // Check whether consentMappingIds is empty and result set consists a mapping id since at this moment - // there can be a situation where an auth resource is created and mapping resource is not created - if (consentMappingIds.isEmpty() && resultSet.getString(ConsentMgtDAOConstants.MAPPING_ID) != null) { - ConsentMappingResource consentMappingResource = new ConsentMappingResource(); - consentMappingResource.setAuthorizationID(resultSet.getString(ConsentMgtDAOConstants.AUTH_ID)); - consentMappingResource.setAccountID(resultSet.getString(ConsentMgtDAOConstants.ACCOUNT_ID)); - consentMappingResource.setMappingID(resultSet.getString(ConsentMgtDAOConstants.MAPPING_ID)); - consentMappingResource.setMappingStatus(resultSet.getString(ConsentMgtDAOConstants.MAPPING_STATUS)); - consentMappingResource.setPermission(resultSet.getString(ConsentMgtDAOConstants.PERMISSION)); - consentMappingResources.add(consentMappingResource); - consentMappingIds.add(consentMappingResource.getMappingID()); - } else { - // Check whether result set consists a mapping id since at this moment, there can be a situation - // where an auth resource is created and mapping resource is not created - if (!consentMappingIds.contains(resultSet.getString(ConsentMgtDAOConstants.MAPPING_ID)) && - resultSet.getString(ConsentMgtDAOConstants.MAPPING_ID) != null) { - ConsentMappingResource consentMappingResource = new ConsentMappingResource(); - consentMappingResource.setAuthorizationID(resultSet.getString(ConsentMgtDAOConstants.AUTH_ID)); - consentMappingResource.setAccountID(resultSet.getString(ConsentMgtDAOConstants.ACCOUNT_ID)); - consentMappingResource.setMappingID(resultSet.getString(ConsentMgtDAOConstants.MAPPING_ID)); - consentMappingResource.setMappingStatus(resultSet.getString(ConsentMgtDAOConstants.MAPPING_STATUS)); - consentMappingResource.setPermission(resultSet.getString(ConsentMgtDAOConstants.PERMISSION)); - consentMappingResources.add(consentMappingResource); - consentMappingIds.add(consentMappingResource.getMappingID()); - } - } - } - - // Set consent attributes, auth resources and account mappings to detailed consent resource - detailedConsentResource.setConsentAttributes(consentAttributesMap); - detailedConsentResource.setAuthorizationResources(authorizationResources); - detailedConsentResource.setConsentMappingResources(consentMappingResources); - } - - void setConsentDataToDetailedConsentResource(ResultSet resultSet, - DetailedConsentResource detailedConsentResource) - throws SQLException { - - detailedConsentResource.setConsentID(resultSet.getString(ConsentMgtDAOConstants.CONSENT_ID)); - detailedConsentResource.setClientID(resultSet.getString(ConsentMgtDAOConstants.CLIENT_ID)); - detailedConsentResource.setReceipt(resultSet.getString(ConsentMgtDAOConstants.RECEIPT)); - detailedConsentResource.setCreatedTime(resultSet.getLong(ConsentMgtDAOConstants.CONSENT_CREATED_TIME)); - detailedConsentResource.setUpdatedTime(resultSet.getLong(ConsentMgtDAOConstants.CONSENT_UPDATED_TIME)); - detailedConsentResource.setConsentType(resultSet.getString(ConsentMgtDAOConstants.CONSENT_TYPE)); - detailedConsentResource.setCurrentStatus(resultSet.getString(ConsentMgtDAOConstants.CURRENT_STATUS)); - detailedConsentResource.setConsentFrequency(resultSet.getInt(ConsentMgtDAOConstants.CONSENT_FREQUENCY)); - detailedConsentResource.setValidityPeriod(resultSet.getLong(ConsentMgtDAOConstants.VALIDITY_TIME)); - detailedConsentResource.setRecurringIndicator(resultSet - .getBoolean(ConsentMgtDAOConstants.RECURRING_INDICATOR)); - } - - /** - * Return a consent mapping resource with data set from the result set. - * - * @param resultSet result set - * @return a consent mapping resource - * @throws SQLException thrown if an error occurs when getting data from the result set - */ - private ConsentMappingResource getConsentMappingResourceWithData(ResultSet resultSet) throws SQLException { - - ConsentMappingResource consentMappingResource = new ConsentMappingResource(); - consentMappingResource.setAuthorizationID(resultSet.getString(ConsentMgtDAOConstants.AUTH_ID)); - consentMappingResource.setMappingID(resultSet.getString(ConsentMgtDAOConstants.MAPPING_ID)); - consentMappingResource.setAccountID(resultSet.getString(ConsentMgtDAOConstants.ACCOUNT_ID)); - consentMappingResource.setPermission(resultSet.getString(ConsentMgtDAOConstants.PERMISSION)); - consentMappingResource.setMappingStatus(resultSet.getString(ConsentMgtDAOConstants.MAPPING_STATUS)); - - return consentMappingResource; - } - - /** - * Sets search parameters to dynamically constructed prepared statement. The outer loop is used to iterate the - * different AND clauses and the inner loop is to iterate the number of placeholders of the current AND clause. - * - * @param preparedStatement dynamically constructed prepared statement - * @param orderedParamsMap map with ordered AND conditions - * @param parameterIndex index which the parameter should be set - * @return the final parameter index - * @throws SQLException thrown if an error occurs in the process - */ - int setDynamicConsentSearchParameters(PreparedStatement preparedStatement, Map orderedParamsMap, - int parameterIndex) - throws SQLException { - - for (Map.Entry entry : orderedParamsMap.entrySet()) { - for (int valueIndex = 0; valueIndex < entry.getValue().size(); valueIndex++) { - preparedStatement.setString(parameterIndex, ((String) entry.getValue().get(valueIndex)).trim()); - parameterIndex++; - } - } - return parameterIndex; - } - - int getResultSetSize(ResultSet resultSet) throws SQLException { - - resultSet.last(); - int resultSetSize = resultSet.getRow(); - - // Point result set back before first - resultSet.beforeFirst(); - return resultSetSize; - } - - void setAuthorizationData(AuthorizationResource authorizationResource, ResultSet resultSet) - throws SQLException { - - authorizationResource.setAuthorizationID(resultSet - .getString(ConsentMgtDAOConstants.AUTH_ID)); - authorizationResource.setConsentID(resultSet.getString(ConsentMgtDAOConstants.CONSENT_ID)); - authorizationResource.setAuthorizationType(resultSet - .getString(ConsentMgtDAOConstants.AUTH_TYPE)); - authorizationResource.setAuthorizationStatus(resultSet - .getString(ConsentMgtDAOConstants.AUTH_STATUS)); - authorizationResource.setUpdatedTime(resultSet - .getLong(ConsentMgtDAOConstants.UPDATED_TIME)); - authorizationResource.setUserID(resultSet.getString(ConsentMgtDAOConstants.USER_ID)); - } - - protected void setAuthorizationDataInResponseForGroupedQuery(ArrayList - authorizationResources, - ResultSet resultSet, String consentId) - throws SQLException { - - //identify duplicate auth data - Set authIdSet = new HashSet<>(); - - // fetch values from group_concat - String[] authIds = resultSet.getString(ConsentMgtDAOConstants.AUTH_ID) != null ? - resultSet.getString(ConsentMgtDAOConstants.AUTH_ID).split(GROUP_BY_SEPARATOR) : null; - String[] authTypes = resultSet.getString(ConsentMgtDAOConstants.AUTH_TYPE) != null ? - resultSet.getString(ConsentMgtDAOConstants.AUTH_TYPE).split(GROUP_BY_SEPARATOR) : null; - String[] authStatues = resultSet.getString(ConsentMgtDAOConstants.AUTH_STATUS) != null ? - resultSet.getString(ConsentMgtDAOConstants.AUTH_STATUS).split(GROUP_BY_SEPARATOR) : null; - String[] updatedTimes = resultSet.getString(ConsentMgtDAOConstants.UPDATED_TIME) != null ? - resultSet.getString(ConsentMgtDAOConstants.UPDATED_TIME).split(GROUP_BY_SEPARATOR) : null; - String[] userIds = resultSet.getString(ConsentMgtDAOConstants.USER_ID) != null ? - resultSet.getString(ConsentMgtDAOConstants.USER_ID).split(GROUP_BY_SEPARATOR) : null; - - for (int index = 0; index < (authIds != null ? authIds.length : 0); index++) { - if (!authIdSet.contains(authIds[index])) { - AuthorizationResource authorizationResource = new AuthorizationResource(); - authIdSet.add(authIds[index]); - authorizationResource.setAuthorizationID(authIds[index]); - authorizationResource.setConsentID(consentId); - if (authTypes != null && authTypes.length > index) { - authorizationResource.setAuthorizationType(authTypes[index]); - } - if (authStatues != null && authStatues.length > index) { - authorizationResource.setAuthorizationStatus(authStatues[index]); - } - if (updatedTimes != null && updatedTimes.length > index) { - authorizationResource.setUpdatedTime(Long.parseLong(updatedTimes[index])); - } - if (userIds != null && userIds.length > index) { - authorizationResource.setUserID(userIds[index]); - } - authorizationResources.add(authorizationResource); - } - } - - } - - protected void setAccountConsentMappingDataInResponse(ArrayList consentMappingResources, - ResultSet resultSet) throws SQLException { - - //identify duplicate mappingIds - Set mappingIdSet = new HashSet<>(); - - // fetch values from group_concat - String[] authIds = resultSet.getString(ConsentMgtDAOConstants.AUTH_MAPPING_ID) != null ? - resultSet.getString(ConsentMgtDAOConstants.AUTH_MAPPING_ID).split(GROUP_BY_SEPARATOR) : null; - String[] mappingIds = resultSet.getString(ConsentMgtDAOConstants.MAPPING_ID) != null ? - resultSet.getString(ConsentMgtDAOConstants.MAPPING_ID).split(GROUP_BY_SEPARATOR) : null; - String[] accountIds = resultSet.getString(ConsentMgtDAOConstants.ACCOUNT_ID) != null ? - resultSet.getString(ConsentMgtDAOConstants.ACCOUNT_ID).split(GROUP_BY_SEPARATOR) : null; - String[] mappingStatues = resultSet.getString(ConsentMgtDAOConstants.MAPPING_STATUS) != null ? - resultSet.getString(ConsentMgtDAOConstants.MAPPING_STATUS).split(GROUP_BY_SEPARATOR) : null; - String[] permissions = resultSet.getString(ConsentMgtDAOConstants.PERMISSION) != null ? - resultSet.getString(ConsentMgtDAOConstants.PERMISSION).split(GROUP_BY_SEPARATOR) : null; - - for (int index = 0; index < (mappingIds != null ? mappingIds.length : 0); index++) { - if (!mappingIdSet.contains(mappingIds[index])) { - ConsentMappingResource consentMappingResource = new ConsentMappingResource(); - if (authIds != null && authIds.length > index) { - consentMappingResource.setAuthorizationID(authIds[index]); - } - consentMappingResource.setMappingID(mappingIds[index]); - if (accountIds != null && accountIds.length > index) { - consentMappingResource.setAccountID(accountIds[index]); - } - if (mappingStatues != null && mappingStatues.length > index) { - consentMappingResource.setMappingStatus(mappingStatues[index]); - } - if (permissions != null && permissions.length > index) { - consentMappingResource.setPermission(permissions[index]); - } - consentMappingResources.add(consentMappingResource); - mappingIdSet.add(mappingIds[index]); - } - } - - } - - void validateAndSetSearchConditions(Map applicableConditionsMap, ArrayList consentIDs, - ArrayList clientIDs, - ArrayList consentTypes, - ArrayList consentStatuses) { - - log.debug("Validate applicable search conditions"); - - if (CollectionUtils.isNotEmpty(consentIDs)) { - applicableConditionsMap.put(COLUMNS_MAP.get(ConsentMgtDAOConstants.CONSENT_IDS), consentIDs); - } - if (CollectionUtils.isNotEmpty(clientIDs)) { - applicableConditionsMap.put(COLUMNS_MAP.get(ConsentMgtDAOConstants.CLIENT_IDS), clientIDs); - } - if (CollectionUtils.isNotEmpty(consentTypes)) { - applicableConditionsMap.put(COLUMNS_MAP.get(ConsentMgtDAOConstants.CONSENT_TYPES), consentTypes); - } - if (CollectionUtils.isNotEmpty(consentStatuses)) { - applicableConditionsMap.put(COLUMNS_MAP.get(ConsentMgtDAOConstants.CONSENT_STATUSES), consentStatuses); - } - } - - ArrayList constructDetailedConsentsSearchResult(ResultSet resultSet, int resultSetSize) - throws SQLException { - - ArrayList detailedConsentResources = new ArrayList<>(); - - while (resultSet.next()) { - - Map consentAttributesMap = new HashMap<>(); - ArrayList consentMappingResources = new ArrayList<>(); - ArrayList authorizationResources = new ArrayList<>(); - DetailedConsentResource detailedConsentResource = new DetailedConsentResource(); - - setConsentDataToDetailedConsentResource(resultSet, detailedConsentResource); - - // Set consent attributes to map if available - if (resultSet.getString(ConsentMgtDAOConstants.ATT_KEY) != null && - StringUtils.isNotBlank(resultSet.getString(ConsentMgtDAOConstants.ATT_KEY)) - && StringUtils.isNotBlank(resultSet.getString(ConsentMgtDAOConstants.ATT_VALUE))) { - // fetch attribute keys and values from group_concat - String[] attributeKeys = resultSet.getString(ConsentMgtDAOConstants.ATT_KEY).split(GROUP_BY_SEPARATOR); - String[] attributeValues = resultSet - .getString(ConsentMgtDAOConstants.ATT_VALUE).split(GROUP_BY_SEPARATOR); - // check if all attribute keys has values - if (attributeKeys.length == attributeValues.length) { - for (int index = 0; index < attributeKeys.length; index++) { - consentAttributesMap.put(attributeKeys[index], attributeValues[index]); - } - } - } - // Set authorization data - setAuthorizationDataInResponseForGroupedQuery(authorizationResources, resultSet, - detailedConsentResource.getConsentID()); - // Set consent account mapping data if available - setAccountConsentMappingDataInResponse(consentMappingResources, resultSet); - - detailedConsentResource.setConsentAttributes(consentAttributesMap); - detailedConsentResource.setAuthorizationResources(authorizationResources); - detailedConsentResource.setConsentMappingResources(consentMappingResources); - - detailedConsentResources.add(detailedConsentResource); - - } - return detailedConsentResources; - } - - @Override - public boolean updateConsentReceipt(Connection connection, String consentID, String consentReceipt) - throws OBConsentDataUpdationException { - - int result; - String updateConsentReceiptPrepStatement = sqlStatements.getUpdateConsentReceiptPreparedStatement(); - - try (PreparedStatement updateConsentReceiptPreparedStmt = - connection.prepareStatement(updateConsentReceiptPrepStatement)) { - - log.debug("Setting parameters to prepared statement to update consent receipt"); - - updateConsentReceiptPreparedStmt.setString(1, consentReceipt); - updateConsentReceiptPreparedStmt.setString(2, consentID); - - // with result, we can determine whether the updating was successful or not - result = updateConsentReceiptPreparedStmt.executeUpdate(); - } catch (SQLException e) { - log.error("Error while updating consent receipt", e); - throw new OBConsentDataUpdationException("Error while updating consent receipt for consent ID: " - + consentID, e); - } - - // Confirm that the data are updated successfully - if (result > 0) { - return true; - } else { - throw new OBConsentDataUpdationException("Failed to update consent receipt properly."); - } - } - - @Override - public boolean updateConsentValidityTime(Connection connection, String consentID, long validityTime) - throws OBConsentDataUpdationException { - - int result; - String updateConsentReceiptPrepStatement = sqlStatements.getUpdateConsentValidityTimePreparedStatement(); - long updatedTime = System.currentTimeMillis() / 1000; - - try (PreparedStatement updateConsentValidityTimePreparedStmt = - connection.prepareStatement(updateConsentReceiptPrepStatement)) { - - log.debug("Setting parameters to prepared statement to update consent receipt"); - - updateConsentValidityTimePreparedStmt.setLong(1, validityTime); - updateConsentValidityTimePreparedStmt.setLong(2, updatedTime); - updateConsentValidityTimePreparedStmt.setString(3, consentID); - - // with result, we can determine whether the updating was successful or not - result = updateConsentValidityTimePreparedStmt.executeUpdate(); - } catch (SQLException e) { - log.error("Error while updating consent validity time", e); - throw new OBConsentDataUpdationException("Error while updating consent validity time for consent ID: " - + consentID, e); - } - - // Confirm that the data are updated successfully - if (result > 0) { - return true; - } else { - throw new OBConsentDataUpdationException("Failed to update consent validity time properly."); - } - } - - @Override - public boolean storeConsentAmendmentHistory(Connection connection, String historyID, long timestamp, - String recordID, String consentDataType, String changedAttributesJsonString, String amendmentReason) - throws OBConsentDataInsertionException { - - String tableID = generateConsentTableId(consentDataType); - - int result; - String insertConsentHistoryPrepStatement = sqlStatements.getInsertConsentHistoryPreparedStatement(); - - try (PreparedStatement insertConsentHistoryPreparedStmt = - connection.prepareStatement(insertConsentHistoryPrepStatement)) { - - if (log.isDebugEnabled()) { - log.debug(String.format("Setting parameters to prepared statement to store consent amendment history " + - "of %s", consentDataType)); - } - - insertConsentHistoryPreparedStmt.setString(1, tableID); - insertConsentHistoryPreparedStmt.setString(2, recordID); - insertConsentHistoryPreparedStmt.setString(3, historyID); - insertConsentHistoryPreparedStmt.setString(4, changedAttributesJsonString); - insertConsentHistoryPreparedStmt.setString(5, amendmentReason); - insertConsentHistoryPreparedStmt.setLong(6, timestamp); - - // with result, we can determine whether the updating was successful or not - result = insertConsentHistoryPreparedStmt.executeUpdate(); - } catch (SQLException e) { - log.error("Error while storing consent amendment history", e); - throw new OBConsentDataInsertionException(String.format("Error while storing consent amendment history of" + - " %s for record ID: %s", consentDataType, recordID), e); - } - - // Confirm that the data are inserted successfully - if (result > 0) { - return true; - } else { - log.error("Failed to store consent amendment history data."); - throw new OBConsentDataInsertionException("Failed to store consent amendment history data properly."); - } - } - - @Override - public Map retrieveConsentAmendmentHistory(Connection connection, - List recordIDsList) throws OBConsentDataRetrievalException { - - String whereClause = ConsentDAOUtils.constructConsentHistoryPreparedStatement(recordIDsList.size()); - String getConsentHistoryPrepStatement = sqlStatements.getGetConsentHistoryPreparedStatement(whereClause); - - try (PreparedStatement getConsentHistoryPreparedStmt = - connection.prepareStatement(getConsentHistoryPrepStatement)) { - - log.debug("Setting parameters to prepared statement to retrieve consent history data"); - - for (int count = 1; count <= recordIDsList.size(); count++) { - getConsentHistoryPreparedStmt.setString(count, recordIDsList.get(count - 1)); - } - - String consentID = recordIDsList.get(0); - try (ResultSet resultSet = getConsentHistoryPreparedStmt.executeQuery()) { - if (resultSet.isBeforeFirst()) { - return constructConsentHistoryRetrievalResult(consentID, resultSet); - } else { - log.error("No records are found for consent ID : " + consentID); - return new HashMap<>(); - } - } catch (SQLException e) { - log.error("Error occurred while reading consent amendment history", e); - throw new OBConsentDataRetrievalException(String.format("Error occurred while retrieving consent " + - "amendment history for consent ID : %s", consentID), e); - } - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_AMENDMENT_HISTORY_RETRIEVE_ERROR_MSG, e); - throw new OBConsentDataRetrievalException( - ConsentMgtDAOConstants.CONSENT_AMENDMENT_HISTORY_RETRIEVE_ERROR_MSG, e); - } - } - - /** - * construct a data map that includes the changed attributes of each consent amendment history entry and. - * return a map of ConsentHistoryResources including this changed attributes data map - * - * @param consentId consent Id - * @param resultSet result set - * @return a map of ConsentHistoryResources - * @throws SQLException thrown if an error occurs when getting data from the result set - */ - private Map constructConsentHistoryRetrievalResult(String consentId, - ResultSet resultSet) - throws SQLException { - - Map consentAmendmentHistoryDataMap = new LinkedHashMap<>(); - - while (resultSet.next()) { - String tableID = resultSet.getString(ConsentMgtDAOConstants.TABLE_ID); - String recordID = resultSet.getString(ConsentMgtDAOConstants.RECORD_ID); - String historyId = resultSet.getString(ConsentMgtDAOConstants.HISTORY_ID); - String changedAttributesString = resultSet.getString(ConsentMgtDAOConstants.CHANGED_VALUES); - String amendmentReason = resultSet.getString(ConsentMgtDAOConstants.REASON); - Long timestamp = resultSet.getLong(ConsentMgtDAOConstants.EFFECTIVE_TIMESTAMP); - - ConsentHistoryResource consentHistoryResource; - Map changedAttributesJsonDataMap; - if (consentAmendmentHistoryDataMap.containsKey(historyId)) { - consentHistoryResource = consentAmendmentHistoryDataMap.get(historyId); - } else { - consentHistoryResource = new ConsentHistoryResource(consentId, historyId); - consentHistoryResource.setTimestamp(timestamp); - consentHistoryResource.setReason(amendmentReason); - } - - changedAttributesJsonDataMap = consentHistoryResource.getChangedAttributesJsonDataMap(); - - if (TABLES_MAP.get(ConsentMgtDAOConstants.TABLE_OB_CONSENT).equalsIgnoreCase(tableID)) { - changedAttributesJsonDataMap.put(ConsentMgtDAOConstants.TYPE_CONSENT_BASIC_DATA, - changedAttributesString); - } else if (TABLES_MAP.get(ConsentMgtDAOConstants.TABLE_OB_CONSENT_ATTRIBUTE).equalsIgnoreCase(tableID)) { - changedAttributesJsonDataMap.put(ConsentMgtDAOConstants.TYPE_CONSENT_ATTRIBUTES_DATA, - changedAttributesString); - } else if (TABLES_MAP.get(ConsentMgtDAOConstants.TABLE_OB_CONSENT_AUTH_RESOURCE) - .equalsIgnoreCase(tableID)) { - Map consentAuthResources; - if (changedAttributesJsonDataMap.containsKey(ConsentMgtDAOConstants.TYPE_CONSENT_AUTH_RESOURCE_DATA)) { - consentAuthResources = (Map) changedAttributesJsonDataMap - .get(ConsentMgtDAOConstants.TYPE_CONSENT_AUTH_RESOURCE_DATA); - } else { - consentAuthResources = new HashMap<>(); - } - consentAuthResources.put(recordID, changedAttributesString); - changedAttributesJsonDataMap.put(ConsentMgtDAOConstants.TYPE_CONSENT_AUTH_RESOURCE_DATA, - consentAuthResources); - } else if (TABLES_MAP.get(ConsentMgtDAOConstants.TABLE_OB_CONSENT_MAPPING).equalsIgnoreCase(tableID)) { - Map consentMappingResources; - if (changedAttributesJsonDataMap.containsKey(ConsentMgtDAOConstants.TYPE_CONSENT_MAPPING_DATA)) { - consentMappingResources = (Map) changedAttributesJsonDataMap - .get(ConsentMgtDAOConstants.TYPE_CONSENT_MAPPING_DATA); - } else { - consentMappingResources = new HashMap<>(); - } - consentMappingResources.put(recordID, changedAttributesString); - changedAttributesJsonDataMap.put(ConsentMgtDAOConstants.TYPE_CONSENT_MAPPING_DATA, - consentMappingResources); - } else { - log.error(String.format("The retrieved tableId : %s has no corresponding consent data type to be" + - " matched", tableID)); - } - consentHistoryResource.setChangedAttributesJsonDataMap(changedAttributesJsonDataMap); - consentAmendmentHistoryDataMap.put(historyId, consentHistoryResource); - } - return consentAmendmentHistoryDataMap; - } - - public ArrayList getExpiringConsents(Connection connection, - String statusesEligibleForExpiration) - throws OBConsentDataRetrievalException { - - List statusesEligibleForExpirationList = Arrays.asList(statusesEligibleForExpiration.split(",")) - .stream().filter(status -> !status.isEmpty()) - .collect(Collectors.toList()); - - String statusesEligibleForExpirationCondition = ConsentDAOUtils.constructStatusesEligibleForExpirationCondition( - statusesEligibleForExpirationList); - String expiringConsentStatement = sqlStatements.getSearchExpiringConsentPreparedStatement( - statusesEligibleForExpirationCondition); - - try (PreparedStatement preparedStatement = - connection.prepareStatement(expiringConsentStatement)) { - - log.debug("Setting parameters to prepared statement to fetch consents eligible for expiration"); - - ArrayList consentIdList = new ArrayList<>(); - - // populate prepared statement - int parameterIndex = 0; - preparedStatement.setString(++parameterIndex, ConsentMgtDAOConstants.CONSENT_EXPIRY_TIME_ATTRIBUTE); - for (String status : statusesEligibleForExpirationList) { - preparedStatement.setString(++parameterIndex, status); - } - - try (ResultSet resultSet = preparedStatement.executeQuery()) { - if (resultSet.isBeforeFirst()) { - while (resultSet.next()) { - consentIdList.add(resultSet.getString(ConsentMgtDAOConstants.CONSENT_ID)); - } - } else { - log.debug("No consents found for expiration check eligibility."); - } - if (!consentIdList.isEmpty()) { - return searchConsents(connection, consentIdList, null, null, null, - null, null, null, null, null); - } else { - return new ArrayList<>(); - } - - } catch (SQLException e) { - log.error("Error occurred while searching consents eligible for expiration", e); - throw new OBConsentDataRetrievalException("Error occurred while searching consents" + - " eligible for expiration", e); - } - } catch (SQLException e) { - log.error("Error while searching consents eligible for expiration", e); - throw new OBConsentDataRetrievalException("Error while updating searching consents eligible for" + - " expiration", e); - } - } - - @Override - public boolean deleteConsentData(Connection connection, String consentID, boolean executeOnRetentionTables) - throws OBConsentDataDeletionException { - - if (log.isDebugEnabled()) { - log.debug(String.format("Deleting consent details for consent_id : %s", consentID)); - } - - int results; - - String deleteConsentAttributePrepStatement = sqlStatements - .getDeleteConsentAttributeByConsentIdPreparedStatement(executeOnRetentionTables); - String deleteConsentFilePrepStatement = sqlStatements - .getDeleteConsentFileResourcePreparedStatement(executeOnRetentionTables); - String deleteConsentMappingPrepStatement = sqlStatements - .getDeleteConsentMappingByAuthIdPreparedStatement(executeOnRetentionTables); - String deleteConsentAuthResourcePrepStatement = sqlStatements - .getDeleteAuthorizationResourcePreparedStatement(executeOnRetentionTables); - String deleteConsentStatusAuditRecordPrepStatement = sqlStatements - .getDeleteConsentStatusAuditRecordsPreparedStatement(executeOnRetentionTables); - String deleteConsentResourcePrepStatement = sqlStatements - .getDeleteConsentPreparedStatement(executeOnRetentionTables); - - try (PreparedStatement deleteConsentAttributesPreparedStmt = - connection.prepareStatement(deleteConsentAttributePrepStatement); - PreparedStatement deleteConsentFilePreparedStmt = - connection.prepareStatement(deleteConsentFilePrepStatement); - PreparedStatement deleteConsentMappingPreparedStmt = - connection.prepareStatement(deleteConsentMappingPrepStatement); - PreparedStatement deleteConsentAuthResourcePreparedStmt = - connection.prepareStatement(deleteConsentAuthResourcePrepStatement); - PreparedStatement deleteConsentStatusAuditPreparedStmt = - connection.prepareStatement(deleteConsentStatusAuditRecordPrepStatement); - PreparedStatement deleteConsentResourcePreparedStmt = - connection.prepareStatement(deleteConsentResourcePrepStatement)) { - - // deleting consent attributes. - log.debug("Setting parameters to prepared statement to delete consent attributes"); - deleteConsentAttributesPreparedStmt.setString(1, consentID); - deleteConsentAttributesPreparedStmt.executeUpdate(); - - // deleting consent file. - log.debug("Setting parameters to prepared statement to delete consent files"); - deleteConsentFilePreparedStmt.setString(1, consentID); - deleteConsentFilePreparedStmt.executeUpdate(); - - // deleting consent mappings - log.debug("Setting parameters to prepared statement to delete consent mappings"); - deleteConsentMappingPreparedStmt.setString(1, consentID); - deleteConsentMappingPreparedStmt.executeUpdate(); - - // deleting consent auth resource. - log.debug("Setting parameters to prepared statement to delete consent auth resource"); - deleteConsentAuthResourcePreparedStmt.setString(1, consentID); - deleteConsentAuthResourcePreparedStmt.executeUpdate(); - - // deleting consent status audit. - log.debug("Setting parameters to prepared statement to delete consent files"); - deleteConsentStatusAuditPreparedStmt.setString(1, consentID); - deleteConsentStatusAuditPreparedStmt.executeUpdate(); - - // deleting consent resource. - log.debug("Setting parameters to prepared statement to delete consent resource"); - deleteConsentResourcePreparedStmt.setString(1, consentID); - results = deleteConsentResourcePreparedStmt.executeUpdate(); - - return results > 0; - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_DATA_DELETE_ERROR_MSG, e); - throw new OBConsentDataDeletionException(ConsentMgtDAOConstants.CONSENT_DATA_DELETE_ERROR_MSG, e); - } - } - - @Override - public ArrayList getListOfConsentIds(Connection connection, boolean fetchFromRetentionTable) - throws OBConsentDataRetrievalException { - - String getConsentIdsPrepStatement = - sqlStatements.getListOfConsentIdsPreparedStatement(fetchFromRetentionTable); - ArrayList consentIDs = new ArrayList<>(); - - try (PreparedStatement getConsentIdsPreparedStmt = - connection.prepareStatement(getConsentIdsPrepStatement)) { - - try (ResultSet resultSet = getConsentIdsPreparedStmt.executeQuery()) { - while (resultSet.next()) { - consentIDs.add(resultSet.getString(ConsentMgtDAOConstants.CONSENT_ID)); - } - } catch (SQLException e) { - log.error("Error occurred while reading consent_id list", e); - throw new OBConsentDataRetrievalException("Error occurred while retrieving consent consent IDs list", - e); - } - - if (log.isDebugEnabled()) { - log.debug("Retrieved the consent id list from consent table"); - } - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_RESOURCE_RETRIEVE_ERROR_MSG, e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.CONSENT_RESOURCE_RETRIEVE_ERROR_MSG, e); - } - return consentIDs; - } - - @Override - public ArrayList getConsentStatusAuditRecordsByConsentId(Connection connection, - ArrayList consentIDs, - Integer limit, Integer offset, - boolean fetchFromRetentionTable) - throws OBConsentDataRetrievalException { - - boolean shouldLimit = true; - boolean shouldOffset = true; - int parameterIndex = 0; - - // Don't limit if either of limit or offset is null - if (limit == null) { - shouldLimit = false; - } - if (offset == null) { - shouldOffset = false; - } - - ArrayList retrievedAuditRecords = new ArrayList<>(); - String constructedConditions = - ConsentDAOUtils.constructConsentAuditRecordSearchPreparedStatement(consentIDs); - - String getConsentStatusAuditRecordsPrepStatement = - sqlStatements.getConsentStatusAuditRecordsByConsentIdsPreparedStatement(constructedConditions, - shouldLimit, shouldOffset, fetchFromRetentionTable); - - try (PreparedStatement getConsentStatusAuditRecordPreparedStmt = - connection.prepareStatement(getConsentStatusAuditRecordsPrepStatement)) { - - log.debug("Setting parameters to prepared statement to retrieve consent status audit records"); - if (!CollectionUtils.isEmpty(consentIDs)) { - for (String consentId : consentIDs) { - parameterIndex++; - getConsentStatusAuditRecordPreparedStmt.setString(parameterIndex, consentId); - } - } - if (limit != null && offset != null) { - getConsentStatusAuditRecordPreparedStmt.setInt(++parameterIndex, - sqlStatements.isLimitBeforeThanOffset() ? limit : offset); - getConsentStatusAuditRecordPreparedStmt.setInt(++parameterIndex, - sqlStatements.isLimitBeforeThanOffset() ? offset : limit); - } else if (limit != null) { - getConsentStatusAuditRecordPreparedStmt.setInt(++parameterIndex, limit); - } - - try (ResultSet resultSet = getConsentStatusAuditRecordPreparedStmt.executeQuery()) { - if (resultSet.isBeforeFirst()) { - while (resultSet.next()) { - ConsentStatusAuditRecord consentStatusAuditRecord = new ConsentStatusAuditRecord(); - consentStatusAuditRecord - .setStatusAuditID(resultSet.getString(ConsentMgtDAOConstants.STATUS_AUDIT_ID)); - consentStatusAuditRecord.setConsentID(resultSet.getString(ConsentMgtDAOConstants.CONSENT_ID)); - consentStatusAuditRecord - .setCurrentStatus(resultSet.getString(ConsentMgtDAOConstants.CURRENT_STATUS)); - consentStatusAuditRecord.setActionBy(resultSet.getString(ConsentMgtDAOConstants.ACTION_BY)); - consentStatusAuditRecord.setActionTime(resultSet.getLong(ConsentMgtDAOConstants.ACTION_TIME)); - consentStatusAuditRecord.setReason(resultSet.getString(ConsentMgtDAOConstants.REASON)); - consentStatusAuditRecord - .setPreviousStatus(resultSet.getString(ConsentMgtDAOConstants.PREVIOUS_STATUS)); - retrievedAuditRecords.add(consentStatusAuditRecord); - } - } - } catch (SQLException e) { - log.error("Error occurred while reading consent status audit records", e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.AUDIT_RECORDS_RETRIEVE_ERROR_MSG, e); - } - - log.debug("Retrieved the consent status audit records successfully"); - - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.AUDIT_RECORDS_RETRIEVE_ERROR_MSG, e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.AUDIT_RECORDS_RETRIEVE_ERROR_MSG, e); - } - return retrievedAuditRecords; - } - - /** - * Generate the tableID based on the type of the consent data record to be stored in consent history table. - * - * @param consentDataType A predefined consent data category based on each consent database table - * @return A identifier assigned for the relevant consent database table - */ - private String generateConsentTableId(String consentDataType) throws OBConsentDataInsertionException { - - String tableId; - if (ConsentMgtDAOConstants.TYPE_CONSENT_BASIC_DATA.equalsIgnoreCase(consentDataType)) { - tableId = TABLES_MAP.get(ConsentMgtDAOConstants.TABLE_OB_CONSENT); - } else if (ConsentMgtDAOConstants.TYPE_CONSENT_AUTH_RESOURCE_DATA.equalsIgnoreCase(consentDataType)) { - tableId = TABLES_MAP.get(ConsentMgtDAOConstants.TABLE_OB_CONSENT_AUTH_RESOURCE); - } else if (ConsentMgtDAOConstants.TYPE_CONSENT_ATTRIBUTES_DATA.equalsIgnoreCase(consentDataType)) { - tableId = TABLES_MAP.get(ConsentMgtDAOConstants.TABLE_OB_CONSENT_ATTRIBUTE); - } else if (ConsentMgtDAOConstants.TYPE_CONSENT_MAPPING_DATA.equalsIgnoreCase(consentDataType)) { - tableId = TABLES_MAP.get(ConsentMgtDAOConstants.TABLE_OB_CONSENT_MAPPING); - } else { - log.error(String.format("Can not find a table matching to the provided consentDataType : %s", - consentDataType)); - throw new OBConsentDataInsertionException("Error occurred while preparing to store consent amendment " + - "history data. Invalid consentDataType provided"); - } - return tableId; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/impl/MssqlConsentCoreDAOImpl.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/impl/MssqlConsentCoreDAOImpl.java deleted file mode 100644 index c966ba8f..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/impl/MssqlConsentCoreDAOImpl.java +++ /dev/null @@ -1,360 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.impl; - -import com.wso2.openbanking.accelerator.consent.mgt.dao.constants.ConsentMgtDAOConstants; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataRetrievalException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentMappingResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.queries.ConsentMgtMssqlDBQueries; -import com.wso2.openbanking.accelerator.consent.mgt.dao.utils.ConsentDAOUtils; -import org.apache.commons.collections.CollectionUtils; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.sql.Types; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.HashSet; -import java.util.Map; -import java.util.Optional; -import java.util.Set; - -/** - * DAO implementation for MSSQL specific methods. - */ -public class MssqlConsentCoreDAOImpl extends ConsentCoreDAOImpl { - - private static Log log = LogFactory.getLog(MssqlConsentCoreDAOImpl.class); - private static final String GROUP_BY_SEPARATOR = "\\|\\|"; - static final Map COLUMNS_MAP = new HashMap() { - { - put(ConsentMgtDAOConstants.CONSENT_IDS, "OBC.CONSENT_ID"); - put(ConsentMgtDAOConstants.CLIENT_IDS, "OBC.CLIENT_ID"); - put(ConsentMgtDAOConstants.CONSENT_TYPES, "OBC.CONSENT_TYPE"); - put(ConsentMgtDAOConstants.CONSENT_STATUSES, "OBC.CURRENT_STATUS"); - put(ConsentMgtDAOConstants.USER_IDS, "OCAR.USER_ID"); - } - }; - - public MssqlConsentCoreDAOImpl(ConsentMgtMssqlDBQueries sqlStatements) { - - super(sqlStatements); - } - - @Override - public ArrayList searchConsents(Connection connection, ArrayList consentIDs, - ArrayList clientIDs, - ArrayList consentTypes, - ArrayList consentStatuses, - ArrayList userIDs, Long fromTime, - Long toTime, Integer limit, Integer offset) - throws OBConsentDataRetrievalException { - - boolean shouldLimit = true; - boolean shouldOffset = true; - int parameterIndex = 0; - Map applicableConditionsMap = new HashMap<>(); - - validateAndSetSearchConditions(applicableConditionsMap, consentIDs, clientIDs, consentTypes, consentStatuses); - - if (limit == null) { - shouldLimit = false; - } - if (offset == null) { - shouldOffset = false; - } - - // logic to set the prepared statement - String constructedConditions = - ConsentDAOUtils.constructConsentSearchPreparedStatement(applicableConditionsMap); - - String userIDFilterCondition = ""; - Map userIdMap = new HashMap<>(); - if (CollectionUtils.isNotEmpty(userIDs)) { - userIdMap.put(COLUMNS_MAP.get(ConsentMgtDAOConstants.USER_IDS), userIDs); - userIDFilterCondition = ConsentDAOUtils.constructUserIdListFilterCondition(userIdMap); - } - String searchConsentsPreparedStatement = - sqlStatements.getSearchConsentsPreparedStatement(constructedConditions, shouldLimit, - shouldOffset, userIDFilterCondition); - - try (PreparedStatement searchConsentsPreparedStmt = - connection.prepareStatement(searchConsentsPreparedStatement, ResultSet.TYPE_SCROLL_INSENSITIVE, - ResultSet.CONCUR_READ_ONLY)) { - - /* Since we don't know the order of the set condition clauses, have to determine the order of them to set - the actual values to the prepared statement */ - Map orderedParamsMap = ConsentDAOUtils - .determineOrderOfParamsToSet(constructedConditions, applicableConditionsMap, COLUMNS_MAP); - - log.debug("Setting parameters to prepared statement to search consents"); - - //determine order of user Ids to set - if (CollectionUtils.isNotEmpty(userIDs)) { - Map orderedUserIdsMap = ConsentDAOUtils - .determineOrderOfParamsToSet(userIDFilterCondition, userIdMap, COLUMNS_MAP); - parameterIndex = setDynamicConsentSearchParameters(searchConsentsPreparedStmt, orderedUserIdsMap, - ++parameterIndex); - parameterIndex = parameterIndex - 1; - } - - parameterIndex = setDynamicConsentSearchParameters(searchConsentsPreparedStmt, orderedParamsMap, - ++parameterIndex); - parameterIndex = parameterIndex - 1; - - if (fromTime != null) { - searchConsentsPreparedStmt.setLong(++parameterIndex, fromTime); - } else { - searchConsentsPreparedStmt.setNull(++parameterIndex, Types.BIGINT); - } - - if (toTime != null) { - searchConsentsPreparedStmt.setLong(++parameterIndex, toTime); - } else { - searchConsentsPreparedStmt.setNull(++parameterIndex, Types.BIGINT); - } - - if (offset != null && limit != null) { - searchConsentsPreparedStmt.setInt(++parameterIndex, offset); - } - if (limit != null) { - searchConsentsPreparedStmt.setInt(++parameterIndex, limit); - } - - ArrayList detailedConsentResources = new ArrayList<>(); - - try (ResultSet resultSet = searchConsentsPreparedStmt.executeQuery()) { - if (resultSet.isBeforeFirst()) { - int resultSetSize = getResultSetSize(resultSet); - detailedConsentResources = constructDetailedConsentsSearchResult(resultSet, resultSetSize); - } - return detailedConsentResources; - } catch (SQLException e) { - log.error("Error occurred while searching detailed consent resources", e); - throw new OBConsentDataRetrievalException("Error occurred while searching detailed " + - "consent resources", e); - } - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_SEARCH_ERROR_MSG, e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.CONSENT_SEARCH_ERROR_MSG, e); - } - } - - ArrayList constructDetailedConsentsSearchResult(ResultSet resultSet, int resultSetSize) - throws SQLException { - - ArrayList detailedConsentResources = new ArrayList<>(); - - while (resultSet.next()) { - - Map consentAttributesMap = new HashMap<>(); - ArrayList consentMappingResources = new ArrayList<>(); - ArrayList authorizationResources = new ArrayList<>(); - DetailedConsentResource detailedConsentResource = new DetailedConsentResource(); - - setConsentDataToDetailedConsentInSearchResponse(resultSet, detailedConsentResource); - - // Set consent attributes to map if available - if (resultSet.getString(ConsentMgtDAOConstants.ATT_KEY) != null && - StringUtils.isNotBlank(resultSet.getString(ConsentMgtDAOConstants.ATT_KEY))) { - // fetch attribute keys and values from group_concat - String[] attKeys = resultSet.getString(ConsentMgtDAOConstants.ATT_KEY).split(GROUP_BY_SEPARATOR); - String[] attValues = resultSet.getString(ConsentMgtDAOConstants.ATT_VALUE).split(GROUP_BY_SEPARATOR); - // check if all attribute keys has values - if (attKeys.length == attValues.length) { - for (int index = 0; index < attKeys.length; index++) { - if (!attKeys[index].isEmpty()) { - consentAttributesMap.put(attKeys[index], attValues[index]); - } - } - } - } - // Set authorization data - setAuthorizationDataInResponseForGroupedQuery(authorizationResources, resultSet, - detailedConsentResource.getConsentID()); - // Set consent account mapping data if available - setAccountConsentMappingDataInResponse(consentMappingResources, resultSet); - - detailedConsentResource.setConsentAttributes(consentAttributesMap); - detailedConsentResource.setAuthorizationResources(authorizationResources); - detailedConsentResource.setConsentMappingResources(consentMappingResources); - - detailedConsentResources.add(detailedConsentResource); - - } - return detailedConsentResources; - } - - void setConsentDataToDetailedConsentInSearchResponse(ResultSet resultSet, - DetailedConsentResource detailedConsentResource) - throws SQLException { - - Optional consentId = Arrays.stream(resultSet.getString(ConsentMgtDAOConstants.CONSENT_ID) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - Optional clientId = Arrays.stream(resultSet.getString(ConsentMgtDAOConstants.CLIENT_ID) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - Optional receipt = Arrays.stream(resultSet.getString(ConsentMgtDAOConstants.RECEIPT) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - Optional createdTime = Arrays.stream(resultSet.getString(ConsentMgtDAOConstants.CONSENT_CREATED_TIME) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - Optional consentUpdatedTime = Arrays.stream( - resultSet.getString(ConsentMgtDAOConstants.CONSENT_UPDATED_TIME) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - Optional consentType = Arrays.stream(resultSet.getString(ConsentMgtDAOConstants.CONSENT_TYPE) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - Optional currentStatus = Arrays.stream(resultSet.getString(ConsentMgtDAOConstants.CURRENT_STATUS) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - Optional frequency = Arrays.stream(resultSet.getString(ConsentMgtDAOConstants.CONSENT_FREQUENCY) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - Optional validityTime = Arrays.stream(resultSet.getString(ConsentMgtDAOConstants.VALIDITY_TIME) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - Optional recurringIndicator = Arrays.stream( - resultSet.getString(ConsentMgtDAOConstants.RECURRING_INDICATOR) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - - if (consentId.isPresent() && clientId.isPresent()) { - detailedConsentResource.setConsentID(consentId.get()); - detailedConsentResource.setClientID(clientId.get()); - } else { - throw new SQLException("CLIENT_ID and CONSENT_ID could not be null."); - } - receipt.ifPresent(detailedConsentResource::setReceipt); - consentType.ifPresent(detailedConsentResource::setConsentType); - currentStatus.ifPresent(detailedConsentResource::setCurrentStatus); - createdTime.ifPresent(e -> detailedConsentResource.setCreatedTime(Long.parseLong(e))); - consentUpdatedTime.ifPresent(e -> detailedConsentResource.setUpdatedTime(Long.parseLong(e))); - frequency.ifPresent(e -> detailedConsentResource.setConsentFrequency(Integer.parseInt(e))); - validityTime.ifPresent(e -> detailedConsentResource.setValidityPeriod(Long.parseLong(e))); - recurringIndicator.ifPresent(e -> detailedConsentResource.setRecurringIndicator(Integer.parseInt(e) != 0)); - } - - protected void setAuthorizationDataInResponseForGroupedQuery(ArrayList - authorizationResources, - ResultSet resultSet, String consentId) - throws SQLException { - - //identify duplicate auth data - Set authIdSet = new HashSet<>(); - - // fetch values from group_concat - String[] authIds = resultSet.getString(ConsentMgtDAOConstants.AUTH_ID) != null ? - resultSet.getString(ConsentMgtDAOConstants.AUTH_ID).split(GROUP_BY_SEPARATOR) : null; - String[] authTypes = resultSet.getString(ConsentMgtDAOConstants.AUTH_TYPE) != null ? - resultSet.getString(ConsentMgtDAOConstants.AUTH_TYPE).split(GROUP_BY_SEPARATOR) : null; - String[] authStatues = resultSet.getString(ConsentMgtDAOConstants.AUTH_STATUS) != null ? - resultSet.getString(ConsentMgtDAOConstants.AUTH_STATUS).split(GROUP_BY_SEPARATOR) : null; - String[] updatedTimes = resultSet.getString(ConsentMgtDAOConstants.UPDATED_TIME) != null ? - resultSet.getString(ConsentMgtDAOConstants.UPDATED_TIME).split(GROUP_BY_SEPARATOR) : null; - String[] userIds = resultSet.getString(ConsentMgtDAOConstants.USER_ID) != null ? - resultSet.getString(ConsentMgtDAOConstants.USER_ID).split(GROUP_BY_SEPARATOR) : null; - - for (int index = 0; index < (authIds != null ? authIds.length : 0); index++) { - if (!authIdSet.contains(authIds[index])) { - AuthorizationResource authorizationResource = new AuthorizationResource(); - authIdSet.add(authIds[index]); - authorizationResource.setAuthorizationID(authIds[index]); - authorizationResource.setConsentID(consentId); - if (authTypes != null && authTypes.length > index) { - authorizationResource.setAuthorizationType(authTypes[index]); - } - if (authStatues != null && authStatues.length > index) { - authorizationResource.setAuthorizationStatus(authStatues[index]); - } - if (updatedTimes != null && updatedTimes.length > index) { - authorizationResource.setUpdatedTime(Long.parseLong(updatedTimes[index])); - } - if (userIds != null && userIds.length > index) { - authorizationResource.setUserID(userIds[index]); - } - authorizationResources.add(authorizationResource); - } - } - - } - - protected void setAccountConsentMappingDataInResponse(ArrayList consentMappingResources, - ResultSet resultSet) throws SQLException { - - //identify duplicate mappingIds - Set mappingIdSet = new HashSet<>(); - - // fetch values from group_concat - String[] authIds = resultSet.getString(ConsentMgtDAOConstants.AUTH_MAPPING_ID) != null ? - resultSet.getString(ConsentMgtDAOConstants.AUTH_MAPPING_ID).split(GROUP_BY_SEPARATOR) : null; - String[] mappingIds = resultSet.getString(ConsentMgtDAOConstants.MAPPING_ID) != null ? - resultSet.getString(ConsentMgtDAOConstants.MAPPING_ID).split(GROUP_BY_SEPARATOR) : null; - String[] accountIds = resultSet.getString(ConsentMgtDAOConstants.ACCOUNT_ID) != null ? - resultSet.getString(ConsentMgtDAOConstants.ACCOUNT_ID).split(GROUP_BY_SEPARATOR) : null; - String[] mappingStatues = resultSet.getString(ConsentMgtDAOConstants.MAPPING_STATUS) != null ? - resultSet.getString(ConsentMgtDAOConstants.MAPPING_STATUS).split(GROUP_BY_SEPARATOR) : null; - String[] permissions = resultSet.getString(ConsentMgtDAOConstants.PERMISSION) != null ? - resultSet.getString(ConsentMgtDAOConstants.PERMISSION).split(GROUP_BY_SEPARATOR) : null; - - for (int index = 0; index < (mappingIds != null ? mappingIds.length : 0); index++) { - if (!mappingIdSet.contains(mappingIds[index])) { - ConsentMappingResource consentMappingResource = new ConsentMappingResource(); - if (authIds != null && authIds.length > index) { - consentMappingResource.setAuthorizationID(authIds[index]); - } - consentMappingResource.setMappingID(mappingIds[index]); - if (accountIds != null && accountIds.length > index) { - consentMappingResource.setAccountID(accountIds[index]); - } - if (mappingStatues != null && mappingStatues.length > index) { - consentMappingResource.setMappingStatus(mappingStatues[index]); - } - if (permissions != null && permissions.length > index) { - consentMappingResource.setPermission(permissions[index]); - } - consentMappingResources.add(consentMappingResource); - mappingIdSet.add(mappingIds[index]); - } - } - - } - - void validateAndSetSearchConditions(Map applicableConditionsMap, ArrayList consentIDs, - ArrayList clientIDs, - ArrayList consentTypes, - ArrayList consentStatuses) { - - log.debug("Validate applicable search conditions"); - - if (CollectionUtils.isNotEmpty(consentIDs)) { - applicableConditionsMap.put(COLUMNS_MAP.get(ConsentMgtDAOConstants.CONSENT_IDS), consentIDs); - } - if (CollectionUtils.isNotEmpty(clientIDs)) { - applicableConditionsMap.put(COLUMNS_MAP.get(ConsentMgtDAOConstants.CLIENT_IDS), clientIDs); - } - if (CollectionUtils.isNotEmpty(consentTypes)) { - applicableConditionsMap.put(COLUMNS_MAP.get(ConsentMgtDAOConstants.CONSENT_TYPES), consentTypes); - } - if (CollectionUtils.isNotEmpty(consentStatuses)) { - applicableConditionsMap.put(COLUMNS_MAP.get(ConsentMgtDAOConstants.CONSENT_STATUSES), consentStatuses); - } - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/impl/OracleConsentCoreDAOImpl.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/impl/OracleConsentCoreDAOImpl.java deleted file mode 100644 index 0e98044d..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/impl/OracleConsentCoreDAOImpl.java +++ /dev/null @@ -1,359 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.impl; - -import com.wso2.openbanking.accelerator.consent.mgt.dao.constants.ConsentMgtDAOConstants; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataRetrievalException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentMappingResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.queries.ConsentMgtOracleDBQueries; -import com.wso2.openbanking.accelerator.consent.mgt.dao.utils.ConsentDAOUtils; -import org.apache.commons.collections.CollectionUtils; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.sql.Types; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.HashSet; -import java.util.Map; -import java.util.Optional; -import java.util.Set; - -/** - * DAO implementation for Oracle specific methods. - */ -public class OracleConsentCoreDAOImpl extends ConsentCoreDAOImpl { - - private static Log log = LogFactory.getLog(OracleConsentCoreDAOImpl.class); - private static final String GROUP_BY_SEPARATOR = "\\|\\|"; - static final Map COLUMNS_MAP = new HashMap() { - { - put(ConsentMgtDAOConstants.CONSENT_IDS, "OBC.CONSENT_ID"); - put(ConsentMgtDAOConstants.CLIENT_IDS, "OBC.CLIENT_ID"); - put(ConsentMgtDAOConstants.CONSENT_TYPES, "OBC.CONSENT_TYPE"); - put(ConsentMgtDAOConstants.CONSENT_STATUSES, "OBC.CURRENT_STATUS"); - put(ConsentMgtDAOConstants.USER_IDS, "OCAR.USER_ID"); - } - }; - - public OracleConsentCoreDAOImpl(ConsentMgtOracleDBQueries sqlStatements) { - - super(sqlStatements); - } - - @Override - public ArrayList searchConsents(Connection connection, ArrayList consentIDs, - ArrayList clientIDs, - ArrayList consentTypes, - ArrayList consentStatuses, - ArrayList userIDs, Long fromTime, - Long toTime, Integer limit, Integer offset) - throws OBConsentDataRetrievalException { - - boolean shouldLimit = true; - boolean shouldOffset = true; - int parameterIndex = 0; - Map applicableConditionsMap = new HashMap<>(); - - validateAndSetSearchConditions(applicableConditionsMap, consentIDs, clientIDs, consentTypes, consentStatuses); - - if (limit == null) { - shouldLimit = false; - } - if (offset == null) { - shouldOffset = false; - } - - // logic to set the prepared statement - String constructedConditions = - ConsentDAOUtils.constructConsentSearchPreparedStatement(applicableConditionsMap); - - String userIDFilterCondition = ""; - Map userIdMap = new HashMap<>(); - if (CollectionUtils.isNotEmpty(userIDs)) { - userIdMap.put(COLUMNS_MAP.get(ConsentMgtDAOConstants.USER_IDS), userIDs); - userIDFilterCondition = ConsentDAOUtils.constructUserIdListFilterCondition(userIdMap); - } - String searchConsentsPreparedStatement = - sqlStatements.getSearchConsentsPreparedStatement(constructedConditions, shouldLimit, - shouldOffset, userIDFilterCondition); - - try (PreparedStatement searchConsentsPreparedStmt = - connection.prepareStatement(searchConsentsPreparedStatement, ResultSet.TYPE_SCROLL_INSENSITIVE, - ResultSet.CONCUR_UPDATABLE)) { - - //determine order of user Ids to set - if (CollectionUtils.isNotEmpty(userIDs)) { - Map orderedUserIdsMap = ConsentDAOUtils - .determineOrderOfParamsToSet(userIDFilterCondition, userIdMap, COLUMNS_MAP); - parameterIndex = setDynamicConsentSearchParameters(searchConsentsPreparedStmt, orderedUserIdsMap, - ++parameterIndex); - parameterIndex = parameterIndex - 1; - } - - /* Since we don't know the order of the set condition clauses, have to determine the order of them to set - the actual values to the prepared statement */ - Map orderedParamsMap = ConsentDAOUtils - .determineOrderOfParamsToSet(constructedConditions, applicableConditionsMap, COLUMNS_MAP); - parameterIndex = setDynamicConsentSearchParameters(searchConsentsPreparedStmt, orderedParamsMap, - ++parameterIndex); - parameterIndex = parameterIndex - 1; - - log.debug("Setting parameters to prepared statement to search consents"); - - if (fromTime != null) { - searchConsentsPreparedStmt.setLong(++parameterIndex, fromTime); - } else { - searchConsentsPreparedStmt.setNull(++parameterIndex, Types.BIGINT); - } - - if (toTime != null) { - searchConsentsPreparedStmt.setLong(++parameterIndex, toTime); - } else { - searchConsentsPreparedStmt.setNull(++parameterIndex, Types.BIGINT); - } - - if (offset != null && limit != null) { - searchConsentsPreparedStmt.setInt(++parameterIndex, offset); - } - if (limit != null) { - searchConsentsPreparedStmt.setInt(++parameterIndex, limit); - } - - ArrayList detailedConsentResources = new ArrayList<>(); - - try (ResultSet resultSet = searchConsentsPreparedStmt.executeQuery()) { - if (resultSet.isBeforeFirst()) { - int resultSetSize = getResultSetSize(resultSet); - detailedConsentResources = constructDetailedConsentsSearchResult(resultSet, resultSetSize); - } - return detailedConsentResources; - } catch (SQLException e) { - log.error("Error occurred while searching detailed consent resources", e); - throw new OBConsentDataRetrievalException("Error occurred while searching detailed " + - "consent resources", e); - } - } catch (SQLException e) { - log.error(ConsentMgtDAOConstants.CONSENT_SEARCH_ERROR_MSG, e); - throw new OBConsentDataRetrievalException(ConsentMgtDAOConstants.CONSENT_SEARCH_ERROR_MSG, e); - } - } - - ArrayList constructDetailedConsentsSearchResult(ResultSet resultSet, int resultSetSize) - throws SQLException { - - ArrayList detailedConsentResources = new ArrayList<>(); - - while (resultSet.next()) { - - Map consentAttributesMap = new HashMap<>(); - ArrayList consentMappingResources = new ArrayList<>(); - ArrayList authorizationResources = new ArrayList<>(); - DetailedConsentResource detailedConsentResource = new DetailedConsentResource(); - - setConsentDataToDetailedConsentInSearchResponse(resultSet, detailedConsentResource); - - // Set consent attributes to map if available - if (resultSet.getString(ConsentMgtDAOConstants.ATT_KEY) != null && - StringUtils.isNotBlank(resultSet.getString(ConsentMgtDAOConstants.ATT_KEY) - .replaceAll(GROUP_BY_SEPARATOR, ""))) { - // fetch attribute keys and values from group_concat - String[] attKeys = resultSet.getString(ConsentMgtDAOConstants.ATT_KEY).split(GROUP_BY_SEPARATOR); - String[] attValues = resultSet.getString(ConsentMgtDAOConstants.ATT_VALUE).split(GROUP_BY_SEPARATOR); - // check if all attribute keys has values - if (attKeys.length == attValues.length) { - for (int index = 0; index < attKeys.length; index++) { - if (!attKeys[index].isEmpty()) { - consentAttributesMap.put(attKeys[index], attValues[index]); - } - } - } - } - // Set authorization data - setAuthorizationDataInResponseForGroupedQuery(authorizationResources, resultSet, - detailedConsentResource.getConsentID()); - // Set consent account mapping data if available - setAccountConsentMappingDataInResponse(consentMappingResources, resultSet); - - detailedConsentResource.setConsentAttributes(consentAttributesMap); - detailedConsentResource.setAuthorizationResources(authorizationResources); - detailedConsentResource.setConsentMappingResources(consentMappingResources); - - detailedConsentResources.add(detailedConsentResource); - - } - return detailedConsentResources; - } - - void setConsentDataToDetailedConsentInSearchResponse(ResultSet resultSet, - DetailedConsentResource detailedConsentResource) - throws SQLException { - - Optional consentId = Arrays.stream(resultSet.getString(ConsentMgtDAOConstants.CONSENT_ID) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - Optional clientId = Arrays.stream(resultSet.getString(ConsentMgtDAOConstants.CLIENT_ID) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - Optional receipt = Arrays.stream(resultSet.getString(ConsentMgtDAOConstants.RECEIPT) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - Optional createdTime = Arrays.stream(resultSet.getString(ConsentMgtDAOConstants.CONSENT_CREATED_TIME) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - Optional consentUpdatedTime = Arrays.stream( - resultSet.getString(ConsentMgtDAOConstants.CONSENT_UPDATED_TIME) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - Optional consentType = Arrays.stream(resultSet.getString(ConsentMgtDAOConstants.CONSENT_TYPE) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - Optional currentStatus = Arrays.stream(resultSet.getString(ConsentMgtDAOConstants.CURRENT_STATUS) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - Optional frequency = Arrays.stream(resultSet.getString(ConsentMgtDAOConstants.CONSENT_FREQUENCY) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - Optional validityTime = Arrays.stream(resultSet.getString(ConsentMgtDAOConstants.VALIDITY_TIME) - .split(GROUP_BY_SEPARATOR)).distinct().findFirst(); - Optional recurringIndicator = - Optional.of(resultSet.getBoolean(ConsentMgtDAOConstants.RECURRING_INDICATOR)); - - if (consentId.isPresent() && clientId.isPresent()) { - detailedConsentResource.setConsentID(consentId.get()); - detailedConsentResource.setClientID(clientId.get()); - } else { - throw new SQLException("CLIENT_ID and CONSENT_ID could not be null."); - } - receipt.ifPresent(detailedConsentResource::setReceipt); - consentType.ifPresent(detailedConsentResource::setConsentType); - currentStatus.ifPresent(detailedConsentResource::setCurrentStatus); - createdTime.ifPresent(e -> detailedConsentResource.setCreatedTime(Long.parseLong(e))); - consentUpdatedTime.ifPresent(e -> detailedConsentResource.setUpdatedTime(Long.parseLong(e))); - frequency.ifPresent(e -> detailedConsentResource.setConsentFrequency(Integer.parseInt(e))); - validityTime.ifPresent(e -> detailedConsentResource.setValidityPeriod(Long.parseLong(e))); - recurringIndicator.ifPresent(detailedConsentResource::setRecurringIndicator); - } - - protected void setAuthorizationDataInResponseForGroupedQuery(ArrayList - authorizationResources, - ResultSet resultSet, String consentId) - throws SQLException { - - //identify duplicate auth data - Set authIdSet = new HashSet<>(); - - // fetch values from group_concat - String[] authIds = resultSet.getString(ConsentMgtDAOConstants.AUTH_ID) != null ? - resultSet.getString(ConsentMgtDAOConstants.AUTH_ID).split(GROUP_BY_SEPARATOR) : null; - String[] authTypes = resultSet.getString(ConsentMgtDAOConstants.AUTH_TYPE) != null ? - resultSet.getString(ConsentMgtDAOConstants.AUTH_TYPE).split(GROUP_BY_SEPARATOR) : null; - String[] authStatues = resultSet.getString(ConsentMgtDAOConstants.AUTH_STATUS) != null ? - resultSet.getString(ConsentMgtDAOConstants.AUTH_STATUS).split(GROUP_BY_SEPARATOR) : null; - String[] updatedTimes = resultSet.getString(ConsentMgtDAOConstants.UPDATED_TIME) != null ? - resultSet.getString(ConsentMgtDAOConstants.UPDATED_TIME).split(GROUP_BY_SEPARATOR) : null; - String[] userIds = resultSet.getString(ConsentMgtDAOConstants.USER_ID) != null ? - resultSet.getString(ConsentMgtDAOConstants.USER_ID).split(GROUP_BY_SEPARATOR) : null; - - for (int index = 0; index < (authIds != null ? authIds.length : 0); index++) { - if (!authIdSet.contains(authIds[index])) { - AuthorizationResource authorizationResource = new AuthorizationResource(); - authIdSet.add(authIds[index]); - authorizationResource.setAuthorizationID(authIds[index]); - authorizationResource.setConsentID(consentId); - if (authTypes != null && authTypes.length > index) { - authorizationResource.setAuthorizationType(authTypes[index]); - } - if (authStatues != null && authStatues.length > index) { - authorizationResource.setAuthorizationStatus(authStatues[index]); - } - if (updatedTimes != null && updatedTimes.length > index) { - authorizationResource.setUpdatedTime(Long.parseLong(updatedTimes[index])); - } - if (userIds != null && userIds.length > index) { - authorizationResource.setUserID(userIds[index]); - } - authorizationResources.add(authorizationResource); - } - } - - } - - protected void setAccountConsentMappingDataInResponse(ArrayList consentMappingResources, - ResultSet resultSet) throws SQLException { - - //identify duplicate mappingIds - Set mappingIdSet = new HashSet<>(); - - // fetch values from group_concat - String[] authIds = resultSet.getString(ConsentMgtDAOConstants.AUTH_MAPPING_ID) != null ? - resultSet.getString(ConsentMgtDAOConstants.AUTH_MAPPING_ID).split(GROUP_BY_SEPARATOR) : null; - String[] mappingIds = resultSet.getString(ConsentMgtDAOConstants.MAPPING_ID) != null ? - resultSet.getString(ConsentMgtDAOConstants.MAPPING_ID).split(GROUP_BY_SEPARATOR) : null; - String[] accountIds = resultSet.getString(ConsentMgtDAOConstants.ACCOUNT_ID) != null ? - resultSet.getString(ConsentMgtDAOConstants.ACCOUNT_ID).split(GROUP_BY_SEPARATOR) : null; - String[] mappingStatues = resultSet.getString(ConsentMgtDAOConstants.MAPPING_STATUS) != null ? - resultSet.getString(ConsentMgtDAOConstants.MAPPING_STATUS).split(GROUP_BY_SEPARATOR) : null; - String[] permissions = resultSet.getString(ConsentMgtDAOConstants.PERMISSION) != null ? - resultSet.getString(ConsentMgtDAOConstants.PERMISSION).split(GROUP_BY_SEPARATOR) : null; - - for (int index = 0; index < (mappingIds != null ? mappingIds.length : 0); index++) { - if (!mappingIdSet.contains(mappingIds[index])) { - ConsentMappingResource consentMappingResource = new ConsentMappingResource(); - if (authIds != null && authIds.length > index) { - consentMappingResource.setAuthorizationID(authIds[index]); - } - consentMappingResource.setMappingID(mappingIds[index]); - if (accountIds != null && accountIds.length > index) { - consentMappingResource.setAccountID(accountIds[index]); - } - if (mappingStatues != null && mappingStatues.length > index) { - consentMappingResource.setMappingStatus(mappingStatues[index]); - } - if (permissions != null && permissions.length > index) { - consentMappingResource.setPermission(permissions[index]); - } - consentMappingResources.add(consentMappingResource); - mappingIdSet.add(mappingIds[index]); - } - } - - } - - void validateAndSetSearchConditions(Map applicableConditionsMap, ArrayList consentIDs, - ArrayList clientIDs, - ArrayList consentTypes, - ArrayList consentStatuses) { - - log.debug("Validate applicable search conditions"); - - if (CollectionUtils.isNotEmpty(consentIDs)) { - applicableConditionsMap.put(COLUMNS_MAP.get(ConsentMgtDAOConstants.CONSENT_IDS), consentIDs); - } - if (CollectionUtils.isNotEmpty(clientIDs)) { - applicableConditionsMap.put(COLUMNS_MAP.get(ConsentMgtDAOConstants.CLIENT_IDS), clientIDs); - } - if (CollectionUtils.isNotEmpty(consentTypes)) { - applicableConditionsMap.put(COLUMNS_MAP.get(ConsentMgtDAOConstants.CONSENT_TYPES), consentTypes); - } - if (CollectionUtils.isNotEmpty(consentStatuses)) { - applicableConditionsMap.put(COLUMNS_MAP.get(ConsentMgtDAOConstants.CONSENT_STATUSES), consentStatuses); - } - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/AuthorizationResource.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/AuthorizationResource.java deleted file mode 100644 index 2641460f..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/AuthorizationResource.java +++ /dev/null @@ -1,107 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.models; - -import com.wso2.openbanking.accelerator.common.util.Generated; - -/** - * Model for the Authorization resource. - */ -public class AuthorizationResource { - - private String authorizationID; - private String consentID; - private String userID; - private String authorizationStatus; - private String authorizationType; - private long updatedTime; - - public AuthorizationResource() { - - } - - @Generated(message = "Excluding constructor because setter methods are explicitly called") - public AuthorizationResource(String consentID, String userID, String authorizationStatus, - String authorizationType, long updatedTime) { - this.consentID = consentID; - this.userID = userID; - this.authorizationStatus = authorizationStatus; - this.authorizationType = authorizationType; - this.updatedTime = updatedTime; - - } - public String getAuthorizationID() { - - return authorizationID; - } - - public String getAuthorizationType() { - - return authorizationType; - } - - public void setAuthorizationType(String authorizationType) { - - this.authorizationType = authorizationType; - } - - public void setAuthorizationID(String authorizationID) { - - this.authorizationID = authorizationID; - } - - public String getConsentID() { - - return consentID; - } - - public void setConsentID(String consentID) { - - this.consentID = consentID; - } - - public String getUserID() { - - return userID; - } - - public void setUserID(String userID) { - - this.userID = userID; - } - - public String getAuthorizationStatus() { - - return authorizationStatus; - } - - public void setAuthorizationStatus(String authorizationStatus) { - - this.authorizationStatus = authorizationStatus; - } - - public long getUpdatedTime() { - - return updatedTime; - } - - public void setUpdatedTime(long updatedTime) { - - this.updatedTime = updatedTime; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentAttributes.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentAttributes.java deleted file mode 100644 index 8a1c6783..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentAttributes.java +++ /dev/null @@ -1,61 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.models; - -import com.wso2.openbanking.accelerator.common.util.Generated; - -import java.util.Map; - -/** - * Model for consent attributes. - */ -public class ConsentAttributes { - - private String consentID; - private Map consentAttributes; - - public ConsentAttributes(){ - - } - - @Generated(message = "Excluding constructor because setter methods are explicitly called") - public ConsentAttributes(String consentID, Map consentAttributes) { - this.consentID = consentID; - this.consentAttributes = consentAttributes; - } - - public String getConsentID() { - - return consentID; - } - - public void setConsentID(String consentID) { - - this.consentID = consentID; - } - - public Map getConsentAttributes() { - - return consentAttributes; - } - - public void setConsentAttributes(Map consentAttributes) { - - this.consentAttributes = consentAttributes; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentFile.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentFile.java deleted file mode 100644 index 44362f3c..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentFile.java +++ /dev/null @@ -1,59 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.models; - -import com.wso2.openbanking.accelerator.common.util.Generated; - -/** - * Model for the consent file. - */ -public class ConsentFile { - - private String consentID; - private String consentFile; - - public ConsentFile() { - - } - - @Generated(message = "Excluding constructor because setter methods are explicitly called") - public ConsentFile(String consentID, String consentFile) { - this.consentID = consentID; - this.consentFile = consentFile; - } - - public String getConsentID() { - - return consentID; - } - - public void setConsentID(String consentID) { - - this.consentID = consentID; - } - - public String getConsentFile() { - - return consentFile; - } - - public void setConsentFile(String consentFile) { - - this.consentFile = consentFile; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentHistoryResource.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentHistoryResource.java deleted file mode 100644 index 8b471d2b..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentHistoryResource.java +++ /dev/null @@ -1,97 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.models; - -import com.wso2.openbanking.accelerator.common.util.Generated; - -import java.util.HashMap; -import java.util.Map; - -/** - * Model for the consent history resource. - */ -public class ConsentHistoryResource { - - private String historyID; - private String consentID; - private long timestamp; - private String reason; - private DetailedConsentResource detailedConsentResource; - private Map changedAttributesJsonDataMap; - - public ConsentHistoryResource() { - - } - - @Generated(message = "Excluding constructor because setter methods are explicitly called") - public ConsentHistoryResource(String consentID, String historyID) { - - this.consentID = consentID; - this.historyID = historyID; - this.changedAttributesJsonDataMap = new HashMap<>(); - } - - public long getTimestamp() { - return timestamp; - } - - public void setTimestamp(long timestamp) { - this.timestamp = timestamp; - } - - public String getReason() { - return reason; - } - - public void setReason(String reason) { - this.reason = reason; - } - - public DetailedConsentResource getDetailedConsentResource() { - return detailedConsentResource; - } - - public void setDetailedConsentResource(DetailedConsentResource detailedConsentResource) { - this.detailedConsentResource = detailedConsentResource; - } - - public Map getChangedAttributesJsonDataMap() { - return changedAttributesJsonDataMap; - } - - public void setChangedAttributesJsonDataMap(Map changedAttributesJsonDataMap) { - this.changedAttributesJsonDataMap = changedAttributesJsonDataMap; - } - - public String getConsentID() { - return consentID; - } - - public void setConsentID(String consentID) { - this.consentID = consentID; - } - - public String getHistoryID() { - return historyID; - } - - public void setHistoryID(String historyID) { - this.historyID = historyID; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentMappingResource.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentMappingResource.java deleted file mode 100644 index 702435c8..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentMappingResource.java +++ /dev/null @@ -1,95 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.models; - -import com.wso2.openbanking.accelerator.common.util.Generated; - -/** - * Model for consent mapping resource. - */ -public class ConsentMappingResource { - - private String mappingID; - private String authorizationID; - private String accountID; - private String permission; - private String mappingStatus; - - public ConsentMappingResource() { - - } - - @Generated(message = "Excluding constructor because setter methods are explicitly called") - public ConsentMappingResource(String authorizationID, String accountID, String permission, - String mappingStatus) { - this.authorizationID = authorizationID; - this.accountID = accountID; - this.permission = permission; - this.mappingStatus = mappingStatus; - } - - public String getMappingID() { - - return mappingID; - } - - public void setMappingID(String mappingID) { - - this.mappingID = mappingID; - } - - public String getAuthorizationID() { - - return authorizationID; - } - - public void setAuthorizationID(String authorizationID) { - - this.authorizationID = authorizationID; - } - - public String getAccountID() { - - return accountID; - } - - public void setAccountID(String accountID) { - - this.accountID = accountID; - } - - public String getPermission() { - - return permission; - } - - public void setPermission(String permission) { - - this.permission = permission; - } - - public String getMappingStatus() { - - return mappingStatus; - } - - public void setMappingStatus(String mappingStatus) { - - this.mappingStatus = mappingStatus; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentResource.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentResource.java deleted file mode 100644 index c1bc3309..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentResource.java +++ /dev/null @@ -1,177 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.models; - -import com.wso2.openbanking.accelerator.common.util.Generated; - -import java.util.Map; - -/** - * Model for the consent resource. - */ -public class ConsentResource { - - private String consentID; - private String clientID; - private String receipt; - private String consentType; - private int consentFrequency; - private long validityPeriod; - private boolean recurringIndicator; - private String currentStatus; - private long createdTime; - private long updatedTime; - - public ConsentResource() { - } - - public ConsentResource(String clientID, String receipt, String consentType, String currentStatus) { - this.clientID = clientID; - this.receipt = receipt; - this.consentType = consentType; - this.currentStatus = currentStatus; - } - - @Generated(message = "Excluding constructor because setter methods are explicitly called") - public ConsentResource(String consentID, String clientID, String receipt, String consentType, - int consentFrequency, long validityPeriod, boolean recurringIndicator, - String currentStatus, long createdTime, long updatedTime) { - this.consentID = consentID; - this.clientID = clientID; - this.receipt = receipt; - this.consentType = consentType; - this.consentFrequency = consentFrequency; - this.validityPeriod = validityPeriod; - this.recurringIndicator = recurringIndicator; - this.currentStatus = currentStatus; - this.createdTime = createdTime; - this.updatedTime = updatedTime; - } - - private Map consentAttributes; - - public long getUpdatedTime() { - - return updatedTime; - } - - public void setUpdatedTime(long updatedTime) { - - this.updatedTime = updatedTime; - } - - public Map getConsentAttributes() { - - return consentAttributes; - } - - public void setConsentAttributes(Map consentAttributes) { - - this.consentAttributes = consentAttributes; - } - - public String getConsentID() { - - return consentID; - } - - public void setConsentID(String consentID) { - - this.consentID = consentID; - } - - public String getClientID() { - - return clientID; - } - - public void setClientID(String clientID) { - - this.clientID = clientID; - } - - public String getReceipt() { - - return receipt; - } - - public void setReceipt(String receipt) { - - this.receipt = receipt; - } - - public String getConsentType() { - - return consentType; - } - - public void setConsentType(String consentType) { - - this.consentType = consentType; - } - - public int getConsentFrequency() { - - return consentFrequency; - } - - public void setConsentFrequency(int consentFrequency) { - - this.consentFrequency = consentFrequency; - } - - public long getValidityPeriod() { - - return validityPeriod; - } - - public void setValidityPeriod(long validityPeriod) { - - this.validityPeriod = validityPeriod; - } - - public boolean isRecurringIndicator() { - - return recurringIndicator; - } - - public void setRecurringIndicator(boolean recurringIndicator) { - - this.recurringIndicator = recurringIndicator; - } - - public String getCurrentStatus() { - - return currentStatus; - } - - public void setCurrentStatus(String currentStatus) { - - this.currentStatus = currentStatus; - } - - public long getCreatedTime() { - - return createdTime; - } - - public void setCreatedTime(long createdTime) { - - this.createdTime = createdTime; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentStatusAuditRecord.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentStatusAuditRecord.java deleted file mode 100644 index c1f01240..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/ConsentStatusAuditRecord.java +++ /dev/null @@ -1,119 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.models; - -import com.wso2.openbanking.accelerator.common.util.Generated; - -/** - * Model for consent status audit record resource. - */ -public class ConsentStatusAuditRecord { - - private String statusAuditID; - private String consentID; - private String currentStatus; - private long actionTime; - private String reason; - private String actionBy; - private String previousStatus; - - public ConsentStatusAuditRecord() { - - } - - @Generated(message = "Excluding constructor because setter methods are explicitly called") - public ConsentStatusAuditRecord(String consentID, String currentStatus, long actionTime, - String reason, String actionBy, String previousStatus) { - this.consentID = consentID; - this.currentStatus = currentStatus; - this.actionTime = actionTime; - this.reason = reason; - this.actionBy = actionBy; - this.previousStatus = previousStatus; - } - - public String getStatusAuditID() { - - return statusAuditID; - } - - public void setStatusAuditID(String statusAuditID) { - - this.statusAuditID = statusAuditID; - } - - public String getConsentID() { - - return consentID; - } - - public void setConsentID(String consentID) { - - this.consentID = consentID; - } - - public String getCurrentStatus() { - - return currentStatus; - } - - public void setCurrentStatus(String currentStatus) { - - this.currentStatus = currentStatus; - } - - public long getActionTime() { - - return actionTime; - } - - public void setActionTime(long actionTime) { - - this.actionTime = actionTime; - } - - public String getReason() { - - return reason; - } - - public void setReason(String reason) { - - this.reason = reason; - } - - public String getActionBy() { - - return actionBy; - } - - public void setActionBy(String actionBy) { - - this.actionBy = actionBy; - } - - public String getPreviousStatus() { - - return previousStatus; - } - - public void setPreviousStatus(String previousStatus) { - - this.previousStatus = previousStatus; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/DetailedConsentResource.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/DetailedConsentResource.java deleted file mode 100644 index e99239e4..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/models/DetailedConsentResource.java +++ /dev/null @@ -1,201 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.models; - -import com.wso2.openbanking.accelerator.common.util.Generated; - -import java.util.ArrayList; -import java.util.Map; - -/** - * Model for Detailed Consent Resource. - */ -public class DetailedConsentResource { - - private String consentID; - private String clientID; - private String receipt; - private String consentType; - private String currentStatus; - private int consentFrequency; - private long validityPeriod; - private long createdTime; - private long updatedTime; - private boolean recurringIndicator; - private Map consentAttributes; - private ArrayList authorizationResources; - private ArrayList consentMappingResources; - - public DetailedConsentResource() { - - } - - @Generated(message = "Excluding constructor because setter methods are explicitly called") - public DetailedConsentResource(String consentID, String clientID, String receipt, String consentType, - String currentStatus, int consentFrequency, long validityPeriod, long createdTime, - long updatedTime, boolean recurringIndicator, - Map consentAttributes, - ArrayList authorizationResources, - ArrayList consentMappingResources) { - this.consentID = consentID; - this.clientID = clientID; - this.receipt = receipt; - this.consentType = consentType; - this.currentStatus = currentStatus; - this.consentFrequency = consentFrequency; - this.validityPeriod = validityPeriod; - this.createdTime = createdTime; - this.updatedTime = updatedTime; - this.recurringIndicator = recurringIndicator; - this.consentAttributes = consentAttributes; - this.authorizationResources = authorizationResources; - this.consentMappingResources = consentMappingResources; - - } - - public long getUpdatedTime() { - - return updatedTime; - } - - public void setUpdatedTime(long updatedTime) { - - this.updatedTime = updatedTime; - } - - public String getConsentID() { - - return consentID; - } - - public void setConsentID(String consentID) { - - this.consentID = consentID; - } - - public String getClientID() { - - return clientID; - } - - public void setClientID(String clientID) { - - this.clientID = clientID; - } - - public String getReceipt() { - - return receipt; - } - - public void setReceipt(String receipt) { - - this.receipt = receipt; - } - - public String getConsentType() { - - return consentType; - } - - public void setConsentType(String consentType) { - - this.consentType = consentType; - } - - public int getConsentFrequency() { - - return consentFrequency; - } - - public void setConsentFrequency(int consentFrequency) { - - this.consentFrequency = consentFrequency; - } - - public long getValidityPeriod() { - - return validityPeriod; - } - - public void setValidityPeriod(long validityPeriod) { - - this.validityPeriod = validityPeriod; - } - - public boolean isRecurringIndicator() { - - return recurringIndicator; - } - - public void setRecurringIndicator(boolean recurringIndicator) { - - this.recurringIndicator = recurringIndicator; - } - - public String getCurrentStatus() { - - return currentStatus; - } - - public void setCurrentStatus(String currentStatus) { - - this.currentStatus = currentStatus; - } - - public long getCreatedTime() { - - return createdTime; - } - - public void setCreatedTime(long createdTime) { - - this.createdTime = createdTime; - } - - public Map getConsentAttributes() { - - return consentAttributes; - } - - public void setConsentAttributes(Map consentAttributes) { - - this.consentAttributes = consentAttributes; - } - - public ArrayList getAuthorizationResources() { - - return authorizationResources; - } - - public void setAuthorizationResources(ArrayList authorizationResources) { - - this.authorizationResources = authorizationResources; - } - - public ArrayList getConsentMappingResources() { - - return consentMappingResources; - } - - public void setConsentMappingResources(ArrayList consentMappingResources) { - - this.consentMappingResources = consentMappingResources; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/persistence/ConsentStoreInitializer.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/persistence/ConsentStoreInitializer.java deleted file mode 100644 index bff6682c..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/persistence/ConsentStoreInitializer.java +++ /dev/null @@ -1,118 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.persistence; - -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.persistence.JDBCPersistenceManager; -import com.wso2.openbanking.accelerator.common.persistence.JDBCRetentionDataPersistenceManager; -import com.wso2.openbanking.accelerator.consent.mgt.dao.ConsentCoreDAO; -import com.wso2.openbanking.accelerator.consent.mgt.dao.impl.ConsentCoreDAOImpl; -import com.wso2.openbanking.accelerator.consent.mgt.dao.impl.MssqlConsentCoreDAOImpl; -import com.wso2.openbanking.accelerator.consent.mgt.dao.impl.OracleConsentCoreDAOImpl; -import com.wso2.openbanking.accelerator.consent.mgt.dao.queries.ConsentMgtCommonDBQueries; -import com.wso2.openbanking.accelerator.consent.mgt.dao.queries.ConsentMgtMssqlDBQueries; -import com.wso2.openbanking.accelerator.consent.mgt.dao.queries.ConsentMgtOracleDBQueries; -import com.wso2.openbanking.accelerator.consent.mgt.dao.queries.ConsentMgtPostgresDBQueries; - -import java.sql.Connection; -import java.sql.SQLException; - -/** - * This class handles consent DAO layer initiation with the relevant SQL statements per database types. - */ -public class ConsentStoreInitializer { - - private static final String MYSQL = "MySQL"; - private static final String H2 = "H2"; - private static final String MICROSOFT = "Microsoft"; - private static final String MS_SQL = "MSSQL"; - private static final String POSTGRE = "PostgreSQL"; - private static final String ORACLE = "Oracle"; - private static ConsentCoreDAO consentCoreDAO = null; - private static ConsentCoreDAO consentRetentionDAO = null; - - /** - * Return the DAO implementation initialized for the relevant database type. - * - * @return the dao implementation - * @throws ConsentManagementException thrown if an error occurs when getting the database connection - */ - public static synchronized ConsentCoreDAO getInitializedConsentCoreDAOImpl() throws ConsentManagementException { - - if (consentCoreDAO == null) { - consentCoreDAO = getDaoInstance(PersistenceManager.CONSENT_PERSISTENCE_MANAGER); - } - return consentCoreDAO; - } - - /** - * Return the DAO implementation initialized for the relevant database type. - * - * @return the dao implementation - * @throws ConsentManagementException thrown if an error occurs when getting the database connection - */ - public static synchronized ConsentCoreDAO getInitializedConsentRetentionDAOImpl() - throws ConsentManagementException { - - if (consentRetentionDAO == null) { - consentRetentionDAO = getDaoInstance(PersistenceManager.RETENTION_PERSISTENCE_MANAGER); - } - return consentRetentionDAO; - } - - private static ConsentCoreDAO getDaoInstance(PersistenceManager persistenceManager) - throws ConsentManagementException { - - Connection connection; - try { - if (persistenceManager.equals(PersistenceManager.CONSENT_PERSISTENCE_MANAGER)) { - connection = JDBCPersistenceManager.getInstance().getDBConnection(); - } else { - connection = JDBCRetentionDataPersistenceManager.getInstance().getDBConnection(); - } - String driverName = connection.getMetaData().getDriverName(); - - ConsentCoreDAO dao; - if (driverName.contains(MYSQL)) { - dao = new ConsentCoreDAOImpl(new ConsentMgtCommonDBQueries()); - } else if (driverName.contains(H2)) { - dao = new ConsentCoreDAOImpl(new ConsentMgtCommonDBQueries()); - } else if (driverName.contains(MS_SQL) || driverName.contains(MICROSOFT)) { - dao = new MssqlConsentCoreDAOImpl(new ConsentMgtMssqlDBQueries()); - } else if (driverName.contains(POSTGRE)) { - dao = new ConsentCoreDAOImpl(new ConsentMgtPostgresDBQueries()); - } else if (driverName.contains(ORACLE)) { - dao = new OracleConsentCoreDAOImpl(new ConsentMgtOracleDBQueries()); - } else { - throw new ConsentManagementException("Unhandled DB driver: " + driverName + " detected : "); - } - return dao; - } catch (SQLException e) { - throw new ConsentManagementException("Error while getting the database connection : ", e); - } - } - - /** - * PersistenceManager types enum. - */ - public enum PersistenceManager { - CONSENT_PERSISTENCE_MANAGER, - RETENTION_PERSISTENCE_MANAGER - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/queries/ConsentMgtCommonDBQueries.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/queries/ConsentMgtCommonDBQueries.java deleted file mode 100644 index 48a27b56..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/queries/ConsentMgtCommonDBQueries.java +++ /dev/null @@ -1,474 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.queries; - -import com.wso2.openbanking.accelerator.consent.mgt.dao.constants.ConsentMgtDAOConstants; -import org.apache.commons.lang.StringUtils; - -/** - * The common database queries used by the consent management DAO layer. - */ -public class ConsentMgtCommonDBQueries { - - public String getStoreConsentPreparedStatement() { - - return "INSERT INTO OB_CONSENT (CONSENT_ID, RECEIPT, CREATED_TIME, UPDATED_TIME, CLIENT_ID, CONSENT_TYPE, " + - "CURRENT_STATUS, CONSENT_FREQUENCY, VALIDITY_TIME, RECURRING_INDICATOR) VALUES (?, ?, ?, ?, ?, ?, ?, " + - "?, ?, ?)"; - } - - public String getStoreAuthorizationPreparedStatement() { - - return "INSERT INTO OB_CONSENT_AUTH_RESOURCE (AUTH_ID, CONSENT_ID, AUTH_TYPE, USER_ID, AUTH_STATUS, " + - "UPDATED_TIME) VALUES (?, ?, ?, ?, ?, ?)"; - } - - public String getStoreConsentMappingPreparedStatement() { - - return "INSERT INTO OB_CONSENT_MAPPING (MAPPING_ID, AUTH_ID, ACCOUNT_ID, PERMISSION, MAPPING_STATUS) VALUES " + - "(?, ?, ?, ?, ?)"; - } - - public String getStoreConsentStatusAuditRecordPreparedStatement() { - - return "INSERT INTO OB_CONSENT_STATUS_AUDIT (STATUS_AUDIT_ID, CONSENT_ID, CURRENT_STATUS, ACTION_TIME, " + - "REASON, ACTION_BY, PREVIOUS_STATUS) VALUES (?, ?, ?, ?, ?, ?, ?)"; - } - - public String getStoreConsentAttributesPreparedStatement() { - - return "INSERT INTO OB_CONSENT_ATTRIBUTE (CONSENT_ID, ATT_KEY, ATT_VALUE) VALUES (?, ?, ?)"; - } - - public String getStoreConsentFilePreparedStatement() { - - return "INSERT INTO OB_CONSENT_FILE (CONSENT_ID, CONSENT_FILE) VALUES (?, ?)"; - } - - public String getUpdateConsentStatusPreparedStatement() { - - return "UPDATE OB_CONSENT SET CURRENT_STATUS = ?, UPDATED_TIME = ? WHERE CONSENT_ID = ?"; - } - - public String getUpdateConsentMappingStatusPreparedStatement() { - - return "UPDATE OB_CONSENT_MAPPING SET MAPPING_STATUS = ? WHERE MAPPING_ID = ?"; - } - - public String getUpdateConsentMappingPermissionPreparedStatement() { - - return "UPDATE OB_CONSENT_MAPPING SET PERMISSION = ? WHERE MAPPING_ID = ?"; - } - - public String getUpdateAuthorizationStatusPreparedStatement() { - - return "UPDATE OB_CONSENT_AUTH_RESOURCE SET AUTH_STATUS = ?, UPDATED_TIME = ? WHERE AUTH_ID = ?"; - } - - public String getUpdateAuthorizationUserPreparedStatement() { - - return "UPDATE OB_CONSENT_AUTH_RESOURCE SET USER_ID = ?, UPDATED_TIME = ? WHERE AUTH_ID = ?"; - } - - public String getGetConsentFileResourcePreparedStatement(boolean fetchFromRetentionTables) { - - // table prefix is to fetch from the consent retention data (purged data) tables. (if enabled) - String tablePrefix = ""; - if (fetchFromRetentionTables) { - tablePrefix = ConsentMgtDAOConstants.RETENTION_TABLE_NAME_PREFIX; - } - return "SELECT * FROM " + tablePrefix + "OB_CONSENT_FILE WHERE CONSENT_ID = ?"; - } - - public String getGetConsentAttributesPreparedStatement() { - - return "SELECT * FROM OB_CONSENT_ATTRIBUTE WHERE CONSENT_ID = ?"; - } - - public String getGetConsentPreparedStatement() { - - return "SELECT * FROM OB_CONSENT WHERE CONSENT_ID = ?"; - } - - public String getGetDetailedConsentPreparedStatement(boolean fetchFromRetentionTables) { - - // table prefix is to fetch from the consent retention data (purged data) tables. (if enabled) - String tablePrefix = ""; - if (fetchFromRetentionTables) { - tablePrefix = ConsentMgtDAOConstants.RETENTION_TABLE_NAME_PREFIX; - } - return "SELECT obc.CONSENT_ID," + - "RECEIPT, " + - "CLIENT_ID, " + - "CONSENT_TYPE, " + - "CURRENT_STATUS, " + - "CONSENT_FREQUENCY, " + - "VALIDITY_TIME, " + - "RECURRING_INDICATOR, " + - "CREATED_TIME AS CONSENT_CREATED_TIME, " + - "obc.UPDATED_TIME AS CONSENT_UPDATED_TIME, " + - "ca.ATT_KEY, " + - "ca.ATT_VALUE, " + - "ocar.AUTH_ID, " + - "ocar.AUTH_STATUS, " + - "ocar.AUTH_TYPE, " + - "ocar.UPDATED_TIME AS AUTH_UPDATED_TIME, " + - "ocar.USER_ID, " + - "cm.ACCOUNT_ID, " + - "cm.MAPPING_ID, " + - "cm.MAPPING_STATUS, " + - "cm.PERMISSION " + - "FROM " + - tablePrefix + "OB_CONSENT obc " + - "LEFT JOIN " + - tablePrefix + "OB_CONSENT_ATTRIBUTE ca ON obc.CONSENT_ID=ca.CONSENT_ID " + - "LEFT JOIN " + - tablePrefix + "OB_CONSENT_AUTH_RESOURCE ocar ON obc.CONSENT_ID=ocar.CONSENT_ID " + - "LEFT JOIN " + - tablePrefix + "OB_CONSENT_MAPPING cm ON ocar.AUTH_ID=cm.AUTH_ID " + - "WHERE obc.CONSENT_ID = ?"; - } - - public String getSearchConsentsPreparedStatement(String whereClause, boolean shouldLimit, boolean shouldOffset, - String userIdFilterClause) { - - String selectClause = "(SELECT * FROM OB_CONSENT " + whereClause + ")"; - String joinType = "LEFT "; - if (StringUtils.isNotEmpty(userIdFilterClause)) { - joinType = "INNER "; - userIdFilterClause = "AND " + userIdFilterClause; - } - - StringBuilder query = new StringBuilder("SELECT OBC.CONSENT_ID, " + - "RECEIPT, " + - "CLIENT_ID, " + - "CONSENT_TYPE, " + - "OBC.CURRENT_STATUS AS CURRENT_STATUS," + - "CONSENT_FREQUENCY," + - "VALIDITY_TIME," + - "RECURRING_INDICATOR," + - "OBC.CREATED_TIME AS CONSENT_CREATED_TIME," + - "OBC.UPDATED_TIME AS CONSENT_UPDATED_TIME," + - "Group_concat(distinct CA.att_key order by CA.att_key SEPARATOR '||' ) AS ATT_KEY, " + - "Group_concat(distinct CA.att_value order by CA.att_key SEPARATOR '||') AS ATT_VALUE, " + - - - "( SELECT Group_concat( OCAR2.auth_id order by OCAR2.auth_id SEPARATOR '||') " + - " FROM OB_CONSENT_AUTH_RESOURCE OCAR2 " + - " WHERE OCAR2.consent_id = OBC.consent_id " + - " GROUP BY OCAR2.consent_id ) AS AUTH_ID, " + - - "( SELECT Group_concat(OCAR2.auth_status order by OCAR2.auth_id SEPARATOR '||') " + - " FROM OB_CONSENT_AUTH_RESOURCE OCAR2 " + - " WHERE OCAR2.consent_id = OBC.consent_id " + - " GROUP BY OCAR2.consent_id ) AS AUTH_STATUS, " + - - "( SELECT Group_concat(OCAR2.auth_type order by OCAR2.auth_id SEPARATOR '||') " + - " FROM OB_CONSENT_AUTH_RESOURCE OCAR2 " + - " WHERE OCAR2.consent_id = OBC.consent_id " + - " GROUP BY OCAR2.consent_id ) AS AUTH_TYPE, " + - - "( SELECT Group_concat(OCAR2.updated_time order by OCAR2.auth_id SEPARATOR '||') " + - " FROM OB_CONSENT_AUTH_RESOURCE OCAR2 " + - " WHERE OCAR2.consent_id = OBC.consent_id " + - " GROUP BY OCAR2.consent_id ) AS UPDATED_TIME, " + - - "( SELECT Group_concat(OCAR2.user_id order by OCAR2.auth_id SEPARATOR '||') " + - " FROM OB_CONSENT_AUTH_RESOURCE OCAR2 " + - " WHERE OCAR2.consent_id = OBC.consent_id " + - " GROUP BY OCAR2.consent_id ) AS USER_ID," + - - - " ( SELECT Group_concat(OCM2.auth_id order by OCM2.mapping_id SEPARATOR '||') " + - " FROM OB_CONSENT_MAPPING OCM2 " + - " JOIN OB_CONSENT_AUTH_RESOURCE OCAR2 ON OCAR2.auth_id = OCM2.auth_id " + - " WHERE OCAR2.consent_id = OBC.consent_id) AS AUTH_MAPPING_ID , " + - - "( SELECT Group_concat(OCM2.account_id order by OCM2.mapping_id SEPARATOR '||') " + - " FROM OB_CONSENT_MAPPING OCM2 " + - " JOIN OB_CONSENT_AUTH_RESOURCE OCAR2 ON OCAR2.auth_id = OCM2.auth_id " + - " WHERE OCAR2.consent_id = OBC.consent_id) AS ACCOUNT_ID , " + - - "( SELECT Group_concat(OCM2.mapping_id order by OCM2.mapping_id SEPARATOR '||') " + - " FROM OB_CONSENT_MAPPING OCM2 " + - " JOIN OB_CONSENT_AUTH_RESOURCE OCAR2 ON OCAR2.auth_id = OCM2.auth_id " + - " WHERE OCAR2.consent_id = OBC.consent_id) AS MAPPING_ID , " + - - "( SELECT Group_concat(OCM2.mapping_status order by OCM2.mapping_id SEPARATOR '||') " + - " FROM OB_CONSENT_MAPPING OCM2 " + - " JOIN OB_CONSENT_AUTH_RESOURCE OCAR2 ON OCAR2.auth_id = OCM2.auth_id " + - " WHERE OCAR2.consent_id = OBC.consent_id) AS MAPPING_STATUS , " + - - "( SELECT Group_concat(OCM2.permission order by OCM2.mapping_id SEPARATOR '||') " + - " FROM OB_CONSENT_MAPPING OCM2 " + - " JOIN OB_CONSENT_AUTH_RESOURCE OCAR2 ON OCAR2.auth_id = OCM2.auth_id " + - " WHERE OCAR2.consent_id = OBC.consent_id) AS PERMISSION " + - - "FROM " + - selectClause + - "AS OBC " + - "LEFT JOIN OB_CONSENT_ATTRIBUTE CA ON OBC.CONSENT_ID=CA.CONSENT_ID " + - joinType + "JOIN OB_CONSENT_AUTH_RESOURCE OCAR ON OBC.CONSENT_ID=OCAR.CONSENT_ID " - + userIdFilterClause + - "LEFT JOIN OB_CONSENT_MAPPING OCM ON OCAR.AUTH_ID=OCM.AUTH_ID WHERE " + - "(OBC.UPDATED_TIME >= COALESCE(?, OBC.UPDATED_TIME) " + - "AND OBC.UPDATED_TIME <= COALESCE(?, OBC.UPDATED_TIME)) " + - "group by OBC.CONSENT_ID ORDER BY OBC.UPDATED_TIME DESC "); - - if (shouldLimit && shouldOffset) { - query.append(" LIMIT ? OFFSET ? "); - } else if (shouldLimit) { - query.append(" LIMIT ? "); - } - - return query.toString(); - } - - public String getGetConsentWithConsentAttributesPreparedStatement() { - - return "SELECT OB_CONSENT.CONSENT_ID, RECEIPT, CREATED_TIME, UPDATED_TIME, CLIENT_ID, CONSENT_TYPE, " + - "CURRENT_STATUS, CONSENT_FREQUENCY, VALIDITY_TIME, RECURRING_INDICATOR, " + - "OB_CONSENT_ATTRIBUTE.ATT_KEY, OB_CONSENT_ATTRIBUTE.ATT_VALUE FROM OB_CONSENT RIGHT JOIN " + - "OB_CONSENT_ATTRIBUTE ON OB_CONSENT.CONSENT_ID = OB_CONSENT_ATTRIBUTE.CONSENT_ID WHERE OB_CONSENT" + - ".CONSENT_ID = ?"; - } - - public String getGetConsentAttributesByNamePreparedStatement() { - - return "SELECT CONSENT_ID, ATT_VALUE FROM OB_CONSENT_ATTRIBUTE WHERE ATT_KEY = ?"; - } - - public String getConsentIdByConsentAttributeNameAndValuePreparedStatement() { - - - return "SELECT CONSENT_ID FROM OB_CONSENT_ATTRIBUTE WHERE ATT_KEY = ? AND ATT_VALUE = ?"; - } - - public String getGetAuthorizationResourcePreparedStatement() { - - return "SELECT * FROM OB_CONSENT_AUTH_RESOURCE WHERE AUTH_ID = ?"; - } - - public String getGetConsentMappingResourcesPreparedStatement() { - - return "SELECT * FROM OB_CONSENT_MAPPING WHERE AUTH_ID = ?"; - } - - public String getGetConsentMappingResourcesForStatusPreparedStatement() { - - return "SELECT * FROM OB_CONSENT_MAPPING WHERE AUTH_ID = ? AND MAPPING_STATUS = ?"; - } - - public String getDeleteConsentAttributePreparedStatement() { - - return "DELETE FROM OB_CONSENT_ATTRIBUTE WHERE CONSENT_ID = ? AND ATT_KEY = ?"; - } - - public String getGetConsentStatusAuditRecordsPreparedStatement(boolean fetchFromRetentionTables) { - - // table prefix is to fetch from the consent retention data (purged data) tables. (if enabled) - String tablePrefix = ""; - if (fetchFromRetentionTables) { - tablePrefix = ConsentMgtDAOConstants.RETENTION_TABLE_NAME_PREFIX; - } - return "SELECT * FROM " + tablePrefix + "OB_CONSENT_STATUS_AUDIT WHERE CONSENT_ID = COALESCE(?, CONSENT_ID) " + - "AND CURRENT_STATUS = COALESCE(?, CURRENT_STATUS) AND ACTION_BY = COALESCE(?, ACTION_BY) " + - "AND STATUS_AUDIT_ID = COALESCE (?, STATUS_AUDIT_ID) AND ACTION_TIME >= COALESCE(?, ACTION_TIME) " + - "AND ACTION_TIME <= COALESCE(?, ACTION_TIME)"; - } - - public String getSearchAuthorizationResourcesPreparedStatement(String whereClause) { - - return "SELECT * FROM OB_CONSENT_AUTH_RESOURCE" + whereClause; - } - - public String getUpdateConsentReceiptPreparedStatement() { - - return "UPDATE OB_CONSENT SET RECEIPT = ? WHERE CONSENT_ID = ?"; - } - - public String getUpdateConsentValidityTimePreparedStatement() { - - return "UPDATE OB_CONSENT SET VALIDITY_TIME = ?, UPDATED_TIME = ? WHERE CONSENT_ID = ?"; - } - - public String getSearchExpiringConsentPreparedStatement(String statusesEligibleForExpirationCondition) { - - return "SELECT OBC.CONSENT_ID " + - " FROM OB_CONSENT_ATTRIBUTE CA " + - " JOIN OB_CONSENT OBC " + - " ON CA.CONSENT_ID = OBC.CONSENT_ID " + - " WHERE CA.ATT_KEY = ? AND OBC.CURRENT_STATUS IN " + statusesEligibleForExpirationCondition; - } - - public String getInsertConsentHistoryPreparedStatement() { - - return "INSERT INTO OB_CONSENT_HISTORY (TABLE_ID, RECORD_ID, HISTORY_ID, CHANGED_VALUES, " + - "REASON, EFFECTIVE_TIMESTAMP) VALUES (?, ?, ?, ?, ?, ?)"; - } - - public String getGetConsentHistoryPreparedStatement(String whereClause) { - - return "SELECT * FROM OB_CONSENT_HISTORY " + whereClause + "ORDER BY EFFECTIVE_TIMESTAMP DESC"; - } - - /** - * SQL query for delete consent attributes. - * @param executeOnRetentionTables whether to execute on retention tables - * @return SQL query for delete consent attributes - */ - public String getDeleteConsentAttributeByConsentIdPreparedStatement(boolean executeOnRetentionTables) { - - // table prefix is to execute on consent retention data (purged data) tables. (if enabled) - String tablePrefix = ""; - if (executeOnRetentionTables) { - tablePrefix = ConsentMgtDAOConstants.RETENTION_TABLE_NAME_PREFIX; - } - return "DELETE FROM " + tablePrefix + "OB_CONSENT_ATTRIBUTE WHERE CONSENT_ID = ?"; - } - - /** - * SQL query for delete consent file. - * @param executeOnRetentionTables whether to execute on retention tables - * @return SQL query for delete consent file - */ - public String getDeleteConsentFileResourcePreparedStatement(boolean executeOnRetentionTables) { - - // table prefix is to execute on consent retention data (purged data) tables. (if enabled) - String tablePrefix = ""; - if (executeOnRetentionTables) { - tablePrefix = ConsentMgtDAOConstants.RETENTION_TABLE_NAME_PREFIX; - } - return "DELETE FROM " + tablePrefix + "OB_CONSENT_FILE WHERE CONSENT_ID = ?"; - } - - /** - * SQL query for delete consent mapping by auth id. - * @param executeOnRetentionTables whether to execute on retention tables - * @return SQL query for delete consent mapping by auth id - */ - public String getDeleteConsentMappingByAuthIdPreparedStatement(boolean executeOnRetentionTables) { - - // table prefix is to execute on consent retention data (purged data) tables. (if enabled) - String tablePrefix = ""; - if (executeOnRetentionTables) { - tablePrefix = ConsentMgtDAOConstants.RETENTION_TABLE_NAME_PREFIX; - } - return "DELETE OBCM FROM " + tablePrefix + "OB_CONSENT_MAPPING OBCM INNER JOIN " + tablePrefix + - "OB_CONSENT_AUTH_RESOURCE OBAR ON OBCM.AUTH_ID = OBAR.AUTH_ID WHERE OBAR.CONSENT_ID = ?"; - } - - /** - * SQL query for delete auth resource. - * @param executeOnRetentionTables whether to execute on retention tables - * @return SQL query for delete auth resource - */ - public String getDeleteAuthorizationResourcePreparedStatement(boolean executeOnRetentionTables) { - - // table prefix is to execute on consent retention data (purged data) tables. (if enabled) - String tablePrefix = ""; - if (executeOnRetentionTables) { - tablePrefix = ConsentMgtDAOConstants.RETENTION_TABLE_NAME_PREFIX; - } - return "DELETE FROM " + tablePrefix + "OB_CONSENT_AUTH_RESOURCE WHERE CONSENT_ID = ?"; - } - - /** - * SQL query for consent status audit record. - * @param executeOnRetentionTables whether to execute on retention tables - * @return SQL query for consent status audit record - */ - public String getDeleteConsentStatusAuditRecordsPreparedStatement(boolean executeOnRetentionTables) { - - // table prefix is to execute on consent retention data (purged data) tables. (if enabled) - String tablePrefix = ""; - if (executeOnRetentionTables) { - tablePrefix = ConsentMgtDAOConstants.RETENTION_TABLE_NAME_PREFIX; - } - return "DELETE FROM " + tablePrefix + "OB_CONSENT_STATUS_AUDIT WHERE CONSENT_ID = ?"; - } - - /** - * SQL query for delete consent. - * @param executeOnRetentionTables whether to execute on retention tables - * @return SQL query for delete consent - */ - public String getDeleteConsentPreparedStatement(boolean executeOnRetentionTables) { - - // table prefix is to execute on consent retention data (purged data) tables. (if enabled) - String tablePrefix = ""; - if (executeOnRetentionTables) { - tablePrefix = ConsentMgtDAOConstants.RETENTION_TABLE_NAME_PREFIX; - } - return "DELETE FROM " + tablePrefix + "OB_CONSENT WHERE CONSENT_ID = ?"; - } - - /** - * SQL query for get list of consent_ids. - * @param fetchFromRetentionTables whether to fetch from retention tables - * @return SQL query for get list of consent_ids - */ - public String getListOfConsentIdsPreparedStatement(boolean fetchFromRetentionTables) { - - // table prefix is to fetch from the consent retention data (purged data) tables. (if enabled) - String tablePrefix = ""; - if (fetchFromRetentionTables) { - tablePrefix = ConsentMgtDAOConstants.RETENTION_TABLE_NAME_PREFIX; - } - return "SELECT CONSENT_ID FROM " + tablePrefix + "OB_CONSENT"; - } - - /** - * SQL query for get consent status audit records by consentIds. - * @param whereClause conditions - * @param shouldLimit whether to consider the Limit parameter - * @param shouldOffset whether to consider the Offset parameter - * @param fetchFromRetentionTables whether to fetch from retention tables - * @return SQL query for get consent status audit records by consentIds - */ - public String getConsentStatusAuditRecordsByConsentIdsPreparedStatement(String whereClause, boolean shouldLimit, - boolean shouldOffset, - boolean fetchFromRetentionTables) { - - // table prefix is to fetch from the consent retention data (purged data) tables. (if enabled) - String tablePrefix = ""; - if (fetchFromRetentionTables) { - tablePrefix = ConsentMgtDAOConstants.RETENTION_TABLE_NAME_PREFIX; - } - StringBuilder query = - new StringBuilder("SELECT * FROM " + tablePrefix + "OB_CONSENT_STATUS_AUDIT " + whereClause); - - if (shouldLimit && shouldOffset) { - query.append(" LIMIT ? OFFSET ? "); - } else if (shouldLimit) { - query.append(" LIMIT ? "); - } - return query.toString(); - } - - /** - * Util method to get the limit offset order for differentiate oracle and mssql pagination. - * @return is limit is before in prepared statement than offset - */ - public boolean isLimitBeforeThanOffset() { - - return true; - } - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/queries/ConsentMgtMssqlDBQueries.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/queries/ConsentMgtMssqlDBQueries.java deleted file mode 100644 index d9b68208..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/queries/ConsentMgtMssqlDBQueries.java +++ /dev/null @@ -1,199 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.queries; - -import com.wso2.openbanking.accelerator.consent.mgt.dao.constants.ConsentMgtDAOConstants; -import org.apache.commons.lang.StringUtils; - -/** - * The Microsoft SQL database queries used by the consent management DAO layer. - */ -public class ConsentMgtMssqlDBQueries extends ConsentMgtCommonDBQueries { - - public String getSearchConsentsPreparedStatement(String whereClause, boolean shouldLimit, boolean shouldOffset, - String userIdFilterClause) { - - String selectClause = "(SELECT * FROM OB_CONSENT)"; - String joinType = " LEFT "; - - if (StringUtils.isNotEmpty(userIdFilterClause)) { - joinType = " INNER "; - userIdFilterClause = "AND " + userIdFilterClause; - } - - if (whereClause.trim().isEmpty()) { - whereClause = " WHERE "; - } else { - whereClause = whereClause + " AND "; - } - - StringBuilder query = new StringBuilder("SELECT OBC.CONSENT_ID, " + - " (SELECT receipt FROM OB_CONSENT WHERE consent_id = obc.consent_id " + - " order by consent_id offset 0 rows FETCH next 1 rows only ) AS RECEIPT, " + - " (SELECT client_id FROM OB_CONSENT WHERE consent_id = obc.consent_id " + - " order by consent_id offset 0 rows FETCH next 1 rows only ) AS CLIENT_ID, " + - " (SELECT consent_type FROM OB_CONSENT WHERE consent_id = obc.consent_id " + - " order by consent_id offset 0 rows FETCH next 1 rows only ) AS CONSENT_TYPE, " + - " (SELECT current_status FROM OB_CONSENT WHERE consent_id = obc.consent_id " + - " order by consent_id offset 0 rows FETCH next 1 rows only ) AS CURRENT_STATUS, " + - " (SELECT consent_frequency FROM OB_CONSENT WHERE consent_id = obc.consent_id " + - " order by consent_id offset 0 rows FETCH next 1 rows only ) AS CONSENT_FREQUENCY, " + - " (SELECT validity_time FROM OB_CONSENT WHERE consent_id = obc.consent_id " + - " order by consent_id offset 0 rows FETCH next 1 rows only ) AS VALIDITY_TIME, " + - " (SELECT recurring_indicator FROM OB_CONSENT WHERE consent_id = obc.consent_id " + - " order by consent_id offset 0 rows FETCH next 1 rows only ) AS RECURRING_INDICATOR, " + - " (SELECT created_time FROM OB_CONSENT WHERE consent_id = obc.consent_id " + - " order by consent_id offset 0 rows FETCH next 1 rows only ) AS CONSENT_CREATED_TIME, " + - " (SELECT updated_time FROM OB_CONSENT WHERE consent_id = obc.consent_id " + - " order by consent_id offset 0 rows FETCH next 1 rows only ) AS CONSENT_UPDATED_TIME, " + - - "( SELECT String_agg(att_key , '||') within GROUP (ORDER BY att_key) " + - " FROM OB_CONSENT_ATTRIBUTE " + - " WHERE consent_id = OBC.consent_id " + - " GROUP BY consent_id ) AS ATT_KEY, " + - - "( SELECT String_agg(att_value , '||') within GROUP (ORDER BY att_key) " + - " FROM OB_CONSENT_ATTRIBUTE " + - " WHERE consent_id = OBC.consent_id " + - " GROUP BY consent_id ) AS ATT_VALUE, " + - - "( SELECT String_agg(auth_id , '||') within GROUP (ORDER BY auth_id) " + - " FROM OB_CONSENT_AUTH_RESOURCE " + - " WHERE consent_id = OBC.consent_id " + - " GROUP BY consent_id ) AS AUTH_ID, " + - - "( SELECT String_agg(auth_status , '||') within GROUP (ORDER BY auth_id) " + - " FROM OB_CONSENT_AUTH_RESOURCE " + - " WHERE consent_id = OBC.consent_id " + - " GROUP BY consent_id ) AS AUTH_STATUS, " + - - "( SELECT String_agg(auth_type , '||') within GROUP (ORDER BY auth_id) " + - " FROM OB_CONSENT_AUTH_RESOURCE " + - " WHERE consent_id = OBC.consent_id " + - " GROUP BY consent_id ) AS AUTH_TYPE, " + - - "( SELECT String_agg(updated_time , '||') within GROUP (ORDER BY auth_id) " + - " FROM OB_CONSENT_AUTH_RESOURCE " + - " WHERE consent_id = OBC.consent_id " + - " GROUP BY consent_id ) AS UPDATED_TIME, " + - - "( SELECT String_agg(user_id , '||') within GROUP (ORDER BY auth_id) " + - " FROM OB_CONSENT_AUTH_RESOURCE " + - " WHERE consent_id = OBC.consent_id " + - " GROUP BY consent_id ) AS USER_ID," + - - - "( SELECT String_agg(OCM2.AUTH_ID , '||') within GROUP (ORDER BY OCM2.mapping_id) " + - " FROM OB_CONSENT_MAPPING OCM2 " + - " JOIN OB_CONSENT_AUTH_RESOURCE ocar2 ON ocar2.auth_id = OCM2.auth_id " + - " WHERE ocar2.consent_id = obc.consent_id) AS AUTH_MAPPING_ID , " + - - "( SELECT String_agg(OCM2.account_id , '||') within GROUP (ORDER BY OCM2.mapping_id) " + - " FROM OB_CONSENT_MAPPING OCM2 " + - " JOIN OB_CONSENT_AUTH_RESOURCE ocar2 ON ocar2.auth_id = OCM2.auth_id " + - " WHERE ocar2.consent_id = obc.consent_id) AS ACCOUNT_ID , " + - - "( SELECT String_agg(OCM2.mapping_id , '||') within GROUP (ORDER BY OCM2.mapping_id) " + - " FROM OB_CONSENT_MAPPING OCM2 " + - " JOIN OB_CONSENT_AUTH_RESOURCE ocar2 ON ocar2.auth_id = OCM2.auth_id " + - " WHERE ocar2.consent_id = obc.consent_id) AS MAPPING_ID , " + - - "( SELECT String_agg(OCM2.mapping_status , '||') within GROUP (ORDER BY OCM2.mapping_id) " + - " FROM OB_CONSENT_MAPPING OCM2 " + - " JOIN OB_CONSENT_AUTH_RESOURCE ocar2 ON ocar2.auth_id = OCM2.auth_id " + - " WHERE ocar2.consent_id = obc.consent_id) AS MAPPING_STATUS , " + - - "( SELECT String_agg(OCM2.permission , '||') within GROUP (ORDER BY OCM2.mapping_id) " + - " FROM OB_CONSENT_MAPPING OCM2 " + - " JOIN OB_CONSENT_AUTH_RESOURCE ocar2 ON ocar2.auth_id = OCM2.auth_id " + - " WHERE ocar2.consent_id = obc.consent_id) AS PERMISSION " + - - - "FROM " + selectClause + "AS OBC " + - "LEFT JOIN OB_CONSENT_ATTRIBUTE CA ON OBC.CONSENT_ID=CA.CONSENT_ID " + - joinType + "JOIN OB_CONSENT_AUTH_RESOURCE OCAR ON OBC.CONSENT_ID=OCAR.CONSENT_ID " - + userIdFilterClause + - "LEFT JOIN OB_CONSENT_MAPPING OCM ON OCAR.AUTH_ID=OCM.AUTH_ID " + whereClause + - " (OBC.UPDATED_TIME >= COALESCE(?, OBC.UPDATED_TIME) " + - " AND OBC.UPDATED_TIME <= COALESCE(?, OBC.UPDATED_TIME)) GROUP BY obc.consent_id " + - " ORDER BY UPDATED_TIME DESC "); - - if (shouldLimit && shouldOffset) { - query.append("OFFSET ? ROWS FETCH NEXT ? ROWS ONLY "); - } else if (shouldLimit) { - query.append("OFFSET 0 ROWS FETCH NEXT ? ROWS ONLY "); - } - - return query.toString(); - } - - /** - * SQL query for delete consent mapping by auth id. - * @param executeOnRetentionTables execute on retention tables - * @return SQL query - */ - public String getDeleteConsentMappingByAuthIdPreparedStatement(boolean executeOnRetentionTables) { - - // table prefix is to execute on consent retention data (purged data) tables. (if enabled) - String tablePrefix = ""; - if (executeOnRetentionTables) { - tablePrefix = ConsentMgtDAOConstants.RETENTION_TABLE_NAME_PREFIX; - } - return "DELETE " + tablePrefix + "OB_CONSENT_MAPPING where MAPPING_ID in (SELECT MAPPING_ID FROM " + - tablePrefix + "OB_CONSENT_MAPPING OBCM INNER JOIN " + tablePrefix + "OB_CONSENT_AUTH_RESOURCE OBAR " + - "ON OBCM.AUTH_ID = OBAR.AUTH_ID WHERE OBAR.CONSENT_ID = ?)"; - } - - /** - * SQL query for get consent status audit records by consentIds. - * @param whereClause conditions - * @param shouldLimit limit - * @param shouldOffset offset - * @param fetchFromRetentionTables fetch from retention tables - * @return SQL query - */ - public String getConsentStatusAuditRecordsByConsentIdsPreparedStatement(String whereClause, boolean shouldLimit, - boolean shouldOffset, - boolean fetchFromRetentionTables) { - - // table prefix is to fetch from the consent retention data (purged data) tables. (if enabled) - String tablePrefix = ""; - if (fetchFromRetentionTables) { - tablePrefix = ConsentMgtDAOConstants.RETENTION_TABLE_NAME_PREFIX; - } - StringBuilder query = - new StringBuilder("SELECT * FROM " + tablePrefix + "OB_CONSENT_STATUS_AUDIT " + whereClause); - - if (shouldLimit && shouldOffset) { - query.append("ORDER BY STATUS_AUDIT_ID OFFSET ? ROWS FETCH NEXT ? ROWS ONLY "); - } else if (shouldLimit) { - query.append("ORDER BY STATUS_AUDIT_ID OFFSET 0 ROWS FETCH NEXT ? ROWS ONLY "); - } - return query.toString(); - } - - /** - * Util method to get the limit offset order for differentiate oracle and mssql pagination. - * @return is limit is before in prepared statement than offset - */ - public boolean isLimitBeforeThanOffset() { - - return false; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/queries/ConsentMgtOracleDBQueries.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/queries/ConsentMgtOracleDBQueries.java deleted file mode 100644 index d321977c..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/queries/ConsentMgtOracleDBQueries.java +++ /dev/null @@ -1,202 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.queries; - -import com.wso2.openbanking.accelerator.consent.mgt.dao.constants.ConsentMgtDAOConstants; -import org.apache.commons.lang.StringUtils; - -/** - * The Oracle database queries used by the consent management DAO layer. - */ -public class ConsentMgtOracleDBQueries extends ConsentMgtCommonDBQueries { - - public String getSearchConsentsPreparedStatement(String whereClause, boolean shouldLimit, boolean shouldOffset, - String userIdFilterClause) { - - String selectClause = "OB_CONSENT "; - String joinType = "LEFT"; - - if (StringUtils.isNotEmpty(userIdFilterClause)) { - joinType = "INNER"; - userIdFilterClause = "AND " + userIdFilterClause; - } - - if (whereClause.trim().isEmpty()) { - whereClause = " WHERE "; - } else { - whereClause = whereClause + " AND "; - } - - StringBuilder query = new StringBuilder("SELECT OBC.CONSENT_ID, " + - " ( SELECT receipt FROM OB_CONSENT WHERE consent_id = obc.consent_id " + - " FETCH first 1 rows only ) AS RECEIPT, " + - " (SELECT client_id FROM OB_CONSENT WHERE consent_id = obc.consent_id " + - " FETCH FIRST 1 rows only ) AS CLIENT_ID, " + - " (SELECT consent_type FROM OB_CONSENT WHERE consent_id = obc.consent_id " + - " FETCH FIRST 1 rows only ) AS CONSENT_TYPE, " + - " (SELECT current_status FROM OB_CONSENT WHERE consent_id = obc.consent_id " + - " FETCH FIRST 1 rows only ) AS current_status, " + - " (SELECT consent_frequency FROM OB_CONSENT WHERE consent_id = obc.consent_id " + - " FETCH FIRST 1 rows only ) AS CONSENT_FREQUENCY, " + - " (SELECT validity_time FROM OB_CONSENT WHERE consent_id = obc.consent_id " + - " FETCH FIRST 1 rows only ) AS VALIDITY_TIME, " + - " (SELECT recurring_indicator FROM OB_CONSENT WHERE consent_id = obc.consent_id " + - " FETCH FIRST 1 rows only ) AS RECURRING_INDICATOR, " + - " (SELECT created_time FROM OB_CONSENT WHERE consent_id = obc.consent_id " + - " FETCH FIRST 1 rows only ) AS consent_created_time, " + - " (SELECT updated_time FROM OB_CONSENT WHERE consent_id = obc.consent_id " + - " FETCH FIRST 1 rows only ) AS consent_updated_time, " + - - " ( SELECT listagg(att_key || '||') within GROUP (ORDER BY att_key) " + - " FROM ob_consent_attribute " + - " WHERE consent_id = obc.consent_id " + - " GROUP BY consent_id ) AS ATT_KEY, " + - - " ( SELECT listagg(att_value || '||') within GROUP (ORDER BY att_key)" + - " FROM ob_consent_attribute" + - " WHERE consent_id = obc.consent_id" + - " GROUP BY consent_id ) AS ATT_VALUE, " + - - " ( SELECT listagg(auth_id || '||') within GROUP (ORDER BY auth_id)" + - " FROM ob_consent_auth_resource" + - " WHERE consent_id = obc.consent_id" + - " GROUP BY consent_id ) AS AUTH_ID, " + - - " ( SELECT listagg(auth_status || '||') within GROUP (ORDER BY auth_id)" + - " FROM ob_consent_auth_resource" + - " WHERE consent_id = obc.consent_id" + - " GROUP BY consent_id ) AS AUTH_STATUS, " + - - " ( SELECT listagg(auth_type || '||') within GROUP (ORDER BY auth_id) " + - " FROM ob_consent_auth_resource " + - " WHERE consent_id = obc.consent_id " + - " GROUP BY consent_id ) AS AUTH_TYPE, " + - - " ( SELECT listagg(updated_time || '||') within GROUP (ORDER BY auth_id) " + - " FROM ob_consent_auth_resource " + - " WHERE consent_id = obc.consent_id " + - " GROUP BY consent_id ) AS UPDATED_TIME, " + - - " ( SELECT listagg(user_id || '||') within GROUP (ORDER BY auth_id) " + - " FROM ob_consent_auth_resource " + - " WHERE consent_id = obc.consent_id " + - " GROUP BY consent_id ) AS USER_ID, " + - - - " ( SELECT listagg(ocm2.auth_id || '||') within GROUP (ORDER BY ocm2.mapping_id) " + - " FROM ob_consent_mapping ocm2 " + - " JOIN ob_consent_auth_resource ocar2 " + - " ON ocar2.auth_id = ocm2.auth_id " + - " WHERE ocar2.consent_id = obc.consent_id) AS AUTH_MAPPING_ID , " + - - " ( SELECT listagg(ocm2.account_id || '||') within GROUP (ORDER BY ocm2.mapping_id) " + - " FROM ob_consent_mapping ocm2 " + - " JOIN ob_consent_auth_resource ocar2 " + - " ON ocar2.auth_id = ocm2.auth_id " + - " WHERE ocar2.consent_id = obc.consent_id) AS ACCOUNT_ID , " + - - " ( SELECT listagg(ocm2.mapping_id || '||') within GROUP (ORDER BY ocm2.mapping_id) " + - " FROM ob_consent_mapping ocm2 " + - " JOIN ob_consent_auth_resource ocar2 " + - " ON ocar2.auth_id = ocm2.auth_id " + - " WHERE ocar2.consent_id = obc.consent_id) AS MAPPING_ID , " + - - " ( SELECT listagg(ocm2.mapping_status || '||') within GROUP (ORDER BY ocm2.mapping_id) " + - " FROM ob_consent_mapping ocm2 " + - " JOIN ob_consent_auth_resource ocar2 " + - " ON ocar2.auth_id = ocm2.auth_id " + - " WHERE ocar2.consent_id = obc.consent_id) AS MAPPING_STATUS , " + - - " ( SELECT listagg(ocm2.permission || '||') within GROUP (ORDER BY ocm2.mapping_id) " + - " FROM ob_consent_mapping ocm2 " + - " JOIN ob_consent_auth_resource ocar2 " + - " ON ocar2.auth_id = ocm2.auth_id " + - " WHERE ocar2.consent_id = obc.consent_id) AS PERMISSION " + - - "FROM " + selectClause + " OBC " + - "LEFT JOIN OB_CONSENT_ATTRIBUTE CA ON OBC.CONSENT_ID=CA.CONSENT_ID " + - joinType + " JOIN OB_CONSENT_AUTH_RESOURCE OCAR ON OBC.CONSENT_ID=OCAR.CONSENT_ID " - + userIdFilterClause + - "LEFT JOIN OB_CONSENT_MAPPING OCM ON OCAR.AUTH_ID=OCM.AUTH_ID " + whereClause + - " (OBC.UPDATED_TIME >= COALESCE(?, OBC.UPDATED_TIME) " + - " AND OBC.UPDATED_TIME <= COALESCE(?, OBC.UPDATED_TIME)) GROUP BY obc.consent_id " + - "ORDER BY UPDATED_TIME DESC "); - - if (shouldLimit && shouldOffset) { - query.append("OFFSET ? ROWS FETCH NEXT ? ROWS ONLY "); - } else if (shouldLimit) { - query.append("FETCH NEXT ? ROWS ONLY"); - } - return query.toString(); - } - - /** - * SQL query for delete consent mapping by auth id. - * @param executeOnRetentionTables whether to execute on retention tables - * @return SQL query to delete consent mapping by auth id - */ - public String getDeleteConsentMappingByAuthIdPreparedStatement(boolean executeOnRetentionTables) { - - // table prefix is to execute on consent retention data (purged data) tables. (if enabled) - String tablePrefix = ""; - if (executeOnRetentionTables) { - tablePrefix = ConsentMgtDAOConstants.RETENTION_TABLE_NAME_PREFIX; - } - return "DELETE FROM " + tablePrefix + "OB_CONSENT_MAPPING where MAPPING_ID in (SELECT MAPPING_ID FROM " + - tablePrefix + "OB_CONSENT_MAPPING OBCM INNER JOIN " + tablePrefix + "OB_CONSENT_AUTH_RESOURCE OBAR " + - "ON OBCM.AUTH_ID = OBAR.AUTH_ID WHERE OBAR.CONSENT_ID = ?)"; - } - - /** - * SQL query for get consent status audit records by consentIds. - * @param whereClause conditions - * @param shouldLimit whether limit should be applied - * @param shouldOffset whether offset should be applied - * @param fetchFromRetentionTables whether to fetch from retention tables - * @return SQL query to retrieve consent status audit records by consentIds - */ - public String getConsentStatusAuditRecordsByConsentIdsPreparedStatement(String whereClause, boolean shouldLimit, - boolean shouldOffset, - boolean fetchFromRetentionTables) { - - // table prefix is to fetch from the consent retention data (purged data) tables. (if enabled) - String tablePrefix = ""; - if (fetchFromRetentionTables) { - tablePrefix = ConsentMgtDAOConstants.RETENTION_TABLE_NAME_PREFIX; - } - StringBuilder query = - new StringBuilder("SELECT * FROM " + tablePrefix + "OB_CONSENT_STATUS_AUDIT " + whereClause); - - if (shouldLimit && shouldOffset) { - query.append("OFFSET ? ROWS FETCH NEXT ? ROWS ONLY "); - } else if (shouldLimit) { - query.append("FETCH NEXT ? ROWS ONLY"); - } - return query.toString(); - } - - /** - * Util method to get the limit offset order for differentiate oracle and mssql pagination. - * @return is limit is before in prepared statement than offset - */ - public boolean isLimitBeforeThanOffset() { - - return false; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/queries/ConsentMgtPostgresDBQueries.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/queries/ConsentMgtPostgresDBQueries.java deleted file mode 100644 index ec5af0d2..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/queries/ConsentMgtPostgresDBQueries.java +++ /dev/null @@ -1,229 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.queries; -import com.wso2.openbanking.accelerator.consent.mgt.dao.constants.ConsentMgtDAOConstants; -import org.apache.commons.lang3.StringUtils; - -/** - * The PostgreSQL database queries used by the consent management DAO layer. - */ -public class ConsentMgtPostgresDBQueries extends ConsentMgtCommonDBQueries { - - /** - * This method returns the detailed consent search query. It constructs the query according to the provided - * parameters dynamically. This queries all consent attributes, authorization resources, mapping resources, consent - * data of the provided consent according to the parameters provided. To avoid fetching same rows multiple times, - * the string_agg function is used to concatenate values using a delimiter. The delimited results are later - * processed and set to the result. - * - * @param whereClause the pre-constructed where dynamic where clause - * @param shouldLimit flag that indicates the limit - * @param shouldOffset flag that indicates the offset - * @param userIdFilterClause the pre-constructed user id filter condition - * @return the constructed prepared statement for consent search function - */ - public String getSearchConsentsPreparedStatement(String whereClause, boolean shouldLimit, boolean shouldOffset, - String userIdFilterClause) { - - String selectClause = "(SELECT * FROM OB_CONSENT " + whereClause + ")"; - String joinType = "LEFT "; - if (StringUtils.isNotEmpty(userIdFilterClause)) { - joinType = "INNER "; - userIdFilterClause = "AND " + userIdFilterClause; - } - - StringBuilder query = new StringBuilder("SELECT " + - " OBC.CONSENT_ID, " + - " RECEIPT, " + - " CLIENT_ID, " + - " CONSENT_TYPE, " + - " OBC.CURRENT_STATUS AS CURRENT_STATUS, " + - " CONSENT_FREQUENCY, " + - " VALIDITY_TIME, " + - " RECURRING_INDICATOR, " + - " OBC.CREATED_TIME AS CONSENT_CREATED_TIME, " + - " OBC.UPDATED_TIME AS CONSENT_UPDATED_TIME, " + - " String_agg(" + - " CA.att_key :: varchar, '||' order by CA.att_key :: varchar" + - " ) AS ATT_KEY," + - " String_agg(" + - " CA.att_value :: varchar, '||' order by CA.att_key :: varchar" + - " ) AS ATT_VALUE," + - " (" + - " SELECT " + - " String_agg(" + - " OCAR2.auth_id:: varchar, '||' order by OCAR2.auth_id :: varchar" + - " )" + - " FROM " + - " OB_CONSENT_AUTH_RESOURCE OCAR2 " + - " WHERE " + - " OCAR2.consent_id = OBC.consent_id " + - " GROUP BY " + - " OCAR2.consent_id" + - " ) AS AUTH_ID, " + - " (" + - " SELECT " + - " String_agg(" + - " OCAR2.auth_status :: varchar, '||' order by OCAR2.auth_id :: varchar" + - " )" + - " FROM " + - " OB_CONSENT_AUTH_RESOURCE OCAR2 " + - " WHERE " + - " OCAR2.consent_id = OBC.consent_id " + - " GROUP BY " + - " OCAR2.consent_id" + - " ) AS AUTH_STATUS, " + - " (" + - " select" + - " String_agg(" + - " OCAR2.auth_type :: varchar, '||' order by OCAR2.auth_id :: varchar" + - " )" + - " FROM " + - " OB_CONSENT_AUTH_RESOURCE OCAR2 " + - " WHERE " + - " OCAR2.consent_id = OBC.consent_id " + - " GROUP BY " + - " OCAR2.consent_id" + - " ) AS AUTH_TYPE, " + - " (" + - " SELECT " + - " String_agg(" + - " OCAR2.updated_time :: varchar, '||' order by OCAR2.auth_id :: varchar" + - " )" + - " FROM " + - " OB_CONSENT_AUTH_RESOURCE OCAR2 " + - " WHERE " + - " OCAR2.consent_id = OBC.consent_id " + - " GROUP BY " + - " OCAR2.consent_id" + - " ) AS UPDATED_TIME, " + - " (" + - " SELECT " + - " String_agg(" + - " OCAR2.user_id :: varchar, '||' order by OCAR2.auth_id :: varchar" + - " )" + - " FROM " + - " OB_CONSENT_AUTH_RESOURCE OCAR2 " + - " WHERE " + - " OCAR2.consent_id = OBC.consent_id " + - " GROUP BY " + - " OCAR2.consent_id" + - " ) AS USER_ID, " + - " (" + - " SELECT " + - " String_agg(" + - " OCAR2.auth_id:: varchar, '||' order by OCM2.mapping_id :: varchar" + - " )" + - " FROM " + - " OB_CONSENT_MAPPING OCM2 " + - " JOIN OB_CONSENT_AUTH_RESOURCE OCAR2 ON OCAR2.auth_id = OCM2.auth_id " + - " WHERE " + - " OCAR2.consent_id = OBC.consent_id" + - " ) AS AUTH_MAPPING_ID, " + - " (" + - " SELECT " + - " String_agg(" + - " OCM2.account_id :: varchar, '||' order by OCM2.mapping_id :: varchar" + - " )" + - " FROM " + - " OB_CONSENT_MAPPING OCM2 " + - " JOIN OB_CONSENT_AUTH_RESOURCE OCAR2 ON OCAR2.auth_id = OCM2.auth_id " + - " WHERE " + - " OCAR2.consent_id = OBC.consent_id" + - " ) AS ACCOUNT_ID, " + - " (" + - " SELECT " + - " String_agg(" + - " OCM2.mapping_id :: varchar, '||' order by OCM2.mapping_id :: varchar" + - " )" + - " FROM " + - " OB_CONSENT_MAPPING OCM2 " + - " JOIN OB_CONSENT_AUTH_RESOURCE OCAR2 ON OCAR2.auth_id = OCM2.auth_id " + - " WHERE " + - " OCAR2.consent_id = OBC.consent_id" + - " ) AS MAPPING_ID, " + - " (" + - " SELECT " + - " String_agg(" + - " OCM2.mapping_status :: varchar, '||' order by OCM2.mapping_id :: varchar" + - " )" + - " FROM " + - " OB_CONSENT_MAPPING OCM2 " + - " JOIN OB_CONSENT_AUTH_RESOURCE OCAR2 ON OCAR2.auth_id = OCM2.auth_id " + - " WHERE " + - " OCAR2.consent_id = OBC.consent_id" + - " ) AS MAPPING_STATUS, " + - " (" + - " SELECT " + - " String_agg(" + - " OCM2.permission :: varchar, '||' order by OCM2.mapping_id :: varchar" + - " )" + - " FROM " + - " OB_CONSENT_MAPPING OCM2 " + - " JOIN OB_CONSENT_AUTH_RESOURCE OCAR2 ON OCAR2.auth_id = OCM2.auth_id " + - " WHERE " + - " OCAR2.consent_id = OBC.consent_id" + - " ) AS PERMISSION" + - " FROM " + - selectClause + - "AS OBC " + - "LEFT JOIN OB_CONSENT_ATTRIBUTE CA ON OBC.CONSENT_ID=CA.CONSENT_ID " + - joinType + "JOIN OB_CONSENT_AUTH_RESOURCE OCAR ON OBC.CONSENT_ID=OCAR.CONSENT_ID " - + userIdFilterClause + - "LEFT JOIN OB_CONSENT_MAPPING OCM ON OCAR.AUTH_ID=OCM.AUTH_ID WHERE " + - "(OBC.UPDATED_TIME >= COALESCE(?, OBC.UPDATED_TIME) " + - "AND OBC.UPDATED_TIME <= COALESCE(?, OBC.UPDATED_TIME)) " + - "group by OBC.CONSENT_ID," + - "OBC.RECEIPT," + - "OBC.CLIENT_ID," + - "OBC.CONSENT_TYPE," + - "CONSENT_FREQUENCY," + - "VALIDITY_TIME," + - "RECURRING_INDICATOR," + - "OBC.CURRENT_STATUS," + - "OBC.CREATED_TIME," + - "OBC.UPDATED_TIME " + - "ORDER BY OBC.UPDATED_TIME DESC"); - - if (shouldLimit && shouldOffset) { - query.append(" LIMIT ? OFFSET ? "); - } else if (shouldLimit) { - query.append(" LIMIT ? "); - } - - return query.toString(); - } - - /** - * SQL query for delete consent mapping by auth id. - * @param executeOnRetentionTables flag to execute on retention tables - * @return SQL query for delete consent mapping by auth id - */ - public String getDeleteConsentMappingByAuthIdPreparedStatement(boolean executeOnRetentionTables) { - - // table prefix is to execute on consent retention data (purged data) tables. (if enabled) - String tablePrefix = ""; - if (executeOnRetentionTables) { - tablePrefix = ConsentMgtDAOConstants.RETENTION_TABLE_NAME_PREFIX; - } - return "DELETE FROM " + tablePrefix + "ob_consent_mapping where MAPPING_ID in (SELECT MAPPING_ID FROM " + - tablePrefix + "ob_consent_mapping OBCM INNER JOIN " + tablePrefix + "ob_consent_auth_resource OBAR " + - "ON OBCM.AUTH_ID = OBAR.AUTH_ID WHERE OBAR.CONSENT_ID = ?)"; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/utils/ConsentDAOUtils.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/utils/ConsentDAOUtils.java deleted file mode 100644 index 5974f4cb..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/dao/utils/ConsentDAOUtils.java +++ /dev/null @@ -1,276 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.utils; - -import com.wso2.openbanking.accelerator.consent.mgt.dao.constants.ConsentMgtDAOConstants; -import org.apache.commons.collections.CollectionUtils; -import org.apache.commons.collections.MapUtils; -import org.apache.commons.lang3.StringUtils; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.TreeMap; - -/** - * Utils class for consent module. - */ -public class ConsentDAOUtils { - - private static final String SPACE = " "; - private static final String COMMA = ","; - private static final String QUOTE = "\'"; - private static final String PLACEHOLDER = "?"; - private static final String LEFT_PARENTHESIS = "("; - private static final String RIGHT_PARENTHESIS = ")"; - private static final Map DB_OPERATORS_MAP = new HashMap() { - { - put(ConsentMgtDAOConstants.IN, "IN"); - put(ConsentMgtDAOConstants.AND, "AND"); - put(ConsentMgtDAOConstants.OR, "OR"); - put(ConsentMgtDAOConstants.WHERE, "WHERE"); - put(ConsentMgtDAOConstants.PLACEHOLDER, "?,"); - put(ConsentMgtDAOConstants.PLAIN_PLACEHOLDER, "?"); - put(ConsentMgtDAOConstants.EQUALS, "="); - } - }; - - public static String constructConsentSearchPreparedStatement(Map applicableConditions) { - - StringBuilder placeHoldersBuilder = new StringBuilder(); - StringBuilder whereClauseBuilder = new StringBuilder(); - whereClauseBuilder.append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.WHERE)); - // If all lists are empty or null, return the default term "where" - if (MapUtils.isEmpty(applicableConditions)) { - return ""; - } - for (Map.Entry entry : applicableConditions.entrySet()) { - // Oracle only allows 1000 values to be used in a SQL "IN" clause. Since more than 1000 consent IDs - // are used in some queries, "OR" clause is used - if (entry.getKey().contains("CONSENT_ID")) { - for (int i = 0; i < entry.getValue().size(); i++) { - whereClauseBuilder - .append(SPACE) - .append(entry.getKey()) - .append(SPACE) - .append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.EQUALS)) - .append(SPACE) - .append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.PLAIN_PLACEHOLDER)) - .append(SPACE) - .append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.OR)); - } - // Delete last OR from the statement - whereClauseBuilder.replace(whereClauseBuilder.length() - 2, - whereClauseBuilder.length(), DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.AND)); - } else { - for (int i = 0; i < entry.getValue().size(); i++) { - placeHoldersBuilder.append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.PLACEHOLDER)); - } - String placeHolders = StringUtils.removeEnd(placeHoldersBuilder.toString(), COMMA); - whereClauseBuilder - .append(SPACE) - .append(entry.getKey()) - .append(SPACE) - .append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.IN)) - .append(LEFT_PARENTHESIS) - .append(placeHolders) - .append(RIGHT_PARENTHESIS) - .append(SPACE) - .append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.AND)); - // Delete all content from old string builder except the starting left parenthesis - placeHoldersBuilder.delete(0, placeHoldersBuilder.length()); - } - } - int size = whereClauseBuilder.length(); - //removing the last AND in the statement - whereClauseBuilder.replace(size - 3, size, ""); - return whereClauseBuilder.toString(); - } - - public static String constructUserIdListFilterCondition(Map userIds) { - - StringBuilder placeHoldersBuilder = new StringBuilder(); - StringBuilder userIdFilterBuilder = new StringBuilder(); - - for (Map.Entry entry : userIds.entrySet()) { - for (int i = 0; i < entry.getValue().size(); i++) { - placeHoldersBuilder.append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.PLACEHOLDER)); - } - String placeHolders = StringUtils.removeEnd(placeHoldersBuilder.toString(), COMMA); - userIdFilterBuilder - .append(SPACE) - .append(entry.getKey()) - .append(SPACE) - .append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.IN)) - .append(LEFT_PARENTHESIS) - .append(placeHolders) - .append(RIGHT_PARENTHESIS) - .append(SPACE); - // Delete all content from old string builder except the starting left parenthesis - placeHoldersBuilder.delete(0, placeHoldersBuilder.length()); - } - return userIdFilterBuilder.toString(); - } - - public static String constructAuthSearchPreparedStatement(Map applicableConditions) { - - StringBuilder whereClauseBuilder = new StringBuilder(); - - // If all lists are empty or null, return the default term "where" - if (MapUtils.isEmpty(applicableConditions)) { - return whereClauseBuilder.toString(); - } - - whereClauseBuilder.append(SPACE).append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.WHERE)); - - int count = 0; - for (Map.Entry entry : applicableConditions.entrySet()) { - - if (count > 0) { - whereClauseBuilder.append(SPACE).append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.AND)); - } - whereClauseBuilder - .append(SPACE) - .append(entry.getKey()) - .append(SPACE) - .append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.EQUALS)) - .append(SPACE) - .append(PLACEHOLDER); - count++; - } - return whereClauseBuilder.toString(); - } - - /** - * Method to construct where clause for consent status audit search condition. - * @param consentIDs List of consent IDs - * @return Filter condition for consent status audit - */ - public static String constructConsentAuditRecordSearchPreparedStatement(ArrayList consentIDs) { - - StringBuilder whereClauseBuilder = new StringBuilder(); - if (!CollectionUtils.isEmpty(consentIDs)) { - whereClauseBuilder.append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.WHERE)); - for (int count = 0; count < consentIDs.size(); count++) { - whereClauseBuilder - .append(SPACE) - .append(ConsentMgtDAOConstants.CONSENT_ID) - .append(SPACE) - .append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.EQUALS)) - .append(SPACE) - .append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.PLAIN_PLACEHOLDER)) - .append(SPACE) - .append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.OR)); - } - // Delete last OR from the statement - whereClauseBuilder.replace(whereClauseBuilder.length() - 2, - whereClauseBuilder.length(), StringUtils.SPACE); - } - return whereClauseBuilder.toString(); - } - - public static TreeMap determineOrderOfParamsToSet(String preparedStatement, Map applicableConditionsMap, Map columnsMap) { - - int indexOfConsentIDsList; - int indexOfClientIdsList; - int indexOfConsentTypesList; - int indexOfConsentStatusesList; - int indexOfUserIDsList; - - // Tree map naturally sorts values in ascending order according to the key - TreeMap sortedIndexesMap = new TreeMap<>(); - - /* Check whether the where condition clauses are in the prepared statement and get the index if exists to - determine the order */ - if (preparedStatement.contains(columnsMap.get(ConsentMgtDAOConstants.CONSENT_IDS))) { - indexOfConsentIDsList = preparedStatement.indexOf(columnsMap.get(ConsentMgtDAOConstants.CONSENT_IDS)); - sortedIndexesMap.put(indexOfConsentIDsList, - applicableConditionsMap.get(columnsMap.get(ConsentMgtDAOConstants.CONSENT_IDS))); - } - if (preparedStatement.contains(columnsMap.get(ConsentMgtDAOConstants.CLIENT_IDS))) { - indexOfClientIdsList = preparedStatement.indexOf(columnsMap.get(ConsentMgtDAOConstants.CLIENT_IDS)); - sortedIndexesMap.put(indexOfClientIdsList, - applicableConditionsMap.get(columnsMap.get(ConsentMgtDAOConstants.CLIENT_IDS))); - } - if (preparedStatement.contains(columnsMap.get(ConsentMgtDAOConstants.CONSENT_TYPES))) { - indexOfConsentTypesList = preparedStatement.indexOf(columnsMap.get(ConsentMgtDAOConstants.CONSENT_TYPES)); - sortedIndexesMap.put(indexOfConsentTypesList, - applicableConditionsMap.get(columnsMap.get(ConsentMgtDAOConstants.CONSENT_TYPES))); - } - if (preparedStatement.contains(columnsMap.get(ConsentMgtDAOConstants.CONSENT_STATUSES))) { - indexOfConsentStatusesList = preparedStatement - .indexOf(columnsMap.get(ConsentMgtDAOConstants.CONSENT_STATUSES)); - sortedIndexesMap.put(indexOfConsentStatusesList, - applicableConditionsMap.get(columnsMap.get(ConsentMgtDAOConstants.CONSENT_STATUSES))); - } - if (preparedStatement.contains(columnsMap.get(ConsentMgtDAOConstants.USER_IDS))) { - indexOfUserIDsList = preparedStatement.indexOf(columnsMap.get(ConsentMgtDAOConstants.USER_IDS)); - sortedIndexesMap.put(indexOfUserIDsList, - applicableConditionsMap.get(columnsMap.get(ConsentMgtDAOConstants.USER_IDS))); - } - return sortedIndexesMap; - } - - /** - * Method to construct excluded statuses search condition. - * - * @param statusesEligibleForExpiration List of statuses eligible for expiration - * @return Filter condition for excluded statuses - */ - public static String constructStatusesEligibleForExpirationCondition(List statusesEligibleForExpiration) { - - StringBuilder placeHoldersBuilder = new StringBuilder(); - StringBuilder statusesEligibleForExpirationFilterBuilder = new StringBuilder(); - - for (int i = 0; i < statusesEligibleForExpiration.size(); i++) { - placeHoldersBuilder.append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.PLACEHOLDER)); - } - String placeHolders = StringUtils.removeEnd(placeHoldersBuilder.toString(), COMMA); - statusesEligibleForExpirationFilterBuilder - .append(SPACE) - .append(LEFT_PARENTHESIS) - .append(placeHolders) - .append(RIGHT_PARENTHESIS) - .append(SPACE); - // Delete all content from old string builder except the starting left parenthesis - placeHoldersBuilder.delete(0, placeHoldersBuilder.length()); - return statusesEligibleForExpirationFilterBuilder.toString(); - } - - public static String constructConsentHistoryPreparedStatement(int recordIdCount) { - - StringBuilder whereClauseBuilder = new StringBuilder(); - whereClauseBuilder.append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.WHERE)); - - for (int count = 0; count < recordIdCount; count++) { - whereClauseBuilder.append(SPACE) - .append(LEFT_PARENTHESIS) - .append(ConsentMgtDAOConstants.RECORD_ID) - .append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.EQUALS)) - .append(PLACEHOLDER) - .append(RIGHT_PARENTHESIS); - if (count < recordIdCount - 1) { - whereClauseBuilder.append(SPACE).append(DB_OPERATORS_MAP.get(ConsentMgtDAOConstants.OR)); - } - } - return whereClauseBuilder.toString(); - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/resources/findbugs-exclude.xml b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/resources/findbugs-exclude.xml deleted file mode 100644 index a16f7b50..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/resources/findbugs-exclude.xml +++ /dev/null @@ -1,43 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/resources/findbugs-include.xml b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/resources/findbugs-include.xml deleted file mode 100644 index 8932a22e..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/dao/impl/OBConsentMgtDAOTests.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/dao/impl/OBConsentMgtDAOTests.java deleted file mode 100644 index d6cdfd1f..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/dao/impl/OBConsentMgtDAOTests.java +++ /dev/null @@ -1,2546 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.impl; - -import com.wso2.openbanking.accelerator.consent.mgt.dao.ConsentCoreDAO; -import com.wso2.openbanking.accelerator.consent.mgt.dao.constants.ConsentMgtDAOConstants; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataDeletionException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataInsertionException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataRetrievalException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataUpdationException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentAttributes; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentFile; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentHistoryResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentMappingResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentStatusAuditRecord; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.queries.ConsentMgtCommonDBQueries; -import com.wso2.openbanking.accelerator.consent.mgt.dao.util.ConsentMgtDAOTestData; -import com.wso2.openbanking.accelerator.consent.mgt.dao.util.DAOUtils; -import org.mockito.Mockito; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.DataProvider; -import org.testng.annotations.Test; - -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.List; -import java.util.Map; -import java.util.UUID; - -/** - * Open banking consent management DAO tests. - */ -public class OBConsentMgtDAOTests { - - private static final String DB_NAME = "CONSENT_DB"; - - private ConsentCoreDAO consentCoreDAO; - private Connection mockedConnection; - private PreparedStatement mockedPreparedStatement; - private ResultSet mockedResultSet; - - @BeforeClass - public void initTest() throws Exception { - - DAOUtils.initializeDataSource(DB_NAME, DAOUtils.getFilePath("dbScripts/h2.sql")); - consentCoreDAO = new ConsentCoreDAOImpl(new ConsentMgtCommonDBQueries()); - mockedConnection = Mockito.mock(Connection.class); - mockedPreparedStatement = Mockito.mock(PreparedStatement.class); - mockedResultSet = Mockito.mock(ResultSet.class); - } - - @DataProvider(name = "storeConsentDataProvider") - public Object[][] storeConsentResourceData() { - - /* - * consentID - * clientID - * receipt - * consentType - * consentFrequency - * validityPeriod - * recurringIndicator - * currentStatus - */ - return ConsentMgtDAOTestData.DataProviders.CONSENT_RESOURCE_DATA_HOLDER; - } - - @Test(dataProvider = "storeConsentDataProvider") - public void testStoreConsentResource(String clientID, String receipt, String consentType, - int consentFrequency, long validityPeriod, boolean recurringIndicator, - String currentStatus) throws Exception { - - ConsentResource storedConsentResource; - ConsentResource consentResource = new ConsentResource(); - consentResource.setReceipt(receipt); - consentResource.setClientID(clientID); - consentResource.setConsentType(consentType); - consentResource.setCurrentStatus(currentStatus); - consentResource.setConsentFrequency(consentFrequency); - consentResource.setValidityPeriod(validityPeriod); - consentResource.setConsentID(UUID.randomUUID().toString()); - consentResource.setRecurringIndicator(true); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - } - Assert.assertNotNull(storedConsentResource); - Assert.assertNotNull(storedConsentResource.getConsentID()); - Assert.assertNotNull(storedConsentResource.getClientID()); - Assert.assertNotNull(storedConsentResource.getConsentType()); - Assert.assertEquals(consentFrequency, storedConsentResource.getConsentFrequency()); - Assert.assertNotNull(storedConsentResource.getValidityPeriod()); - Assert.assertTrue(storedConsentResource.isRecurringIndicator()); - Assert.assertNotNull(storedConsentResource.getCreatedTime()); - Assert.assertNotNull(storedConsentResource.getCurrentStatus()); - } - - @Test (expectedExceptions = OBConsentDataInsertionException.class) - public void testStoreConsentResourceInsertionError() throws Exception { - - Mockito.doReturn(Mockito.mock(PreparedStatement.class)).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(0).when(Mockito.mock(PreparedStatement.class)).executeUpdate(); - - consentCoreDAO.storeConsentResource(mockedConnection, ConsentMgtDAOTestData.getSampleTestConsentResource()); - } - - @Test (expectedExceptions = OBConsentDataInsertionException.class) - public void testStoreConsentResourceSQLError() throws Exception { - - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.storeConsentResource(mockedConnection, ConsentMgtDAOTestData.getSampleTestConsentResource()); - } - - @DataProvider(name = "storeAuthorizationDataProvider") - public Object[][] storeAuthorizationResourceData() { - - /* - * authorizationID - * consentID - * authorizationType - * userID - * authorizationStatus - */ - return ConsentMgtDAOTestData.DataProviders.AUTHORIZATION_RESOURCE_DATA_HOLDER; - } - - @Test (dataProvider = "storeAuthorizationDataProvider") - public void testStoreAuthorizationResource(String authorizationType, - String userID, String authorizationStatus) throws Exception { - - ConsentResource storedConsentResource; - AuthorizationResource storedAuthorizationResource; - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - AuthorizationResource authorizationResource = new AuthorizationResource(); - authorizationResource.setConsentID(storedConsentResource.getConsentID()); - authorizationResource.setAuthorizationType(authorizationType); - authorizationResource.setUserID(userID); - authorizationResource.setAuthorizationStatus(authorizationStatus); - - storedAuthorizationResource = consentCoreDAO.storeAuthorizationResource(connection, - authorizationResource); - } - Assert.assertNotNull(storedAuthorizationResource.getConsentID()); - Assert.assertNotNull(storedAuthorizationResource.getAuthorizationType()); - Assert.assertNotNull(storedAuthorizationResource.getUserID()); - Assert.assertNotNull(storedAuthorizationResource.getAuthorizationStatus()); - Assert.assertNotNull(storedAuthorizationResource.getUpdatedTime()); - Assert.assertNotNull(storedAuthorizationResource.getAuthorizationID()); - } - - @Test (expectedExceptions = OBConsentDataInsertionException.class) - public void testStoreAuthorizationResourceInsertionError() throws Exception { - - ConsentResource storedConsentResource = ConsentMgtDAOTestData.getSampleStoredTestConsentResource(); - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(0).when(mockedPreparedStatement).executeUpdate(); - - consentCoreDAO.storeAuthorizationResource(mockedConnection, ConsentMgtDAOTestData. - getSampleTestAuthorizationResource(storedConsentResource.getConsentID())); - } - - @Test (expectedExceptions = OBConsentDataInsertionException.class) - public void testStoreAuthorizationResourceSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - - consentCoreDAO.storeAuthorizationResource(mockedConnection, ConsentMgtDAOTestData. - getSampleTestAuthorizationResource(Mockito.anyString())); - } - - @DataProvider(name = "storeConsentMappingDataProvider") - public Object[][] storeConsentMappingResourceData() { - - /* - * accountID - * permission - * mappingStatus - */ - return ConsentMgtDAOTestData.DataProviders.CONSENT_MAPPING_RESOURCE_DATA_HOLDER; - } - - @Test (dataProvider = "storeConsentMappingDataProvider") - public void testStoreConsentMappingResource(String accountID, String permission, - String mappingStatus) throws Exception { - - ConsentResource consentResource; - AuthorizationResource authorizationResource; - ConsentMappingResource consentMappingResource; - ConsentResource storedConsentResource; - AuthorizationResource storedAuthorizationResource; - ConsentMappingResource storedConsentMappingResource; - - consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - authorizationResource = ConsentMgtDAOTestData - .getSampleTestAuthorizationResource(storedConsentResource.getConsentID()); - - storedAuthorizationResource = consentCoreDAO.storeAuthorizationResource(connection, - authorizationResource); - - consentMappingResource = new ConsentMappingResource(); - consentMappingResource.setAuthorizationID(storedAuthorizationResource.getAuthorizationID()); - consentMappingResource.setAccountID(accountID); - consentMappingResource.setPermission(permission); - consentMappingResource.setMappingStatus(mappingStatus); - - storedConsentMappingResource = consentCoreDAO.storeConsentMappingResource(connection, - consentMappingResource); - } - Assert.assertNotNull(storedConsentMappingResource.getMappingID()); - Assert.assertNotNull(storedConsentMappingResource.getAuthorizationID()); - Assert.assertNotNull(storedConsentMappingResource.getAccountID()); - Assert.assertNotNull(storedConsentMappingResource.getPermission()); - Assert.assertNotNull(storedConsentMappingResource.getMappingStatus()); - } - - @Test(dataProvider = "storeConsentMappingDataProvider") - public void testStoreConsentMappingResourceWithID(String accountID, String permission, - String mappingStatus) throws Exception { - - ConsentResource consentResource; - AuthorizationResource authorizationResource; - ConsentMappingResource consentMappingResource; - ConsentResource storedConsentResource; - AuthorizationResource storedAuthorizationResource; - ConsentMappingResource storedConsentMappingResource; - - consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - authorizationResource = ConsentMgtDAOTestData - .getSampleTestAuthorizationResource(storedConsentResource.getConsentID()); - authorizationResource.setAuthorizationID("db0b943d-38e2-47e4-bb78-8a242d279b5a"); - authorizationResource.setUpdatedTime(1669917425); - storedAuthorizationResource = consentCoreDAO.storeAuthorizationResource(connection, - authorizationResource); - - consentMappingResource = new ConsentMappingResource(); - consentMappingResource.setAuthorizationID(storedAuthorizationResource.getAuthorizationID()); - consentMappingResource.setAccountID(accountID); - consentMappingResource.setMappingID("aa4c943d-38e2-47e5-bb78-8a242d279b5a"); - consentMappingResource.setPermission(permission); - consentMappingResource.setMappingStatus(mappingStatus); - - storedConsentMappingResource = consentCoreDAO.storeConsentMappingResource(connection, - consentMappingResource); - } - Assert.assertTrue(storedConsentMappingResource.getMappingID().equals("aa4c943d-38e2-47e5-bb78-8a242d279b5a")); - Assert.assertTrue( - storedConsentMappingResource.getAuthorizationID().equals("db0b943d-38e2-47e4-bb78-8a242d279b5a")); - Assert.assertNotNull(storedConsentMappingResource.getAccountID()); - Assert.assertNotNull(storedConsentMappingResource.getPermission()); - Assert.assertNotNull(storedConsentMappingResource.getMappingStatus()); - } - - @Test(expectedExceptions = OBConsentDataInsertionException.class) - public void testStoreConsentMappingResourceInsertionError() throws Exception { - - ConsentMappingResource sampleConsentMappingResource = - ConsentMgtDAOTestData.getSampleTestConsentMappingResource(ConsentMgtDAOTestData - .getSampleStoredTestAuthorizationResource().getAuthorizationID()); - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(0).when(mockedPreparedStatement).executeUpdate(); - - consentCoreDAO.storeConsentMappingResource(mockedConnection, sampleConsentMappingResource); - } - - @Test (expectedExceptions = OBConsentDataInsertionException.class) - public void testStoreConsentMappingResourceSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.storeConsentMappingResource(mockedConnection, new ConsentMappingResource()); - } - - @Test(expectedExceptions = OBConsentDataInsertionException.class) - public void testStoreNullConsentMappingResource() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.storeConsentMappingResource(mockedConnection, null); - } - - @DataProvider(name = "storeConsentStatusAuditRecordDataProvider") - public Object[][] storeConsentStatusAuditRecordData() { - - /* - * currentStatus - * reason - * actionBy - * previousStatus - */ - return ConsentMgtDAOTestData.DataProviders.CONSENT_STATUS_AUDIT_RECORD_DATA_HOLDER; - } - - @Test (dataProvider = "storeConsentStatusAuditRecordDataProvider") - public void testStoreConsentStatusAuditRecord(String currentStatus, String reason, - String actionBy, String previousStatus) throws Exception { - - ConsentResource storedConsentResource; - ConsentStatusAuditRecord storedConsentStatusAuditRecord; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - ConsentStatusAuditRecord consentStatusAuditRecord = new ConsentStatusAuditRecord(); - consentStatusAuditRecord.setConsentID(storedConsentResource.getConsentID()); - consentStatusAuditRecord.setCurrentStatus(currentStatus); - consentStatusAuditRecord.setReason(reason); - consentStatusAuditRecord.setActionBy(actionBy); - consentStatusAuditRecord.setPreviousStatus(previousStatus); - - - storedConsentStatusAuditRecord = consentCoreDAO.storeConsentStatusAuditRecord(connection, - consentStatusAuditRecord); - } - Assert.assertNotNull(storedConsentStatusAuditRecord.getConsentID()); - Assert.assertNotNull(storedConsentStatusAuditRecord.getCurrentStatus()); - Assert.assertNotNull(storedConsentStatusAuditRecord.getReason()); - Assert.assertNotNull(storedConsentStatusAuditRecord.getActionBy()); - Assert.assertNotNull(storedConsentStatusAuditRecord.getPreviousStatus()); - Assert.assertNotNull(storedConsentStatusAuditRecord.getActionTime()); - Assert.assertNotNull(storedConsentStatusAuditRecord.getStatusAuditID()); - } - - @Test(dataProvider = "storeConsentStatusAuditRecordDataProvider") - public void testStoreConsentStatusAuditRecordWithConsentId(String currentStatus, String reason, - String actionBy, String previousStatus) - throws Exception { - - ConsentResource storedConsentResource; - ConsentStatusAuditRecord storedConsentStatusAuditRecord; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - consentResource.setConsentID("234ba17f-c3ac-4493-9049-d71f99c36dc2"); - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - ConsentStatusAuditRecord consentStatusAuditRecord = new ConsentStatusAuditRecord(); - consentStatusAuditRecord.setConsentID(storedConsentResource.getConsentID()); - consentStatusAuditRecord.setCurrentStatus(currentStatus); - consentStatusAuditRecord.setReason(reason); - consentStatusAuditRecord.setActionBy(actionBy); - consentStatusAuditRecord.setPreviousStatus(previousStatus); - consentStatusAuditRecord.setActionTime(1669917425); - - storedConsentStatusAuditRecord = consentCoreDAO.storeConsentStatusAuditRecord(connection, - consentStatusAuditRecord); - } - Assert.assertTrue( - storedConsentStatusAuditRecord.getConsentID().equals("234ba17f-c3ac-4493-9049-d71f99c36dc2")); - Assert.assertNotNull(storedConsentStatusAuditRecord.getCurrentStatus()); - Assert.assertNotNull(storedConsentStatusAuditRecord.getReason()); - Assert.assertNotNull(storedConsentStatusAuditRecord.getActionBy()); - Assert.assertNotNull(storedConsentStatusAuditRecord.getPreviousStatus()); - Assert.assertNotNull(storedConsentStatusAuditRecord.getActionTime()); - Assert.assertNotNull(storedConsentStatusAuditRecord.getStatusAuditID()); - } - - @Test (expectedExceptions = OBConsentDataInsertionException.class) - public void testStoreConsentStatusAuditRecordInsertionError() throws Exception { - - ConsentStatusAuditRecord sampleConsentStatusAuditRecord = ConsentMgtDAOTestData - .getSampleTestConsentStatusAuditRecord(ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, - ConsentMgtDAOTestData.SAMPLE_CURRENT_STATUS); - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(0).when(mockedPreparedStatement).executeUpdate(); - consentCoreDAO.storeConsentStatusAuditRecord(mockedConnection, sampleConsentStatusAuditRecord); - } - - @Test(expectedExceptions = OBConsentDataInsertionException.class) - public void testStoreNullConsentStatusAuditRecord() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(0).when(mockedPreparedStatement).executeUpdate(); - consentCoreDAO.storeConsentStatusAuditRecord(mockedConnection, null); - } - - @Test(expectedExceptions = OBConsentDataInsertionException.class) - public void testStoreConsentStatusAuditRecordSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.storeConsentStatusAuditRecord(mockedConnection, new ConsentStatusAuditRecord()); - } - - @DataProvider(name = "storeConsentAttributesDataProvider") - public Object[][] storeConsentAttributesData() { - - /* - * consentAttributesMap - */ - return ConsentMgtDAOTestData.DataProviders.CONSENT_ATTRIBUTES_DATA_HOLDER; - } - - @Test (dataProvider = "storeConsentAttributesDataProvider") - public void testStoreConsentAttributes(Map consentAttributes) throws Exception { - - ConsentResource storedConsentResource; - ConsentAttributes consentAttributesResource; - boolean isConsentAttributesStored; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - consentAttributesResource = new ConsentAttributes(); - consentAttributesResource.setConsentID(storedConsentResource.getConsentID()); - consentAttributesResource.setConsentAttributes(consentAttributes); - - isConsentAttributesStored = consentCoreDAO.storeConsentAttributes(connection, consentAttributesResource); - } - Assert.assertTrue(isConsentAttributesStored); - } - - @Test (expectedExceptions = OBConsentDataInsertionException.class) - public void testStoreConsentAttributesSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.storeConsentAttributes(mockedConnection, ConsentMgtDAOTestData - .getSampleTestConsentAttributesObject(ConsentMgtDAOTestData.getSampleStoredTestConsentResource() - .getConsentID())); - } - - @DataProvider(name = "storeConsentFileDataProvider") - public Object[][] storeConsentFileData() { - - /* - * consentFile - */ - return ConsentMgtDAOTestData.DataProviders.CONSENT_FILE_DATA_HOLDER; - } - - @Test (dataProvider = "storeConsentFileDataProvider") - public void testStoreConsentFile(String fileContent) throws Exception { - - ConsentResource storedConsentResource; - ConsentFile consentFileResource; - boolean isConsentFileStored; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - consentFileResource = new ConsentFile(); - consentFileResource.setConsentID(storedConsentResource.getConsentID()); - consentFileResource.setConsentFile(fileContent); - - isConsentFileStored = consentCoreDAO.storeConsentFile(connection, consentFileResource); - } - Assert.assertTrue(isConsentFileStored); - } - - @Test (expectedExceptions = OBConsentDataInsertionException.class) - public void testStoreConsentFileInsertionError() throws Exception { - - ConsentFile sampleConsentFileResource = - ConsentMgtDAOTestData.getSampleConsentFileObject(ConsentMgtDAOTestData.SAMPLE_CONSENT_FILE); - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(0).when(mockedPreparedStatement).executeUpdate(); - consentCoreDAO.storeConsentFile(mockedConnection, sampleConsentFileResource); - } - - @Test (expectedExceptions = OBConsentDataInsertionException.class) - public void testStoreConsentFileSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.storeConsentFile(mockedConnection, Mockito.anyObject()); - } - - @DataProvider(name = "updateConsentStatusDataProvider") - public Object[][] updateConsentStatusData() { - - /* - * newConsentStatus - */ - return ConsentMgtDAOTestData.DataProviders.CONSENT_STATUS_UPDATE_DATA_HOLDER; - } - - @Test (dataProvider = "updateConsentStatusDataProvider") - public void testUpdateConsentStatus(String newConsentStatus) throws Exception { - - ConsentResource storedConsentResource; - ConsentResource updatedConsentResource; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - updatedConsentResource = consentCoreDAO.updateConsentStatus(connection, - storedConsentResource.getConsentID(), newConsentStatus); - } - Assert.assertNotNull(updatedConsentResource.getConsentID()); - Assert.assertNotNull(updatedConsentResource.getCurrentStatus()); - } - - @Test (expectedExceptions = OBConsentDataUpdationException.class) - public void testUpdateConsentStatusSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.updateConsentStatus(mockedConnection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, - ConsentMgtDAOTestData.SAMPLE_CURRENT_STATUS); - } - - @Test (dataProvider = "updateConsentStatusDataProvider", expectedExceptions = OBConsentDataUpdationException.class) - public void testUpdateConsentStatusWithUnmatchedConsentID(String newConsentStatus) throws Exception { - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - consentCoreDAO.updateConsentStatus(connection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, - newConsentStatus); - } - } - - @DataProvider(name = "updateConsentMappingStatusDataProvider") - public Object[][] updateConsentMappingStatusData() { - - /* - * newMappingStatus - */ - return ConsentMgtDAOTestData.DataProviders.CONSENT_MAPPING_STATUS_UPDATE_DATA_HOLDER; - } - - @Test (dataProvider = "updateConsentMappingStatusDataProvider") - public void testUpdateConsentMappingStatus(String newMappingStatus) throws Exception { - - boolean isConsentMappingStatusUpdated; - ConsentResource storedConsentResource; - AuthorizationResource authorizationResource; - AuthorizationResource storedAuthorizationResource; - ConsentMappingResource consentMappingResource; - ConsentMappingResource storedConsentMappingResource; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - authorizationResource = ConsentMgtDAOTestData - .getSampleTestAuthorizationResource(storedConsentResource.getConsentID()); - - storedAuthorizationResource = consentCoreDAO.storeAuthorizationResource(connection, authorizationResource); - - consentMappingResource = - ConsentMgtDAOTestData - .getSampleTestConsentMappingResource(storedAuthorizationResource.getAuthorizationID()); - - storedConsentMappingResource = consentCoreDAO.storeConsentMappingResource(connection, - consentMappingResource); - - ArrayList mappingIDs = new ArrayList() { - { - add(storedConsentMappingResource.getMappingID()); - } - }; - - isConsentMappingStatusUpdated = consentCoreDAO.updateConsentMappingStatus(connection, mappingIDs, - newMappingStatus); - } - Assert.assertTrue(isConsentMappingStatusUpdated); - } - - @Test (expectedExceptions = OBConsentDataUpdationException.class) - public void testUpdateConsentMappingStatusSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.updateConsentMappingStatus(mockedConnection, ConsentMgtDAOTestData.UNMATCHED_MAPPING_IDS, - ConsentMgtDAOTestData.SAMPLE_MAPPING_STATUS); - } - - @DataProvider(name = "updateAuthorizationStatusDataProvider") - public Object[][] updateAuthorizationStatusData() { - - /* - * newAuthorizationStatus - */ - return ConsentMgtDAOTestData.DataProviders.CONSENT_AUTHORIZATION_STATUS_UPDATE_DATA_HOLDER; - } - - @Test (dataProvider = "updateAuthorizationStatusDataProvider") - public void testUpdateAuthorizationStatus(String newAuthorizationStatus) throws Exception { - - ConsentResource storedConsentResource; - AuthorizationResource authorizationResource; - AuthorizationResource storedAuthorizationResource; - AuthorizationResource updatedAuthorizationResource; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - authorizationResource = ConsentMgtDAOTestData - .getSampleTestAuthorizationResource(storedConsentResource.getConsentID()); - - storedAuthorizationResource = consentCoreDAO.storeAuthorizationResource(connection, authorizationResource); - - updatedAuthorizationResource = consentCoreDAO.updateAuthorizationStatus(connection, - storedAuthorizationResource.getAuthorizationID(), newAuthorizationStatus); - } - Assert.assertNotNull(updatedAuthorizationResource.getUpdatedTime()); - Assert.assertNotNull(updatedAuthorizationResource.getAuthorizationID()); - Assert.assertNotNull(updatedAuthorizationResource.getAuthorizationStatus()); - } - - @Test (expectedExceptions = OBConsentDataUpdationException.class) - public void testUpdateAuthorizationStatusSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.updateAuthorizationStatus(mockedConnection, ConsentMgtDAOTestData.SAMPLE_AUTHORIZATION_ID, - ConsentMgtDAOTestData.SAMPLE_AUTHORIZATION_STATUS); - } - - @Test (dataProvider = "updateAuthorizationStatusDataProvider", - expectedExceptions = OBConsentDataUpdationException.class) - public void testUpdateAuthorizationStatusWithUnmatchedAuthID(String newAuthorizationStatus) throws Exception { - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - consentCoreDAO.updateAuthorizationStatus(connection, - ConsentMgtDAOTestData.SAMPLE_AUTHORIZATION_ID, newAuthorizationStatus); - } - } - - @DataProvider(name = "updateAuthorizationUserDataProvider") - public Object[][] updateAuthorizationUsersData() { - - /* - * newAuthorizationUser - */ - return ConsentMgtDAOTestData.DataProviders.CONSENT_AUTHORIZATION_USER_UPDATE_DATA_HOLDER; - } - - @Test (dataProvider = "updateAuthorizationUserDataProvider") - public void testUpdateAuthorizationUser(String newUserID) throws Exception { - - ConsentResource storedConsentResource; - AuthorizationResource authorizationResource; - AuthorizationResource storedAuthorizationResource; - AuthorizationResource updatedAuthorizationResource; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - authorizationResource = ConsentMgtDAOTestData - .getSampleTestAuthorizationResource(storedConsentResource.getConsentID()); - - storedAuthorizationResource = consentCoreDAO.storeAuthorizationResource(connection, authorizationResource); - - updatedAuthorizationResource = consentCoreDAO.updateAuthorizationUser(connection, - storedAuthorizationResource.getAuthorizationID(), newUserID); - } - Assert.assertNotNull(updatedAuthorizationResource.getUserID()); - Assert.assertNotNull(updatedAuthorizationResource.getUpdatedTime()); - Assert.assertNotNull(updatedAuthorizationResource.getAuthorizationID()); - } - - @Test (expectedExceptions = OBConsentDataUpdationException.class) - public void testUpdateAuthorizationUserSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.updateAuthorizationUser(mockedConnection, ConsentMgtDAOTestData.SAMPLE_AUTHORIZATION_ID, - ConsentMgtDAOTestData.SAMPLE_USER_ID); - } - - @Test (dataProvider = "updateAuthorizationUserDataProvider", - expectedExceptions = OBConsentDataUpdationException.class) - public void testUpdateAuthorizationUserWithUnmatchedAuthID(String newUserID) throws Exception { - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - consentCoreDAO.updateAuthorizationUser(connection, ConsentMgtDAOTestData.SAMPLE_AUTHORIZATION_ID, - newUserID); - } - } - - @Test - public void testRetrieveConsentResource() throws Exception { - - ConsentResource storedConsentResource; - ConsentResource retrievedConsentResource; - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - retrievedConsentResource = consentCoreDAO.getConsentResource(connection, - storedConsentResource.getConsentID()); - } - - Assert.assertNotNull(retrievedConsentResource); - Assert.assertEquals(retrievedConsentResource.getConsentID(), storedConsentResource.getConsentID()); - Assert.assertNotNull(retrievedConsentResource.getConsentID()); - Assert.assertNotNull(retrievedConsentResource.getClientID()); - Assert.assertNotNull(retrievedConsentResource.getConsentType()); - Assert.assertEquals(consentResource.getConsentFrequency(), storedConsentResource.getConsentFrequency()); - Assert.assertNotNull(retrievedConsentResource.getValidityPeriod()); - Assert.assertTrue(retrievedConsentResource.isRecurringIndicator()); - Assert.assertNotNull(retrievedConsentResource.getCreatedTime()); - Assert.assertNotNull(retrievedConsentResource.getCurrentStatus()); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentResourceResultSetError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - consentCoreDAO.getConsentResource(mockedConnection, Mockito.anyString()); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentResourceSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.getConsentResource(mockedConnection, Mockito.anyObject()); - } - - @Test(expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentResourceWithUnmatchedConsentID() throws Exception { - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - consentCoreDAO.getConsentResource(connection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID); - } - } - - @Test - public void testRetrieveDetailedConsentResource() throws Exception { - - ConsentResource storedConsentResource; - AuthorizationResource storedAuthorizationResource; - ConsentMappingResource storedConsentMappingResource; - DetailedConsentResource retrievedDetailedConsentResource; - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - storedConsentResource = consentCoreDAO.storeConsentResource(connection, - ConsentMgtDAOTestData.getSampleTestConsentResource()); - consentCoreDAO.storeConsentAttributes(connection, - ConsentMgtDAOTestData.getSampleTestConsentAttributesObject(storedConsentResource.getConsentID())); - storedAuthorizationResource = consentCoreDAO.storeAuthorizationResource(connection, - ConsentMgtDAOTestData.getSampleTestAuthorizationResource(storedConsentResource.getConsentID())); - storedConsentMappingResource = consentCoreDAO.storeConsentMappingResource(connection, - ConsentMgtDAOTestData.getSampleTestConsentMappingResource(storedAuthorizationResource - .getAuthorizationID())); - retrievedDetailedConsentResource = consentCoreDAO.getDetailedConsentResource(connection, - storedConsentResource.getConsentID(), false); - } - - Assert.assertNotNull(retrievedDetailedConsentResource); - Assert.assertEquals(retrievedDetailedConsentResource.getConsentID(), storedConsentResource.getConsentID()); - Assert.assertEquals(retrievedDetailedConsentResource.getConsentID(), storedConsentResource.getConsentID()); - Assert.assertEquals(retrievedDetailedConsentResource.getClientID(), storedConsentResource.getClientID()); - Assert.assertEquals(retrievedDetailedConsentResource.getReceipt(), storedConsentResource.getReceipt()); - Assert.assertEquals(retrievedDetailedConsentResource.getConsentType(), storedConsentResource.getConsentType()); - Assert.assertEquals(retrievedDetailedConsentResource.getCurrentStatus(), - storedConsentResource.getCurrentStatus()); - Assert.assertEquals(retrievedDetailedConsentResource.getConsentFrequency(), - storedConsentResource.getConsentFrequency()); - Assert.assertEquals(retrievedDetailedConsentResource.getValidityPeriod(), - storedConsentResource.getValidityPeriod()); - Assert.assertEquals(retrievedDetailedConsentResource.isRecurringIndicator(), - storedConsentResource.isRecurringIndicator()); - Assert.assertNotNull(retrievedDetailedConsentResource.getConsentAttributes()); - Assert.assertEquals(retrievedDetailedConsentResource.getAuthorizationResources().get(0).getAuthorizationID(), - storedAuthorizationResource.getAuthorizationID()); - Assert.assertEquals(retrievedDetailedConsentResource.getConsentMappingResources().get(0).getMappingID(), - storedConsentMappingResource.getMappingID()); - } - - @Test - public void testRetrieveDetailedConsentResourceWithoutAttributes() throws Exception { - - ConsentResource storedConsentResource; - AuthorizationResource storedAuthorizationResource; - ConsentMappingResource storedConsentMappingResource; - DetailedConsentResource retrievedDetailedConsentResource; - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - storedConsentResource = consentCoreDAO.storeConsentResource(connection, - ConsentMgtDAOTestData.getSampleTestConsentResource()); - storedAuthorizationResource = consentCoreDAO.storeAuthorizationResource(connection, - ConsentMgtDAOTestData.getSampleTestAuthorizationResource(storedConsentResource.getConsentID())); - storedConsentMappingResource = consentCoreDAO.storeConsentMappingResource(connection, - ConsentMgtDAOTestData.getSampleTestConsentMappingResource(storedAuthorizationResource - .getAuthorizationID())); - retrievedDetailedConsentResource = consentCoreDAO.getDetailedConsentResource(connection, - storedConsentResource.getConsentID(), false); - } - - Assert.assertNotNull(retrievedDetailedConsentResource); - Assert.assertEquals(retrievedDetailedConsentResource.getConsentID(), storedConsentResource.getConsentID()); - Assert.assertEquals(retrievedDetailedConsentResource.getConsentID(), storedConsentResource.getConsentID()); - Assert.assertEquals(retrievedDetailedConsentResource.getClientID(), storedConsentResource.getClientID()); - Assert.assertEquals(retrievedDetailedConsentResource.getReceipt(), storedConsentResource.getReceipt()); - Assert.assertEquals(retrievedDetailedConsentResource.getConsentType(), storedConsentResource.getConsentType()); - Assert.assertEquals(retrievedDetailedConsentResource.getCurrentStatus(), - storedConsentResource.getCurrentStatus()); - Assert.assertEquals(retrievedDetailedConsentResource.getConsentFrequency(), - storedConsentResource.getConsentFrequency()); - Assert.assertEquals(retrievedDetailedConsentResource.getValidityPeriod(), - storedConsentResource.getValidityPeriod()); - Assert.assertEquals(retrievedDetailedConsentResource.isRecurringIndicator(), - storedConsentResource.isRecurringIndicator()); - Assert.assertNotNull(retrievedDetailedConsentResource.getConsentAttributes()); - Assert.assertEquals(retrievedDetailedConsentResource.getAuthorizationResources().get(0).getAuthorizationID(), - storedAuthorizationResource.getAuthorizationID()); - Assert.assertEquals(retrievedDetailedConsentResource.getConsentMappingResources().get(0).getMappingID(), - storedConsentMappingResource.getMappingID()); - } - - @Test - public void testRetrieveDetailedConsentResourceWithMultipleConsentAttributeKeys() throws Exception { - - ConsentResource storedConsentResource; - AuthorizationResource storedAuthorizationResourceOne; - AuthorizationResource storedAuthorizationResourceTwo; - DetailedConsentResource retrievedDetailedConsentResource; - String accountIdOne = "123456"; - String accountIdTwo = "789123"; - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - storedConsentResource = consentCoreDAO.storeConsentResource(connection, - ConsentMgtDAOTestData.getSampleTestConsentResource()); - consentCoreDAO.storeConsentAttributes(connection, - ConsentMgtDAOTestData.getSampleTestConsentAttributesObject(storedConsentResource.getConsentID())); - // create two auth resources for same consent id - storedAuthorizationResourceOne = consentCoreDAO.storeAuthorizationResource(connection, - ConsentMgtDAOTestData.getSampleTestAuthorizationResource(storedConsentResource.getConsentID())); - storedAuthorizationResourceTwo = consentCoreDAO.storeAuthorizationResource(connection, - ConsentMgtDAOTestData.getSampleTestAuthorizationResource(storedConsentResource.getConsentID())); - // create a total of three mapping resources for created auth resources - // mapping resources for first auth resource with two account ids - consentCoreDAO.storeConsentMappingResource(connection, - ConsentMgtDAOTestData - .getSampleTestConsentMappingResourceWithAccountId(storedAuthorizationResourceOne - .getAuthorizationID(), accountIdOne)); - consentCoreDAO.storeConsentMappingResource(connection, - ConsentMgtDAOTestData - .getSampleTestConsentMappingResourceWithAccountId(storedAuthorizationResourceOne - .getAuthorizationID(), accountIdTwo)); - // mapping resource for second auth resource with a single account id - consentCoreDAO.storeConsentMappingResource(connection, - ConsentMgtDAOTestData - .getSampleTestConsentMappingResourceWithAccountId(storedAuthorizationResourceTwo - .getAuthorizationID(), accountIdOne)); - retrievedDetailedConsentResource = consentCoreDAO.getDetailedConsentResource(connection, - storedConsentResource.getConsentID(), false); - } - - Assert.assertNotNull(retrievedDetailedConsentResource); - Assert.assertEquals(retrievedDetailedConsentResource.getConsentID(), storedConsentResource.getConsentID()); - Assert.assertEquals(retrievedDetailedConsentResource.getAuthorizationResources().get(0).getAuthorizationID(), - storedAuthorizationResourceOne.getAuthorizationID()); - Assert.assertEquals(retrievedDetailedConsentResource.getAuthorizationResources().get(1).getAuthorizationID(), - storedAuthorizationResourceTwo.getAuthorizationID()); - /* according to the created consent resource, retrieved consent resource should contain two auth resources and - three mapping resources - */ - Assert.assertEquals(retrievedDetailedConsentResource.getAuthorizationResources().size(), 2); - Assert.assertEquals(retrievedDetailedConsentResource.getConsentMappingResources().size(), 3); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveDetailedConsentResourceError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.getDetailedConsentResource(mockedConnection, Mockito.anyString(), false); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveDetailedConsentResourceRetrieveError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(mockedResultSet).when(mockedPreparedStatement).executeQuery(); - Mockito.doReturn(false).when(mockedResultSet).next(); - consentCoreDAO.getDetailedConsentResource(mockedConnection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, false); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveDetailedConsentResourceResultSetError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - consentCoreDAO.getDetailedConsentResource(mockedConnection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, false); - } - - @Test - public void testRetrieveConsentWithAttributesResource() throws Exception { - - ConsentAttributes consentAttributesResource; - ConsentResource retrievedConsentResource; - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - consentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - consentAttributesResource = ConsentMgtDAOTestData - .getSampleTestConsentAttributesObject(consentResource.getConsentID()); - consentCoreDAO.storeConsentAttributes(connection, consentAttributesResource); - - retrievedConsentResource = consentCoreDAO.getConsentResourceWithAttributes(connection, - consentResource.getConsentID()); - } - Assert.assertNotNull(retrievedConsentResource); - Assert.assertEquals(retrievedConsentResource.getConsentID(), consentResource.getConsentID()); - Assert.assertNotNull(retrievedConsentResource.getConsentID()); - Assert.assertNotNull(retrievedConsentResource.getClientID()); - Assert.assertNotNull(retrievedConsentResource.getConsentType()); - Assert.assertEquals(consentResource.getConsentFrequency(), retrievedConsentResource.getConsentFrequency()); - Assert.assertNotNull(retrievedConsentResource.getValidityPeriod()); - Assert.assertTrue(retrievedConsentResource.isRecurringIndicator()); - Assert.assertNotNull(retrievedConsentResource.getCreatedTime()); - Assert.assertNotNull(retrievedConsentResource.getCurrentStatus()); - Assert.assertNotNull(retrievedConsentResource.getConsentAttributes()); - - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentWithAttributesResourceResultRetrieveError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString(), Mockito.anyInt(), Mockito.anyInt()); - Mockito.doReturn(mockedResultSet).when(mockedPreparedStatement).executeQuery(); - Mockito.doReturn(false).when(mockedResultSet).next(); - consentCoreDAO.getConsentResourceWithAttributes(mockedConnection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentWithAttributesResourceResultSetError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString(), Mockito.anyInt(), Mockito.anyInt()); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - consentCoreDAO.getConsentResourceWithAttributes(mockedConnection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentWithAttributesResourceSQLError() throws Exception { - - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString(), - Mockito.anyInt(), Mockito.anyInt()); - consentCoreDAO.getConsentResourceWithAttributes(mockedConnection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID); - } - - @Test (dataProvider = "storeConsentFileDataProvider") - public void testRetrieveConsentFileResource(String consentFile) throws Exception { - - ConsentFile consentFileResource; - ConsentFile retrievedConsentFileResource; - ConsentResource retrievedConsentResource; - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - consentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - retrievedConsentResource = consentCoreDAO.getConsentResource(connection, consentResource.getConsentID()); - - consentFileResource = new ConsentFile(); - consentFileResource.setConsentID(retrievedConsentResource.getConsentID()); - consentFileResource.setConsentFile(consentFile); - - consentCoreDAO.storeConsentFile(connection, consentFileResource); - - retrievedConsentFileResource = consentCoreDAO.getConsentFile(connection, - consentFileResource.getConsentID(), false); - } - Assert.assertNotNull(retrievedConsentFileResource.getConsentID()); - Assert.assertNotNull(retrievedConsentFileResource.getConsentFile()); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentFileResourceNoRecordsFoundError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(mockedResultSet).when(mockedPreparedStatement).executeQuery(); - Mockito.doReturn(false).when(mockedResultSet).next(); - consentCoreDAO.getConsentFile(mockedConnection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, false); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentFileResourceSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.getConsentFile(mockedConnection, Mockito.anyObject(), false); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentFileResourceRetrieveError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - consentCoreDAO.getConsentFile(mockedConnection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, false); - } - - @Test(expectedExceptions = OBConsentDataRetrievalException.class, dataProvider = "storeConsentFileDataProvider") - public void testRetrieveConsentFileResourceWithUnmatchedConsentID(String consentFile) throws Exception { - - ConsentFile consentFileResource; - ConsentResource retrievedConsentResource; - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - consentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - retrievedConsentResource = consentCoreDAO.getConsentResource(connection, consentResource.getConsentID()); - - consentFileResource = new ConsentFile(); - consentFileResource.setConsentID(retrievedConsentResource.getConsentID()); - consentFileResource.setConsentFile(consentFile); - - consentCoreDAO.storeConsentFile(connection, consentFileResource); - consentCoreDAO.getConsentFile(connection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, false); - } - } - - @DataProvider(name = "getConsentAttributesDataProvider") - public Object[][] getConsentAttributesData() { - - /* - * consentAttributeKeys - */ - return ConsentMgtDAOTestData.DataProviders.CONSENT_ATTRIBUTES_GET_DATA_HOLDER; - } - - @Test (dataProvider = "getConsentAttributesDataProvider") - public void testRetrieveConsentAttributes(ArrayList consentAttributeKeys) throws Exception { - - ConsentAttributes consentAttributesResource; - ConsentAttributes retrievedConsentAttributesResource; - ConsentResource retrievedConsentResource; - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - consentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - retrievedConsentResource = consentCoreDAO.getConsentResource(connection, consentResource.getConsentID()); - - consentAttributesResource = ConsentMgtDAOTestData - .getSampleTestConsentAttributesObject(retrievedConsentResource.getConsentID()); - - consentCoreDAO.storeConsentAttributes(connection, consentAttributesResource); - - retrievedConsentAttributesResource = consentCoreDAO.getConsentAttributes(connection, - retrievedConsentResource.getConsentID(), consentAttributeKeys); - } - Assert.assertNotNull(retrievedConsentAttributesResource.getConsentID()); - Assert.assertNotNull(retrievedConsentAttributesResource.getConsentAttributes()); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentAttributesRetrieveError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(mockedResultSet).when(mockedPreparedStatement).executeQuery(); - Mockito.doReturn(false).when(mockedResultSet).isBeforeFirst(); - consentCoreDAO.getConsentAttributes(mockedConnection, Mockito.anyString()); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentAttributesResultSetError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - consentCoreDAO.getConsentAttributes(mockedConnection, Mockito.anyString()); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentAttributesSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.getConsentAttributes(mockedConnection, Mockito.anyObject()); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentAttributesResultSetErrorOverloadedMethod() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - consentCoreDAO.getConsentAttributes(mockedConnection, Mockito.anyString(), - ConsentMgtDAOTestData.SAMPLE_CONSENT_ATTRIBUTES_KEYS); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentAttributesSQLErrorOverloadedMethod() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.getConsentAttributes(mockedConnection, Mockito.anyObject()); - } - - @Test - public void testRetrieveConsentAttributesWithNoKeys() throws Exception { - - ConsentAttributes consentAttributesResource; - ConsentAttributes retrievedConsentAttributesResource; - ConsentResource retrievedConsentResource; - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - consentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - retrievedConsentResource = consentCoreDAO.getConsentResource(connection, consentResource.getConsentID()); - - consentAttributesResource = ConsentMgtDAOTestData - .getSampleTestConsentAttributesObject(retrievedConsentResource.getConsentID()); - - consentCoreDAO.storeConsentAttributes(connection, consentAttributesResource); - - retrievedConsentAttributesResource = consentCoreDAO.getConsentAttributes(connection, - retrievedConsentResource.getConsentID()); - } - Assert.assertNotNull(retrievedConsentAttributesResource.getConsentID()); - Assert.assertNotNull(retrievedConsentAttributesResource.getConsentAttributes()); - } - - @Test (dataProvider = "getConsentAttributesDataProvider", - expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentAttributesWithUnmatchedConsentID(ArrayList consentAttributeKeys) - throws Exception { - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - consentCoreDAO.getConsentAttributes(connection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, - consentAttributeKeys); - } - } - - @Test - public void testRetrieveConsentAttributesByName() throws Exception { - - ConsentAttributes consentAttributesResource; - Map retrievedValuesMap; - ConsentResource retrievedConsentResource; - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - consentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - retrievedConsentResource = consentCoreDAO.getConsentResource(connection, consentResource.getConsentID()); - - consentAttributesResource = ConsentMgtDAOTestData - .getSampleTestConsentAttributesObject(retrievedConsentResource.getConsentID()); - - consentCoreDAO.storeConsentAttributes(connection, consentAttributesResource); - - retrievedValuesMap = consentCoreDAO.getConsentAttributesByName(connection, - "x-request-id"); - - } - Assert.assertTrue(retrievedValuesMap.containsKey(consentAttributesResource.getConsentID())); - Assert.assertTrue(retrievedValuesMap.containsValue(consentAttributesResource.getConsentAttributes().get("x" + - "-request-id"))); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentAttributesByNameSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.getConsentAttributesByName(mockedConnection, Mockito.anyObject()); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentAttributesByNameResultSetError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - consentCoreDAO.getConsentAttributesByName(mockedConnection, Mockito.anyString()); - } - - @Test - public void testRetrieveAuthorizationResource() throws Exception { - - ConsentResource storedConsentResource; - AuthorizationResource authorizationResource; - AuthorizationResource storedAuthorizationResource; - AuthorizationResource retrievedAuthorizationResource; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - authorizationResource = ConsentMgtDAOTestData - .getSampleTestAuthorizationResource(storedConsentResource.getConsentID()); - - storedAuthorizationResource = consentCoreDAO.storeAuthorizationResource(connection, authorizationResource); - - retrievedAuthorizationResource = consentCoreDAO.getAuthorizationResource(connection, - storedAuthorizationResource.getAuthorizationID()); - } - Assert.assertNotNull(retrievedAuthorizationResource.getUpdatedTime()); - Assert.assertNotNull(retrievedAuthorizationResource.getAuthorizationID()); - Assert.assertNotNull(retrievedAuthorizationResource.getAuthorizationStatus()); - Assert.assertNotNull(retrievedAuthorizationResource.getUserID()); - Assert.assertNotNull(retrievedAuthorizationResource.getAuthorizationType()); - Assert.assertEquals(retrievedAuthorizationResource.getConsentID(), storedAuthorizationResource.getConsentID()); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveAuthorizationResourceResultSetError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - consentCoreDAO.getAuthorizationResource(mockedConnection, ConsentMgtDAOTestData.SAMPLE_AUTHORIZATION_ID); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveAuthorizationResourceSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.getAuthorizationResource(mockedConnection, ConsentMgtDAOTestData.SAMPLE_AUTHORIZATION_ID); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveAuthorizationResourceWithUnmatchedAuthID() throws Exception { - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - consentCoreDAO.getAuthorizationResource(connection, - ConsentMgtDAOTestData.SAMPLE_AUTHORIZATION_ID); - } - } - - @Test - public void testRetrieveConsentStatusAuditRecordsWithConsentID() throws Exception { - - ConsentResource storedConsentResource; - ConsentStatusAuditRecord consentStatusAuditRecord; - ConsentStatusAuditRecord storedConsentStatusAuditRecord; - ArrayList retrievedConsentStatusAuditRecords; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - consentStatusAuditRecord = ConsentMgtDAOTestData - .getSampleTestConsentStatusAuditRecord(storedConsentResource.getConsentID(), - storedConsentResource.getCurrentStatus()); - - storedConsentStatusAuditRecord = consentCoreDAO.storeConsentStatusAuditRecord(connection, - consentStatusAuditRecord); - - connection.commit(); - - retrievedConsentStatusAuditRecords = consentCoreDAO.getConsentStatusAuditRecords(connection, - storedConsentStatusAuditRecord.getConsentID(), null, - null, null, - null, null , false); - } - Assert.assertNotNull(retrievedConsentStatusAuditRecords); - for (ConsentStatusAuditRecord record: - retrievedConsentStatusAuditRecords) { - Assert.assertEquals(storedConsentStatusAuditRecord.getConsentID(), record.getConsentID()); - } - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentStatusAuditRecordsResultSetError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - consentCoreDAO.getConsentStatusAuditRecords(mockedConnection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, - ConsentMgtDAOTestData.SAMPLE_CURRENT_STATUS, ConsentMgtDAOTestData.SAMPLE_ACTION_BY, - ConsentMgtDAOTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtDAOTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, ConsentMgtDAOTestData.SAMPLE_AUDIT_ID, false); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentStatusAuditRecordsSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.getConsentStatusAuditRecords(mockedConnection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, - ConsentMgtDAOTestData.SAMPLE_CURRENT_STATUS, ConsentMgtDAOTestData.SAMPLE_ACTION_BY, - ConsentMgtDAOTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtDAOTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, ConsentMgtDAOTestData.SAMPLE_AUDIT_ID, false); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentStatusAuditRecordByConsentIDWithUnmatchedConsentID() throws Exception { - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - consentCoreDAO.getConsentStatusAuditRecords(connection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, - null, null, null, null, null, false); - } - } - - @Test - public void testRetrieveConsentStatusAuditRecordsByConsentIDAndStatus() throws Exception { - - ConsentResource storedConsentResource; - ConsentStatusAuditRecord consentStatusAuditRecord; - ConsentStatusAuditRecord storedConsentStatusAuditRecord; - ArrayList retrievedConsentStatusAuditRecords; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - consentStatusAuditRecord = ConsentMgtDAOTestData - .getSampleTestConsentStatusAuditRecord(storedConsentResource.getConsentID(), - storedConsentResource.getCurrentStatus()); - - storedConsentStatusAuditRecord = consentCoreDAO.storeConsentStatusAuditRecord(connection, - consentStatusAuditRecord); - - retrievedConsentStatusAuditRecords = consentCoreDAO.getConsentStatusAuditRecords(connection, - storedConsentStatusAuditRecord.getConsentID(), storedConsentStatusAuditRecord.getCurrentStatus(), - null, null, null, null, false); - } - Assert.assertNotNull(retrievedConsentStatusAuditRecords); - for (ConsentStatusAuditRecord record: - retrievedConsentStatusAuditRecords) { - Assert.assertEquals(storedConsentStatusAuditRecord.getConsentID(), record.getConsentID()); - Assert.assertEquals(storedConsentStatusAuditRecord.getCurrentStatus(), record.getCurrentStatus()); - } - } - - @Test - public void testRetrieveConsentStatusAuditRecordsByConsentIDStatusAndActionBy() throws Exception { - - ConsentResource storedConsentResource; - ConsentStatusAuditRecord consentStatusAuditRecord; - ConsentStatusAuditRecord storedConsentStatusAuditRecord; - ArrayList retrievedConsentStatusAuditRecords; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - consentStatusAuditRecord = ConsentMgtDAOTestData - .getSampleTestConsentStatusAuditRecord(storedConsentResource.getConsentID(), - storedConsentResource.getCurrentStatus()); - - storedConsentStatusAuditRecord = consentCoreDAO.storeConsentStatusAuditRecord(connection, - consentStatusAuditRecord); - - retrievedConsentStatusAuditRecords = consentCoreDAO.getConsentStatusAuditRecords(connection, - storedConsentStatusAuditRecord.getConsentID(), storedConsentStatusAuditRecord.getCurrentStatus(), - storedConsentStatusAuditRecord.getActionBy(), null, - null, null, false); - } - Assert.assertNotNull(retrievedConsentStatusAuditRecords); - for (ConsentStatusAuditRecord record: - retrievedConsentStatusAuditRecords) { - Assert.assertEquals(storedConsentStatusAuditRecord.getConsentID(), record.getConsentID()); - Assert.assertEquals(storedConsentStatusAuditRecord.getCurrentStatus(), record.getCurrentStatus()); - Assert.assertEquals(storedConsentStatusAuditRecord.getActionBy(), record.getActionBy()); - } - } - - @Test - public void testRetrieveConsentAuditRecordByAuditRecordID() throws Exception { - - ConsentResource storedConsentResource; - ConsentStatusAuditRecord consentStatusAuditRecord; - ConsentStatusAuditRecord storedConsentStatusAuditRecord; - ArrayList retrievedConsentStatusAuditRecords; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - consentStatusAuditRecord = ConsentMgtDAOTestData - .getSampleTestConsentStatusAuditRecord(storedConsentResource.getConsentID(), - storedConsentResource.getCurrentStatus()); - - storedConsentStatusAuditRecord = consentCoreDAO.storeConsentStatusAuditRecord(connection, - consentStatusAuditRecord); - - retrievedConsentStatusAuditRecords = consentCoreDAO.getConsentStatusAuditRecords(connection, - null, null, null, null, null, - storedConsentStatusAuditRecord.getStatusAuditID(), false); - } - Assert.assertNotNull(retrievedConsentStatusAuditRecords); - for (ConsentStatusAuditRecord record: - retrievedConsentStatusAuditRecords) { - Assert.assertEquals(storedConsentStatusAuditRecord.getConsentID(), record.getConsentID()); - Assert.assertEquals(storedConsentStatusAuditRecord.getStatusAuditID(), record.getStatusAuditID()); - } - } - - @Test - public void testRetrieveConsentAuditRecordForGivenTime() throws Exception { - - long fromTime; - long toTime; - ConsentResource storedConsentResource; - ConsentStatusAuditRecord consentStatusAuditRecord; - ConsentStatusAuditRecord storedConsentStatusAuditRecord; - ArrayList retrievedConsentStatusAuditRecords; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - consentStatusAuditRecord = ConsentMgtDAOTestData - .getSampleTestConsentStatusAuditRecord(storedConsentResource.getConsentID(), - storedConsentResource.getCurrentStatus()); - - storedConsentStatusAuditRecord = consentCoreDAO.storeConsentStatusAuditRecord(connection, - consentStatusAuditRecord); - - fromTime = Long.sum(storedConsentStatusAuditRecord.getActionTime(), -60); - toTime = Long.sum(storedConsentStatusAuditRecord.getActionTime(), 60); - - retrievedConsentStatusAuditRecords = consentCoreDAO.getConsentStatusAuditRecords(connection, - storedConsentStatusAuditRecord.getConsentID(), null, - null, fromTime, toTime, null, false); - } - Assert.assertNotNull(retrievedConsentStatusAuditRecords); - for (ConsentStatusAuditRecord record: - retrievedConsentStatusAuditRecords) { - Assert.assertEquals(storedConsentResource.getConsentID(), record.getConsentID()); - Assert.assertTrue((record.getActionTime() >= fromTime) && (record.getActionTime() <= toTime)); - } - } - - @Test - public void testRetrieveConsentMappingResource() throws Exception { - - ConsentResource storedConsentResource; - AuthorizationResource authorizationResource; - AuthorizationResource storedAuthorizationResource; - ConsentMappingResource consentMappingResource; - ArrayList retrievedConsentMappingResources; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - authorizationResource = ConsentMgtDAOTestData - .getSampleTestAuthorizationResource(storedConsentResource.getConsentID()); - - storedAuthorizationResource = consentCoreDAO.storeAuthorizationResource(connection, - authorizationResource); - - consentMappingResource = - ConsentMgtDAOTestData - .getSampleTestConsentMappingResource(storedAuthorizationResource.getAuthorizationID()); - - consentCoreDAO.storeConsentMappingResource(connection, - consentMappingResource); - - retrievedConsentMappingResources = consentCoreDAO.getConsentMappingResources(connection, - storedAuthorizationResource.getAuthorizationID()); - } - Assert.assertNotNull(retrievedConsentMappingResources); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentMappingResourceResultSetError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - consentCoreDAO.getConsentMappingResources(mockedConnection, ConsentMgtDAOTestData.SAMPLE_AUTHORIZATION_ID); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentMappingResourceSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.getConsentMappingResources(mockedConnection, ConsentMgtDAOTestData.SAMPLE_AUTHORIZATION_ID); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentMappingResourceRetrieveErrorOverloadedMethod() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(mockedResultSet).when(mockedPreparedStatement).executeQuery(); - Mockito.doReturn(false).when(mockedResultSet).isBeforeFirst(); - consentCoreDAO.getConsentMappingResources(mockedConnection, Mockito.anyString(), - ConsentMgtDAOTestData.SAMPLE_MAPPING_STATUS); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentMappingResourceResultSetErrorOverloadedMethod() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - consentCoreDAO.getConsentMappingResources(mockedConnection, ConsentMgtDAOTestData.SAMPLE_AUTHORIZATION_ID, - ConsentMgtDAOTestData.SAMPLE_MAPPING_STATUS); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentMappingResourceSQLErrorOverloadedMethod() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.getConsentMappingResources(mockedConnection, ConsentMgtDAOTestData.SAMPLE_AUTHORIZATION_ID, - ConsentMgtDAOTestData.SAMPLE_MAPPING_STATUS); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentMappingResourceWithUnmatchedAuthID() throws Exception { - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - consentCoreDAO.getConsentMappingResources(connection, - ConsentMgtDAOTestData.SAMPLE_AUTHORIZATION_ID); - } - } - - @Test - public void testRetrieveConsentMappingResourceWithMappingStatus() throws Exception { - - ConsentResource storedConsentResource; - AuthorizationResource authorizationResource; - AuthorizationResource storedAuthorizationResource; - ConsentMappingResource storedConsentMappingResource; - ConsentMappingResource consentMappingResource; - ArrayList retrievedConsentMappingResources; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - authorizationResource = ConsentMgtDAOTestData - .getSampleTestAuthorizationResource(storedConsentResource.getConsentID()); - - storedAuthorizationResource = consentCoreDAO.storeAuthorizationResource(connection, - authorizationResource); - - consentMappingResource = - ConsentMgtDAOTestData - .getSampleTestConsentMappingResource(storedAuthorizationResource.getAuthorizationID()); - - storedConsentMappingResource = consentCoreDAO.storeConsentMappingResource(connection, - consentMappingResource); - - retrievedConsentMappingResources = consentCoreDAO.getConsentMappingResources(connection, - storedAuthorizationResource.getAuthorizationID(), storedConsentMappingResource.getMappingStatus()); - } - Assert.assertNotNull(retrievedConsentMappingResources); - } - - @Test - public void testDeleteConsentAttribute() throws Exception { - - ConsentResource storedConsentResource; - ConsentAttributes consentAttributesResource; - boolean isDeleted; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - consentAttributesResource = ConsentMgtDAOTestData - .getSampleTestConsentAttributesObject(storedConsentResource.getConsentID()); - - consentCoreDAO.storeConsentAttributes(connection, consentAttributesResource); - - isDeleted = consentCoreDAO.deleteConsentAttributes(connection, storedConsentResource.getConsentID(), - ConsentMgtDAOTestData.SAMPLE_CONSENT_ATTRIBUTES_KEYS); - } - Assert.assertTrue(isDeleted); - } - - @Test - public void testConsentSearchWithConsentIDsList() throws Exception { - - ArrayList detailedConsentResources; - ArrayList consentIDs = new ArrayList<>(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - storeDataForConsentSearchTest(consentIDs, connection); - detailedConsentResources = consentCoreDAO.searchConsents(connection, consentIDs, null, - null, null, null, null, null, - 10, 0); - } - - Assert.assertNotNull(detailedConsentResources); - for (DetailedConsentResource resource : detailedConsentResources) { - Assert.assertNotNull(resource.getAuthorizationResources()); - Assert.assertNotNull(resource.getConsentMappingResources()); - Assert.assertNotNull(resource.getConsentAttributes()); - - for (AuthorizationResource authResource : resource.getAuthorizationResources()) { - Assert.assertEquals(resource.getConsentID(), authResource.getConsentID()); - } - } - } - - @Test - public void testConsentSearchWithConsentIDsListAndTime() throws Exception { - - ArrayList detailedConsentResources; - ArrayList consentIDs = new ArrayList<>(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - storeDataForConsentSearchTest(consentIDs, connection); - detailedConsentResources = consentCoreDAO.searchConsents(connection, consentIDs, null, - null, null, null, 1669917425L, 1669917425L, - 10, 0); - } - - Assert.assertNotNull(detailedConsentResources); - - } - - @Test - public void testConsentSearchWithClientIDsList() throws Exception { - - ArrayList detailedConsentResources; - ArrayList consentIDs = new ArrayList<>(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - storeDataForConsentSearchTest(consentIDs, connection); - detailedConsentResources = consentCoreDAO.searchConsents(connection, null, - ConsentMgtDAOTestData.SAMPLE_CLIENT_IDS_LIST, null, null, null, - null, null, 10, 0); - } - - Assert.assertNotNull(detailedConsentResources); - for (DetailedConsentResource resource : detailedConsentResources) { - Assert.assertNotNull(resource.getAuthorizationResources()); - Assert.assertNotNull(resource.getConsentMappingResources()); - Assert.assertNotNull(resource.getConsentAttributes()); - - for (AuthorizationResource authResource : resource.getAuthorizationResources()) { - Assert.assertEquals(resource.getConsentID(), authResource.getConsentID()); - } - } - } - - @Test - public void testConsentSearchWithConsentStatusesList() throws Exception { - - ArrayList detailedConsentResources; - ArrayList consentIDs = new ArrayList<>(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - storeDataForConsentSearchTest(consentIDs, connection); - detailedConsentResources = consentCoreDAO.searchConsents(connection, null, null, - null, ConsentMgtDAOTestData.SAMPLE_CONSENT_STATUSES_LIST, null, null, - null, 10, 0); - } - - Assert.assertNotNull(detailedConsentResources); - for (DetailedConsentResource resource : detailedConsentResources) { - Assert.assertNotNull(resource.getAuthorizationResources()); - Assert.assertNotNull(resource.getConsentMappingResources()); - Assert.assertNotNull(resource.getConsentAttributes()); - - for (AuthorizationResource authResource : resource.getAuthorizationResources()) { - Assert.assertEquals(resource.getConsentID(), authResource.getConsentID()); - } - } - } - - @Test - public void testConsentSearchWithConsentTypesList() throws Exception { - - ArrayList detailedConsentResources; - ArrayList consentIDs = new ArrayList<>(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - storeDataForConsentSearchTest(consentIDs, connection); - detailedConsentResources = consentCoreDAO.searchConsents(connection, null, null, - null, ConsentMgtDAOTestData.SAMPLE_CONSENT_STATUSES_LIST, null, null, - null, 10, 0); - } - - Assert.assertNotNull(detailedConsentResources); - for (DetailedConsentResource resource : detailedConsentResources) { - Assert.assertNotNull(resource.getAuthorizationResources()); - Assert.assertNotNull(resource.getConsentMappingResources()); - Assert.assertNotNull(resource.getConsentAttributes()); - - for (AuthorizationResource authResource : resource.getAuthorizationResources()) { - Assert.assertEquals(resource.getConsentID(), authResource.getConsentID()); - } - } - } - - @Test - public void testConsentSearchWithUserIDsList() throws Exception { - - ArrayList detailedConsentResources; - ArrayList consentIDs = new ArrayList<>(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - storeDataForConsentSearchTest(consentIDs, connection); - detailedConsentResources = consentCoreDAO.searchConsents(connection, null, null, - null, null, ConsentMgtDAOTestData.SAMPLE_USER_IDS_LIST, null, - null, 10, 0); - } - - Assert.assertNotNull(detailedConsentResources); - for (DetailedConsentResource resource : detailedConsentResources) { - Assert.assertNotNull(resource.getAuthorizationResources()); - Assert.assertNotNull(resource.getConsentMappingResources()); - Assert.assertNotNull(resource.getConsentAttributes()); - - for (AuthorizationResource authResource : resource.getAuthorizationResources()) { - Assert.assertEquals(resource.getConsentID(), authResource.getConsentID()); - } - } - } - - @Test - public void testConsentSearchWithoutLimitAndOffset() throws Exception { - - ArrayList detailedConsentResources; - ArrayList consentIDs = new ArrayList<>(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - storeDataForConsentSearchTest(consentIDs, connection); - detailedConsentResources = consentCoreDAO.searchConsents(connection, consentIDs, null, - null, null, null, null, null, null, - null); - } - - Assert.assertNotNull(detailedConsentResources); - for (DetailedConsentResource resource : detailedConsentResources) { - Assert.assertNotNull(resource.getAuthorizationResources()); - Assert.assertNotNull(resource.getConsentMappingResources()); - Assert.assertNotNull(resource.getConsentAttributes()); - - for (AuthorizationResource authResource : resource.getAuthorizationResources()) { - Assert.assertEquals(resource.getConsentID(), authResource.getConsentID()); - } - } - } - - @Test - public void testConsentSearchWithoutLimitButOffset() throws Exception { - - ArrayList detailedConsentResources; - ArrayList consentIDs = new ArrayList<>(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - storeDataForConsentSearchTest(consentIDs, connection); - detailedConsentResources = consentCoreDAO.searchConsents(connection, consentIDs, null, - null, null, null, null, null, null, - 1); - } - - Assert.assertNotNull(detailedConsentResources); - for (DetailedConsentResource resource : detailedConsentResources) { - Assert.assertNotNull(resource.getAuthorizationResources()); - Assert.assertNotNull(resource.getConsentMappingResources()); - Assert.assertNotNull(resource.getConsentAttributes()); - - for (AuthorizationResource authResource : resource.getAuthorizationResources()) { - Assert.assertEquals(resource.getConsentID(), authResource.getConsentID()); - } - } - } - - @Test - public void testConsentSearchForNullValues() throws Exception { - - ResultSet mockedResultSetTemp = Mockito.mock(ResultSet.class); - Mockito.doReturn(null).when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.AUTH_ID); - Mockito.doReturn(null).when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.MAPPING_ID); - ConsentCoreDAOImpl dao = new ConsentCoreDAOImpl(new ConsentMgtCommonDBQueries()); - ArrayList authorizationResources = new ArrayList<>(); - ArrayList consentMappingResources = new ArrayList<>(); - dao.setAuthorizationDataInResponseForGroupedQuery(authorizationResources, - mockedResultSetTemp, ""); - dao.setAccountConsentMappingDataInResponse(consentMappingResources, - mockedResultSetTemp); - Assert.assertTrue(authorizationResources.size() == 0); - Assert.assertTrue(consentMappingResources.size() == 0); - } - - @Test - public void testConsentSearchForNoneNullValues() throws Exception { - - ResultSet mockedResultSetTemp = Mockito.mock(ResultSet.class); - Mockito.doReturn("test").when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.AUTH_ID); - Mockito.doReturn("test").when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.MAPPING_ID); - Mockito.doReturn("test").when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.ACCOUNT_ID); - Mockito.doReturn("test").when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.MAPPING_STATUS); - Mockito.doReturn("test").when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.PERMISSION); - Mockito.doReturn("test").when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.AUTH_TYPE); - Mockito.doReturn("test").when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.AUTH_STATUS); - Mockito.doReturn("123456").when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.UPDATED_TIME); - Mockito.doReturn("test").when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.USER_ID); - ConsentCoreDAOImpl dao = new ConsentCoreDAOImpl(new ConsentMgtCommonDBQueries()); - ArrayList authorizationResources = new ArrayList<>(); - ArrayList consentMappingResources = new ArrayList<>(); - dao.setAuthorizationDataInResponseForGroupedQuery(authorizationResources, - mockedResultSetTemp, ""); - dao.setAccountConsentMappingDataInResponse(consentMappingResources, - mockedResultSetTemp); - Assert.assertTrue(authorizationResources.size() != 0); - Assert.assertTrue(consentMappingResources.size() != 0); - } - - @Test - public void testConsentSearchForNoneNullValuesNegativeCase() throws Exception { - - ResultSet mockedResultSetTemp = Mockito.mock(ResultSet.class); - Mockito.doReturn("test").when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.AUTH_ID); - Mockito.doReturn("test").when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.MAPPING_ID); - Mockito.doReturn("1,2").when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.ACCOUNT_ID); - Mockito.doReturn("test").when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.MAPPING_STATUS); - Mockito.doReturn("test").when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.PERMISSION); - Mockito.doReturn("test").when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.AUTH_TYPE); - Mockito.doReturn("test").when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.AUTH_STATUS); - Mockito.doReturn("123456").when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.UPDATED_TIME); - Mockito.doReturn("test,test2").when(mockedResultSetTemp).getString(ConsentMgtDAOConstants.USER_ID); - ConsentCoreDAOImpl dao = new ConsentCoreDAOImpl(new ConsentMgtCommonDBQueries()); - ArrayList authorizationResources = new ArrayList<>(); - ArrayList consentMappingResources = new ArrayList<>(); - dao.setAuthorizationDataInResponseForGroupedQuery(authorizationResources, - mockedResultSetTemp, ""); - dao.setAccountConsentMappingDataInResponse(consentMappingResources, - mockedResultSetTemp); - Assert.assertTrue(authorizationResources.size() != 0); - Assert.assertTrue(consentMappingResources.size() != 0); - } - - @Test - public void testConsentSearchWithoutOffsetButLimit() throws Exception { - - ArrayList detailedConsentResources; - ArrayList consentIDs = new ArrayList<>(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - storeDataForConsentSearchTest(consentIDs, connection); - detailedConsentResources = consentCoreDAO.searchConsents(connection, consentIDs, null, - null, null, null, null, null, 1, - null); - } - - Assert.assertNotNull(detailedConsentResources); - for (DetailedConsentResource resource : detailedConsentResources) { - Assert.assertNotNull(resource.getAuthorizationResources()); - Assert.assertNotNull(resource.getConsentMappingResources()); - Assert.assertNotNull(resource.getConsentAttributes()); - - for (AuthorizationResource authResource : resource.getAuthorizationResources()) { - Assert.assertEquals(resource.getConsentID(), authResource.getConsentID()); - } - } - } - - @Test - public void testConsentSearchWithNoParams() throws Exception { - - ArrayList detailedConsentResources; - ArrayList consentIDs = new ArrayList<>(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - storeDataForConsentSearchTest(consentIDs, connection); - detailedConsentResources = consentCoreDAO.searchConsents(connection, null, null, - null, null, null, null, - null, 10, 0); - } - - Assert.assertNotNull(detailedConsentResources); - for (DetailedConsentResource resource : detailedConsentResources) { - Assert.assertNotNull(resource.getAuthorizationResources()); - Assert.assertNotNull(resource.getConsentMappingResources()); - Assert.assertNotNull(resource.getConsentAttributes()); - - for (AuthorizationResource authResource : resource.getAuthorizationResources()) { - Assert.assertEquals(resource.getConsentID(), authResource.getConsentID()); - } - } - } - - @Test - public void testConsentSearchWithTimePeriod() throws Exception { - - ArrayList detailedConsentResources; - ArrayList consentIDs = new ArrayList<>(); - long currentTime = System.currentTimeMillis() / 1000; - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - storeDataForConsentSearchTest(consentIDs, connection); - detailedConsentResources = consentCoreDAO.searchConsents(connection, null, null, - null, null, null, currentTime, - currentTime + 100, 10, 0); - } - - Assert.assertNotNull(detailedConsentResources); - for (DetailedConsentResource resource : detailedConsentResources) { - Assert.assertNotNull(resource.getAuthorizationResources()); - Assert.assertNotNull(resource.getConsentMappingResources()); - Assert.assertNotNull(resource.getConsentAttributes()); - - for (AuthorizationResource authResource : resource.getAuthorizationResources()) { - Assert.assertEquals(resource.getConsentID(), authResource.getConsentID()); - } - - Assert.assertTrue((currentTime <= resource.getUpdatedTime()) - && (currentTime + 100 >= resource.getUpdatedTime())); - } - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testSearchConsentsSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString(), - Mockito.anyInt(), Mockito.anyInt()); - consentCoreDAO.searchConsents(mockedConnection, null, null, null, - null, null, null, null, null, null); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testSearchConsentsPreparedResultSetError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString(), Mockito.anyInt(), Mockito.anyInt()); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - consentCoreDAO.searchConsents(mockedConnection, null, null, null, - null, null, null, null, null, null); - } - - @Test - public void testSearchConsentAuthorizations() throws Exception { - - ArrayList authorizationResources; - ConsentResource storedConsentResource; - AuthorizationResource storedAuthorizationResource; - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, ConsentMgtDAOTestData - .getSampleTestConsentResource()); - storedAuthorizationResource = consentCoreDAO.storeAuthorizationResource(connection, - ConsentMgtDAOTestData.getSampleTestAuthorizationResource(storedConsentResource.getConsentID())); - - authorizationResources = consentCoreDAO.searchConsentAuthorizations(connection, - storedConsentResource.getConsentID(), storedAuthorizationResource.getUserID()); - } - - Assert.assertNotNull(authorizationResources); - Assert.assertEquals(storedAuthorizationResource.getAuthorizationID(), - authorizationResources.get(0).getAuthorizationID()); - Assert.assertEquals(storedAuthorizationResource.getConsentID(), - authorizationResources.get(0).getConsentID()); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testSearchConsentAuthorizationsSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.searchConsentAuthorizations(mockedConnection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, - ConsentMgtDAOTestData.SAMPLE_USER_ID); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testSearchConsentAuthorizationsResultSetError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - consentCoreDAO.searchConsentAuthorizations(mockedConnection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, - ConsentMgtDAOTestData.SAMPLE_USER_ID); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testSearchConsentAuthorizationsNoRecordsFoundError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(mockedResultSet).when(mockedPreparedStatement).executeQuery(); - Mockito.doReturn(false).when(mockedResultSet).isBeforeFirst(); - consentCoreDAO.searchConsentAuthorizations(mockedConnection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, - ConsentMgtDAOTestData.SAMPLE_USER_ID); - } - - @Test (expectedExceptions = OBConsentDataDeletionException.class) - public void testDeleteConsentAttributeSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.deleteConsentAttributes(mockedConnection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, - ConsentMgtDAOTestData.SAMPLE_CONSENT_ATTRIBUTES_KEYS); - } - - private void storeDataForConsentSearchTest(ArrayList consentIDs, - Connection connection) throws OBConsentDataInsertionException { - - ArrayList authIDs = new ArrayList<>(); - - // Store 3 consent resources - ArrayList consentResources = ConsentMgtDAOTestData.getSampleConsentResourcesList(); - for (ConsentResource resource : consentResources) { - consentIDs.add(consentCoreDAO.storeConsentResource(connection, resource).getConsentID()); - } - - // Store 2 authorization resources for each stored consent - ArrayList authorizationResources = - ConsentMgtDAOTestData.getSampleAuthorizationResourcesList(consentIDs); - for (AuthorizationResource resource : authorizationResources) { - authIDs.add(consentCoreDAO.storeAuthorizationResource(connection, resource).getAuthorizationID()); - } - - // Store 2 consent mapping resources for each authorization resource - ArrayList consentMappingResources = - ConsentMgtDAOTestData.getSampleConsentMappingResourcesList(authIDs); - for (ConsentMappingResource resource : consentMappingResources) { - consentCoreDAO.storeConsentMappingResource(connection, resource); - } - - // Store consent attributes - for (String consentID : consentIDs) { - ConsentAttributes consentAttributesResource = new ConsentAttributes(); - consentAttributesResource.setConsentID(consentID); - consentAttributesResource.setConsentAttributes(ConsentMgtDAOTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP); - consentCoreDAO.storeConsentAttributes(connection, consentAttributesResource); - } - } - - @Test - public void testUpdateConsentReceipt() throws Exception { - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - String newConsentReceipt = "{\"amendedReceipt\":\"amendedData\"}"; - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - ConsentResource storedConsentResource = consentCoreDAO.storeConsentResource(connection, - consentResource); - Assert.assertTrue(consentCoreDAO.updateConsentReceipt(connection, storedConsentResource.getConsentID(), - newConsentReceipt)); - } - } - - @Test (expectedExceptions = OBConsentDataUpdationException.class) - public void testUpdateConsentReceiptSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.updateConsentReceipt(mockedConnection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, - ConsentMgtDAOTestData.SAMPLE_CONSENT_RECEIPT); - } - - @Test (expectedExceptions = OBConsentDataUpdationException.class) - public void testUpdateConsentReceiptUpdateError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection).prepareStatement(Mockito.anyString()); - Mockito.doReturn(0).when(mockedPreparedStatement).executeUpdate(); - consentCoreDAO.updateConsentReceipt(mockedConnection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, - ConsentMgtDAOTestData.SAMPLE_CONSENT_RECEIPT); - } - - @Test - public void testUpdateConsentValidityTime() throws Exception { - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - long newConsentValidityTime = 12345; - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - ConsentResource storedConsentResource = consentCoreDAO.storeConsentResource(connection, - consentResource); - Assert.assertTrue(consentCoreDAO.updateConsentValidityTime(connection, storedConsentResource.getConsentID(), - newConsentValidityTime)); - } - } - - @Test (expectedExceptions = OBConsentDataUpdationException.class) - public void testUpdateConsentValidityTimeSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.updateConsentValidityTime(mockedConnection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, - ConsentMgtDAOTestData.SAMPLE_CONSENT_VALIDITY_PERIOD); - } - - @Test (expectedExceptions = OBConsentDataUpdationException.class) - public void testUpdateConsentValidityTimeUpdateError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection).prepareStatement(Mockito.anyString()); - Mockito.doReturn(0).when(mockedPreparedStatement).executeUpdate(); - consentCoreDAO.updateConsentValidityTime(mockedConnection, ConsentMgtDAOTestData.SAMPLE_CONSENT_ID, - ConsentMgtDAOTestData.SAMPLE_CONSENT_VALIDITY_PERIOD); - } - - @Test - public void testRetrieveConsentIdByConsentAttributeNameAndValue() throws Exception { - - ConsentAttributes consentAttributesResource; - ArrayList consentIdList; - ConsentResource retrievedConsentResource; - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - consentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - retrievedConsentResource = consentCoreDAO.getConsentResource(connection, consentResource.getConsentID()); - - consentAttributesResource = ConsentMgtDAOTestData - .getSampleTestConsentAttributesObject(retrievedConsentResource.getConsentID()); - - consentCoreDAO.storeConsentAttributes(connection, consentAttributesResource); - - consentIdList = consentCoreDAO.getConsentIdByConsentAttributeNameAndValue(connection, - "payment-type", "domestic-payments"); - - } - Assert.assertFalse(consentIdList.isEmpty()); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentIdByConsentAttributeNameAndValueSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.getConsentIdByConsentAttributeNameAndValue(mockedConnection, "payment-type", - "domestic-payments"); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentIdByConsentAttributeNameAndValueResultSetError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - consentCoreDAO.getConsentIdByConsentAttributeNameAndValue(mockedConnection, "payment-type", - "domestic-payments"); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentIdByConsentAttributeNameAndValueNoRecordsFoundError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(mockedResultSet).when(mockedPreparedStatement).executeQuery(); - Mockito.doReturn(false).when(mockedResultSet).isBeforeFirst(); - consentCoreDAO.getConsentIdByConsentAttributeNameAndValue(mockedConnection, "payment-type", - "domestic-payments"); - } - - @Test - public void testRetrieveExpiringConsentsWithNoEligibility() throws Exception { - - ConsentAttributes consentAttributesResource; - ArrayList expirationEligibleConsents; - ConsentResource retrievedConsentResource; - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleStoredTestConsentResource(); - consentResource.setCurrentStatus(ConsentMgtDAOTestData.SAMPLE_EXPIRED_STATUS); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - consentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - retrievedConsentResource = consentCoreDAO.getConsentResource(connection, consentResource.getConsentID()); - - consentAttributesResource = ConsentMgtDAOTestData - .getSampleTestConsentAttributesObject(retrievedConsentResource.getConsentID()); - consentAttributesResource.getConsentAttributes().put( - ConsentMgtDAOConstants.CONSENT_EXPIRY_TIME_ATTRIBUTE, "1632918113"); - - consentCoreDAO.storeConsentAttributes(connection, consentAttributesResource); - - expirationEligibleConsents = consentCoreDAO.getExpiringConsents(connection, - "Authorized,awaitingAuthorisation"); - - } - Assert.assertTrue(expirationEligibleConsents.isEmpty()); - } - - @Test (dependsOnMethods = {"testRetrieveExpiringConsentsWithNoEligibility"}) - public void testRetrieveExpiringConsentsWithEligibility() throws Exception { - - ConsentAttributes consentAttributesResource; - ArrayList expirationEligibleConsents; - ConsentResource retrievedConsentResource; - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - consentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - retrievedConsentResource = consentCoreDAO.getConsentResource(connection, consentResource.getConsentID()); - - consentAttributesResource = ConsentMgtDAOTestData - .getSampleTestConsentAttributesObject(retrievedConsentResource.getConsentID()); - consentAttributesResource.getConsentAttributes().put( - ConsentMgtDAOConstants.CONSENT_EXPIRY_TIME_ATTRIBUTE, "1632918113"); - - consentCoreDAO.storeConsentAttributes(connection, consentAttributesResource); - - expirationEligibleConsents = consentCoreDAO.getExpiringConsents(connection, - "Authorized,awaitingAuthorisation"); - - } - Assert.assertFalse(expirationEligibleConsents.isEmpty()); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveExpiringConsentsDataRetrievalError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection).prepareStatement(Mockito.anyString()); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - consentCoreDAO.getExpiringConsents(mockedConnection, "Authorized,awaitingAuthorisation"); - } - - @DataProvider(name = "storeConsentHistoryDataProvider") - public Object[][] storeConsentHistoryData() { - - /* - * historyID - * consentID - * changedAttributes - * consentType - * amendedTimestamp - * amendmentReason - */ - return ConsentMgtDAOTestData.DataProviders.CONSENT_HISTORY_DATA_HOLDER; - } - - @Test (dataProvider = "storeConsentHistoryDataProvider") - public void testStoreConsentAmendmentHistory(String historyID, String recordID, String changedAttributes, - String consentType, long amendedTimestamp, String amendmentReason) throws Exception { - - boolean result; - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - result = consentCoreDAO.storeConsentAmendmentHistory(connection, historyID, amendedTimestamp, - recordID, consentType, changedAttributes, amendmentReason); - } - Assert.assertTrue(result); - } - - @Test (dataProvider = "storeConsentHistoryDataProvider", expectedExceptions = OBConsentDataInsertionException.class) - public void testStoreConsentAmendmentHistoryWithInvalidConsentType(String historyID, String recordID, - String changedAttributes, String consentType, long amendedTimestamp, String amendmentReason) throws Exception { - - boolean result; - consentType = "sampleConsentType"; - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - result = consentCoreDAO.storeConsentAmendmentHistory(connection, historyID, amendedTimestamp, - recordID, consentType, changedAttributes, amendmentReason); - } - Assert.assertTrue(result); - } - - @Test (dataProvider = "storeConsentHistoryDataProvider", expectedExceptions = OBConsentDataInsertionException.class) - public void testStoreConsentAmendmentHistoryInsertionError(String historyID, String recordID, - String changedAttributes, String consentType, long amendedTimestamp, String amendmentReason) - throws Exception { - - Mockito.doReturn(Mockito.mock(PreparedStatement.class)).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(0).when(Mockito.mock(PreparedStatement.class)).executeUpdate(); - - consentCoreDAO.storeConsentAmendmentHistory(mockedConnection, historyID, amendedTimestamp, - recordID, consentType, changedAttributes, amendmentReason); - } - - @Test (dataProvider = "storeConsentHistoryDataProvider", expectedExceptions = OBConsentDataInsertionException.class) - public void testStoreConsentAmendmentHistorySQLError(String historyID, String recordID, - String changedAttributes, String consentType, long amendedTimestamp, String amendmentReason) - throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.storeConsentAmendmentHistory(mockedConnection, historyID, amendedTimestamp, - recordID, consentType, changedAttributes, amendmentReason); - } - - @Test(dependsOnMethods = {"testStoreConsentAmendmentHistory"}) - public void testRetrieveConsentAmendmentHistory() throws Exception { - - Map consentHistoryResourcesDataMap; - String expectedConsentDataTypes[] = { ConsentMgtDAOConstants.TYPE_CONSENT_BASIC_DATA, - ConsentMgtDAOConstants.TYPE_CONSENT_ATTRIBUTES_DATA, - ConsentMgtDAOConstants.TYPE_CONSENT_MAPPING_DATA, - ConsentMgtDAOConstants.TYPE_CONSENT_AUTH_RESOURCE_DATA, - "AmendedReason"}; - - List expectedConsentDataTypesList = Arrays.asList(expectedConsentDataTypes); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - consentHistoryResourcesDataMap = consentCoreDAO.retrieveConsentAmendmentHistory(connection, - ConsentMgtDAOTestData.getRecordIDListOfSampleConsentHistory()); - Assert.assertNotNull(consentHistoryResourcesDataMap); - for (Map.Entry consentHistoryDataEntry : - consentHistoryResourcesDataMap.entrySet()) { - Assert.assertEquals(ConsentMgtDAOTestData.SAMPLE_HISTORY_ID, consentHistoryDataEntry.getKey()); - Map consentHistoryData = - consentHistoryDataEntry.getValue().getChangedAttributesJsonDataMap(); - for (Map.Entry consentHistoryDataTypeEntry : - consentHistoryData.entrySet()) { - Assert.assertNotNull(consentHistoryDataTypeEntry.getKey()); - Assert.assertTrue(expectedConsentDataTypesList.contains((consentHistoryDataTypeEntry.getKey()))); - Assert.assertNotNull(consentHistoryDataTypeEntry.getValue()); - } - } - } - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentAmendmentHistoryDataRetrievalError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection).prepareStatement(Mockito.anyString()); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - consentCoreDAO.retrieveConsentAmendmentHistory(mockedConnection, - ConsentMgtDAOTestData.getRecordIDListOfSampleConsentHistory()); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testRetrieveConsentAmendmentHistoryPrepStmtSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.retrieveConsentAmendmentHistory(mockedConnection, - ConsentMgtDAOTestData.getRecordIDListOfSampleConsentHistory()); - } - - @Test - public void testRetrieveConsentAmendmentHistoryNoRecordsFound() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(mockedResultSet).when(mockedPreparedStatement).executeQuery(); - Mockito.doReturn(false).when(mockedResultSet).isBeforeFirst(); - - Map result = consentCoreDAO.retrieveConsentAmendmentHistory(mockedConnection, - ConsentMgtDAOTestData.getRecordIDListOfSampleConsentHistory()); - Assert.assertEquals(result.size(), 0); - } - - @Test - public void testDeleteConsentData() throws Exception { - - boolean isDeleted; - boolean isDeletedOnRetentionTable; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(1).when(mockedPreparedStatement).executeUpdate(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - ConsentResource storeConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - isDeleted = consentCoreDAO.deleteConsentData(mockedConnection, storeConsentResource.getConsentID(), - false); - isDeletedOnRetentionTable = consentCoreDAO.deleteConsentData(mockedConnection, - storeConsentResource.getConsentID(), true); - } - Assert.assertTrue(isDeleted); - Assert.assertTrue(isDeletedOnRetentionTable); - } - - @Test - public void testDeleteConsentData2() throws Exception { - - boolean isDeleted; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(0).when(mockedPreparedStatement).executeUpdate(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - ConsentResource storeConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - isDeleted = consentCoreDAO.deleteConsentData(mockedConnection, storeConsentResource.getConsentID(), - false); - } - Assert.assertTrue(!isDeleted); - } - - @Test (expectedExceptions = OBConsentDataDeletionException.class) - public void testDeleteConsentDataSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.deleteConsentData(mockedConnection, "", false); - } - - @Test - public void testGetConsentStatusAuditRecordsByConsentId() throws Exception { - - ConsentResource storedConsentResource; - ConsentStatusAuditRecord consentStatusAuditRecord; - ConsentStatusAuditRecord storedConsentStatusAuditRecord; - ArrayList retrievedConsentStatusAuditRecords; - ArrayList retrievedConsentStatusAuditRecordsInRetentionTable; - ArrayList retrievedConsentStatusAuditRecordsWithLimit; - ArrayList retrievedConsentStatusAuditRecordsWithLimitAndOffset; - ArrayList retrievedConsentStatusAuditRecordsWithLimitOnly; - ArrayList retrievedConsentStatusAuditRecordsWithOffsetOnly; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - - consentStatusAuditRecord = ConsentMgtDAOTestData - .getSampleTestConsentStatusAuditRecord(storedConsentResource.getConsentID(), - storedConsentResource.getCurrentStatus()); - - storedConsentStatusAuditRecord = consentCoreDAO.storeConsentStatusAuditRecord(connection, - consentStatusAuditRecord); - - connection.commit(); - ArrayList consentIds = new ArrayList<>(); - consentIds.add(storedConsentStatusAuditRecord.getConsentID()); - retrievedConsentStatusAuditRecords = consentCoreDAO.getConsentStatusAuditRecordsByConsentId(connection, - consentIds, null, null, false); - retrievedConsentStatusAuditRecordsInRetentionTable = - consentCoreDAO.getConsentStatusAuditRecordsByConsentId(connection, consentIds, null, null, true); - retrievedConsentStatusAuditRecordsWithLimit = - consentCoreDAO.getConsentStatusAuditRecordsByConsentId(connection, consentIds, 10, 0, false); - retrievedConsentStatusAuditRecordsWithLimitAndOffset = - consentCoreDAO.getConsentStatusAuditRecordsByConsentId(connection, consentIds, 10, 1, false); - retrievedConsentStatusAuditRecordsWithLimitOnly = - consentCoreDAO.getConsentStatusAuditRecordsByConsentId(connection, consentIds, 10, null, false); - retrievedConsentStatusAuditRecordsWithOffsetOnly = - consentCoreDAO.getConsentStatusAuditRecordsByConsentId(connection, consentIds, null, 1, false); - } - Assert.assertNotNull(retrievedConsentStatusAuditRecords); - Assert.assertTrue(retrievedConsentStatusAuditRecordsInRetentionTable.isEmpty()); - Assert.assertNotNull(retrievedConsentStatusAuditRecordsWithLimit); - Assert.assertTrue(retrievedConsentStatusAuditRecordsWithLimit.size() > 0); - Assert.assertTrue(retrievedConsentStatusAuditRecordsWithLimitAndOffset.isEmpty()); - Assert.assertTrue(!retrievedConsentStatusAuditRecordsWithLimitOnly.isEmpty()); - Assert.assertTrue(!retrievedConsentStatusAuditRecordsWithOffsetOnly.isEmpty()); - for (ConsentStatusAuditRecord record : - retrievedConsentStatusAuditRecords) { - Assert.assertEquals(storedConsentStatusAuditRecord.getConsentID(), record.getConsentID()); - } - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testGetConsentStatusAuditRecordsByConsentIdSQLError() throws Exception { - ArrayList consentIds = new ArrayList<>(); - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.getConsentStatusAuditRecordsByConsentId(mockedConnection, - consentIds, null, null, false); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testGetConsentStatusAuditRecordsByConsentIdSQLErrorForResults() throws Exception { - ArrayList consentIds = new ArrayList<>(); - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - consentCoreDAO.getConsentStatusAuditRecordsByConsentId(mockedConnection, - consentIds, null, null, false); - } - - @Test - public void testGetListOfConsentIds() throws Exception { - - ConsentResource storedConsentResource; - ArrayList listOfConsentIds; - ArrayList listOfConsentIdsInRetentionTable; - - ConsentResource consentResource = ConsentMgtDAOTestData.getSampleTestConsentResource(); - - try (Connection connection = DAOUtils.getConnection(DB_NAME)) { - - storedConsentResource = consentCoreDAO.storeConsentResource(connection, consentResource); - listOfConsentIds = consentCoreDAO.getListOfConsentIds(connection, false); - listOfConsentIdsInRetentionTable = consentCoreDAO.getListOfConsentIds(connection, true); - } - Assert.assertNotNull(listOfConsentIds); - Assert.assertTrue(listOfConsentIdsInRetentionTable.isEmpty()); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testGetListOfConsentIdsSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - consentCoreDAO.getListOfConsentIds(mockedConnection, false); - } - - @Test (expectedExceptions = OBConsentDataRetrievalException.class) - public void testGetListOfConsentIdsSQLErrorForResults() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - consentCoreDAO.getListOfConsentIds(mockedConnection, false); - - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/dao/util/ConsentMgtDAOTestData.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/dao/util/ConsentMgtDAOTestData.java deleted file mode 100644 index 91e082f7..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/dao/util/ConsentMgtDAOTestData.java +++ /dev/null @@ -1,558 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.util; - -import com.wso2.openbanking.accelerator.consent.mgt.dao.constants.ConsentMgtDAOConstants; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentAttributes; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentFile; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentMappingResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentStatusAuditRecord; -import net.minidev.json.JSONObject; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.UUID; - -/** - * Consent management DAO test data. - */ -public class ConsentMgtDAOTestData { - - public static final String SAMPLE_CONSENT_RECEIPT = "{\"validUntil\": \"2020-10-20\", \"frequencyPerDay\": 1," + - " \"recurringIndicator\": false, \"combinedServiceIndicator\": true}"; - - public static final String SAMPLE_CONSENT_TYPE = "accounts"; - - public static final int SAMPLE_CONSENT_FREQUENCY = 1; - - public static final Long SAMPLE_CONSENT_VALIDITY_PERIOD = 1638337852L; - - public static final String SAMPLE_CONSENT_ID = "2222"; - - public static final String SAMPLE_AUTHORIZATION_ID = "3333"; - - public static final boolean SAMPLE_RECURRING_INDICATOR = true; - - public static final String SAMPLE_CURRENT_STATUS = "Authorized"; - - public static final String SAMPLE_PREVIOUS_STATUS = "Received"; - - public static final String SAMPLE_AUTHORIZATION_TYPE = "authorizationType"; - - public static final String SAMPLE_USER_ID = "admin@wso2.com"; - - public static final String SAMPLE_AUDIT_ID = "4321234"; - - public static final String SAMPLE_NEW_USER_ID = "ann@gold.com"; - - public static final String SAMPLE_AUTHORIZATION_STATUS = "awaitingAuthorization"; - - public static final String SAMPLE_EXPIRED_STATUS = "Expired"; - - public static final String SAMPLE_ACCOUNT_ID = "123456789"; - - public static final String SAMPLE_MAPPING_ID = "12345"; - - public static final String SAMPLE_MAPPING_ID_2 = "67890"; - - public static final String SAMPLE_MAPPING_STATUS = "active"; - - public static final String SAMPLE_NEW_MAPPING_STATUS = "inactive"; - - public static final String SAMPLE_PERMISSION = "samplePermission"; - - public static final String SAMPLE_REASON = "sample reason"; - - public static final String SAMPLE_ACTION_BY = "admin@wso2.com"; - - public static final String SAMPLE_HISTORY_ID = "1234"; - - public static final Long SAMPLE_UPDATED_TIME = 1638337892L; - - public static final String SAMPLE_AMENDMENT_REASON = "sampleReason"; - - public static final Map SAMPLE_CONSENT_ATTRIBUTES_MAP = new HashMap() { - { - put("x-request-id", UUID.randomUUID().toString()); - put("idenpotency-key", UUID.randomUUID().toString()); - put("payment-type", "domestic-payments"); - put("sessionDataKey", "{\"sessionDataKey\":\"a0c8cd6d-eca0-4c4d-9544-2b39e7e1c180\",\"userId\":\"01Z79\"}"); - } - }; - - public static final ArrayList SAMPLE_CONSENT_ATTRIBUTES_KEYS = new ArrayList() { - { - add("x-request-id"); - add("idenpotency-key"); - } - }; - - public static final ArrayList UNMATCHED_MAPPING_IDS = new ArrayList() { - { - add("4444"); - add("5555"); - } - }; - - private static final ArrayList SAMPLE_CONSENT_RECEIPTS_LIST = new ArrayList() { - { - add("{\"element1\": \"value1\"}"); - add("{\"element2\": \"value2\"}"); - add("{\"element3\": \"value3\"}"); - } - }; - - public static final ArrayList SAMPLE_CONSENT_TYPES_LIST = new ArrayList() { - { - add("accounts"); - add("payments"); - add("cof"); - } - }; - - public static final ArrayList SAMPLE_CONSENT_STATUSES_LIST = new ArrayList() { - { - add("created"); - add("authorized"); - add("awaitingAuthorization"); - - } - }; - - public static final ArrayList SAMPLE_CLIENT_IDS_LIST = new ArrayList() { - { - add("clientID1"); - add("clientID2"); - add("clientID3"); - - } - }; - - public static final ArrayList SAMPLE_USER_IDS_LIST = new ArrayList() { - { - add("userID1"); - add("userID2"); - add("userID3"); - } - }; - - private static final ArrayList SAMPLE_VALIDITY_PERIOD_LIST = new ArrayList() { - { - add(1613454661L); - add(1623654661L); - add(1633654671L); - } - }; - - private static final JSONObject SAMPLE_CONSENT_BASIC_DATA_CHANGED_ATTRIBUTES_JSON = new JSONObject() { - { - put("RECEIPT", SAMPLE_CONSENT_RECEIPT); - put("VALIDITY_TIME", SAMPLE_CONSENT_VALIDITY_PERIOD); - put("UPDATED_TIME", SAMPLE_UPDATED_TIME); - } - }; - - public static final JSONObject SAMPLE_CONSENT_ATTRIBUTES_CHANGED_ATTRIBUTES_JSON = new JSONObject() { - { - put("x-request-id", UUID.randomUUID().toString()); - put("idempotency-key", UUID.randomUUID().toString()); - } - }; - - public static final JSONObject SAMPLE_CONSENT_MAPPINGS_CHANGED_ATTRIBUTES_JSON = new JSONObject() { - { - put("MAPPING_STATUS", SAMPLE_MAPPING_STATUS); - } - }; - - public static final String SAMPLE_CONSENT_FILE = "sample file content"; - - /** - * Data Providers class. - */ - public static final class DataProviders { - - /* - * consentID - * clientID - * receipt - * consentType - * consentFrequency - * validityPeriod - * recurringIndicator - * currentStatus - * createdTime - */ - public static final Object[][] CONSENT_RESOURCE_DATA_HOLDER = new Object[][] { - - { - UUID.randomUUID().toString(), - SAMPLE_CONSENT_RECEIPT, - SAMPLE_CONSENT_TYPE, - SAMPLE_CONSENT_FREQUENCY, - SAMPLE_CONSENT_VALIDITY_PERIOD, - SAMPLE_RECURRING_INDICATOR, - SAMPLE_CURRENT_STATUS, - } - }; - - /* - * authorizationType - * userID - * authorizationStatus - */ - public static final Object[][] AUTHORIZATION_RESOURCE_DATA_HOLDER = new Object[][] { - - { - SAMPLE_AUTHORIZATION_TYPE, - SAMPLE_USER_ID, - SAMPLE_AUTHORIZATION_STATUS - } - }; - - /* - * accountID - * permission - * mappingStatus - */ - public static final Object[][] CONSENT_MAPPING_RESOURCE_DATA_HOLDER = new Object[][] { - - { - SAMPLE_ACCOUNT_ID, - SAMPLE_PERMISSION, - SAMPLE_MAPPING_STATUS - } - }; - - /* - * currentStatus - * reason - * actionBy - * currentStatus - */ - public static final Object[][] CONSENT_STATUS_AUDIT_RECORD_DATA_HOLDER = new Object[][] { - - { - SAMPLE_CURRENT_STATUS, - SAMPLE_REASON, - SAMPLE_ACTION_BY, - SAMPLE_CURRENT_STATUS - } - }; - - /* - * consentAttributesMap - */ - public static final Object[][] CONSENT_ATTRIBUTES_DATA_HOLDER = new Object[][] { - - { - SAMPLE_CONSENT_ATTRIBUTES_MAP - } - }; - - /* - * consentFile - */ - public static final Object[][] CONSENT_FILE_DATA_HOLDER = new Object[][] { - - { - SAMPLE_CONSENT_FILE - } - }; - - /* - * newConsentStatus - */ - public static final Object[][] CONSENT_STATUS_UPDATE_DATA_HOLDER = new Object[][] { - - { - SAMPLE_CURRENT_STATUS - } - }; - - /* - * newMappingStatus - */ - public static final Object[][] CONSENT_MAPPING_STATUS_UPDATE_DATA_HOLDER = new Object[][] { - - { - SAMPLE_NEW_MAPPING_STATUS - } - }; - - /* - * newAuthorizationStatus - */ - public static final Object[][] CONSENT_AUTHORIZATION_STATUS_UPDATE_DATA_HOLDER = new Object[][] { - - { - SAMPLE_CURRENT_STATUS - } - }; - - /* - * newAuthorizationUser - */ - public static final Object[][] CONSENT_AUTHORIZATION_USER_UPDATE_DATA_HOLDER = new Object[][] { - - { - SAMPLE_NEW_USER_ID - } - }; - - /* - * consentAttributeKeys - */ - public static final Object[][] CONSENT_ATTRIBUTES_GET_DATA_HOLDER = new Object[][] { - - { - SAMPLE_CONSENT_ATTRIBUTES_KEYS - } - }; - - /* - * historyID - * consentID - * changedAttributes - * consentType - * amendedTimestamp - */ - public static final Object[][] CONSENT_HISTORY_DATA_HOLDER = new Object[][] { - - { - SAMPLE_HISTORY_ID, - SAMPLE_CONSENT_ID, - SAMPLE_CONSENT_BASIC_DATA_CHANGED_ATTRIBUTES_JSON.toString(), - ConsentMgtDAOConstants.TYPE_CONSENT_BASIC_DATA, - SAMPLE_UPDATED_TIME, - SAMPLE_AMENDMENT_REASON - }, - { - SAMPLE_HISTORY_ID, - SAMPLE_CONSENT_ID, - SAMPLE_CONSENT_ATTRIBUTES_CHANGED_ATTRIBUTES_JSON.toString(), - ConsentMgtDAOConstants.TYPE_CONSENT_ATTRIBUTES_DATA, - SAMPLE_UPDATED_TIME, - SAMPLE_AMENDMENT_REASON - }, - { - SAMPLE_HISTORY_ID, - SAMPLE_MAPPING_ID, - SAMPLE_CONSENT_MAPPINGS_CHANGED_ATTRIBUTES_JSON.toString(), - ConsentMgtDAOConstants.TYPE_CONSENT_MAPPING_DATA, - SAMPLE_UPDATED_TIME, - SAMPLE_AMENDMENT_REASON - }, - { - SAMPLE_HISTORY_ID, - SAMPLE_MAPPING_ID_2, - SAMPLE_CONSENT_MAPPINGS_CHANGED_ATTRIBUTES_JSON.toString(), - ConsentMgtDAOConstants.TYPE_CONSENT_MAPPING_DATA, - SAMPLE_UPDATED_TIME, - SAMPLE_AMENDMENT_REASON - }, - { - SAMPLE_HISTORY_ID, - SAMPLE_AUTHORIZATION_ID, - "null", - ConsentMgtDAOConstants.TYPE_CONSENT_AUTH_RESOURCE_DATA, - SAMPLE_UPDATED_TIME, - SAMPLE_AMENDMENT_REASON - } - }; - } - - public static List getRecordIDListOfSampleConsentHistory() { - - List recordIdList = new ArrayList<>(); - recordIdList.add(ConsentMgtDAOTestData.SAMPLE_CONSENT_ID); - recordIdList.add(ConsentMgtDAOTestData.SAMPLE_MAPPING_ID); - recordIdList.add(ConsentMgtDAOTestData.SAMPLE_MAPPING_ID_2); - recordIdList.add(ConsentMgtDAOTestData.SAMPLE_AUTHORIZATION_ID); - return recordIdList; - } - - public static ConsentResource getSampleTestConsentResource() { - - ConsentResource consentResource = new ConsentResource(); - consentResource.setReceipt(ConsentMgtDAOTestData.SAMPLE_CONSENT_RECEIPT); - consentResource.setClientID(UUID.randomUUID().toString()); - consentResource.setConsentType(ConsentMgtDAOTestData.SAMPLE_CONSENT_TYPE); - consentResource.setCurrentStatus(ConsentMgtDAOTestData.SAMPLE_CURRENT_STATUS); - consentResource.setConsentFrequency(ConsentMgtDAOTestData.SAMPLE_CONSENT_FREQUENCY); - consentResource.setValidityPeriod(ConsentMgtDAOTestData.SAMPLE_CONSENT_VALIDITY_PERIOD); - consentResource.setRecurringIndicator(ConsentMgtDAOTestData.SAMPLE_RECURRING_INDICATOR); - - return consentResource; - } - - public static ConsentResource getSampleStoredTestConsentResource() { - - ConsentResource consentResource = new ConsentResource(); - consentResource.setConsentID(UUID.randomUUID().toString()); - consentResource.setReceipt(ConsentMgtDAOTestData.SAMPLE_CONSENT_RECEIPT); - consentResource.setClientID(UUID.randomUUID().toString()); - consentResource.setConsentType(ConsentMgtDAOTestData.SAMPLE_CONSENT_TYPE); - consentResource.setCurrentStatus(ConsentMgtDAOTestData.SAMPLE_CURRENT_STATUS); - consentResource.setConsentFrequency(ConsentMgtDAOTestData.SAMPLE_CONSENT_FREQUENCY); - consentResource.setValidityPeriod(ConsentMgtDAOTestData.SAMPLE_CONSENT_VALIDITY_PERIOD); - consentResource.setRecurringIndicator(ConsentMgtDAOTestData.SAMPLE_RECURRING_INDICATOR); - - return consentResource; - } - - /** - * Generated three sample consent resources for testing purposes. - * - * @return sample consent resources list - */ - public static ArrayList getSampleConsentResourcesList() { - - ArrayList consentResources = new ArrayList<>(); - - for (int i = 0; i < 3; i++) { - ConsentResource consentResource = new ConsentResource(); - consentResource.setReceipt(SAMPLE_CONSENT_RECEIPTS_LIST.get(i)); - consentResource.setClientID(SAMPLE_CLIENT_IDS_LIST.get(i)); - consentResource.setConsentType(SAMPLE_CONSENT_TYPES_LIST.get(i)); - consentResource.setCurrentStatus(SAMPLE_CONSENT_STATUSES_LIST.get(i)); - consentResource.setConsentFrequency(0); - consentResource.setValidityPeriod(SAMPLE_VALIDITY_PERIOD_LIST.get(i)); - consentResource.setRecurringIndicator(false); - consentResources.add(consentResource); - } - return consentResources; - } - - public static ArrayList getSampleAuthorizationResourcesList(ArrayList consentIDs) { - - ArrayList authorizationResources = new ArrayList<>(); - - for (int i = 0; i < consentIDs.size(); i++) { - for (int j = 0; j < 2; j++) { - AuthorizationResource authorizationResource = new AuthorizationResource(); - authorizationResource.setConsentID(consentIDs.get(i)); - authorizationResource.setAuthorizationType(SAMPLE_AUTHORIZATION_TYPE); - authorizationResource.setUserID(SAMPLE_USER_IDS_LIST.get(i)); - authorizationResource.setAuthorizationStatus(SAMPLE_AUTHORIZATION_STATUS); - authorizationResources.add(authorizationResource); - } - } - return authorizationResources; - } - - public static ArrayList getSampleConsentMappingResourcesList(ArrayList authIDs) { - - ArrayList consentMappingResources = new ArrayList<>(); - - for (int i = 0; i < authIDs.size(); i++) { - for (int j = 0; j < 2; j++) { - ConsentMappingResource consentMappingResource = new ConsentMappingResource(); - consentMappingResource.setAuthorizationID(authIDs.get(i)); - consentMappingResource.setAccountID(SAMPLE_ACCOUNT_ID); - consentMappingResource.setPermission(SAMPLE_PERMISSION); - consentMappingResource.setMappingStatus(SAMPLE_MAPPING_STATUS); - consentMappingResources.add(consentMappingResource); - } - } - return consentMappingResources; - } - - public static AuthorizationResource getSampleTestAuthorizationResource(String consentID) { - - AuthorizationResource authorizationResource = new AuthorizationResource(); - authorizationResource.setConsentID(consentID); - authorizationResource.setAuthorizationType(ConsentMgtDAOTestData.SAMPLE_AUTHORIZATION_TYPE); - authorizationResource.setUserID(ConsentMgtDAOTestData.SAMPLE_USER_ID); - authorizationResource.setAuthorizationStatus(ConsentMgtDAOTestData.SAMPLE_AUTHORIZATION_STATUS); - - return authorizationResource; - } - - public static AuthorizationResource getSampleStoredTestAuthorizationResource() { - - AuthorizationResource authorizationResource = new AuthorizationResource(); - authorizationResource.setConsentID(UUID.randomUUID().toString()); - authorizationResource.setAuthorizationID(UUID.randomUUID().toString()); - authorizationResource.setAuthorizationType(ConsentMgtDAOTestData.SAMPLE_AUTHORIZATION_TYPE); - authorizationResource.setUserID(ConsentMgtDAOTestData.SAMPLE_USER_ID); - authorizationResource.setAuthorizationStatus(ConsentMgtDAOTestData.SAMPLE_AUTHORIZATION_STATUS); - authorizationResource.setUpdatedTime(System.currentTimeMillis() / 1000); - - return authorizationResource; - } - - public static ConsentMappingResource getSampleTestConsentMappingResource(String authorizationID) { - - ConsentMappingResource consentMappingResource = new ConsentMappingResource(); - consentMappingResource.setAuthorizationID(authorizationID); - consentMappingResource.setAccountID(ConsentMgtDAOTestData.SAMPLE_ACCOUNT_ID); - consentMappingResource.setPermission(ConsentMgtDAOTestData.SAMPLE_PERMISSION); - consentMappingResource.setMappingStatus(ConsentMgtDAOTestData.SAMPLE_MAPPING_STATUS); - - return consentMappingResource; - } - - public static ConsentMappingResource getSampleTestConsentMappingResourceWithAccountId(String authorizationID, - String accountID) { - - ConsentMappingResource consentMappingResource = new ConsentMappingResource(); - consentMappingResource.setAuthorizationID(authorizationID); - consentMappingResource.setAccountID(accountID); - consentMappingResource.setPermission(ConsentMgtDAOTestData.SAMPLE_PERMISSION); - consentMappingResource.setMappingStatus(ConsentMgtDAOTestData.SAMPLE_MAPPING_STATUS); - - return consentMappingResource; - } - - public static ConsentStatusAuditRecord getSampleTestConsentStatusAuditRecord(String consentID, - String currentStatus) { - - ConsentStatusAuditRecord consentStatusAuditRecord = new ConsentStatusAuditRecord(); - consentStatusAuditRecord.setConsentID(consentID); - consentStatusAuditRecord.setCurrentStatus(currentStatus); - consentStatusAuditRecord.setReason(ConsentMgtDAOTestData.SAMPLE_REASON); - consentStatusAuditRecord.setActionBy(ConsentMgtDAOTestData.SAMPLE_ACTION_BY); - consentStatusAuditRecord.setPreviousStatus(ConsentMgtDAOTestData.SAMPLE_PREVIOUS_STATUS); - - return consentStatusAuditRecord; - } - - public static ConsentAttributes getSampleTestConsentAttributesObject(String consentID) { - - ConsentAttributes consentAttributes = new ConsentAttributes(); - consentAttributes.setConsentID(consentID); - consentAttributes.setConsentAttributes(ConsentMgtDAOTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP); - - return consentAttributes; - } - - public static ConsentFile getSampleConsentFileObject(String fileContent) { - - ConsentFile consentFile = new ConsentFile(); - consentFile.setConsentID(UUID.randomUUID().toString()); - consentFile.setConsentFile(fileContent); - - return consentFile; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/dao/util/DAOUtils.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/dao/util/DAOUtils.java deleted file mode 100644 index dd019282..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/dao/util/DAOUtils.java +++ /dev/null @@ -1,64 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.dao.util; - -import org.apache.commons.dbcp.BasicDataSource; -import org.apache.commons.lang3.StringUtils; - -import java.nio.file.Paths; -import java.sql.Connection; -import java.sql.SQLException; -import java.util.HashMap; -import java.util.Map; - -/** - * DAO utils. - */ -public class DAOUtils { - - private static Map dataSourceMap = new HashMap<>(); - - public static void initializeDataSource(String databaseName, String scriptPath) throws Exception { - BasicDataSource dataSource = new BasicDataSource(); - dataSource.setDriverClassName("org.h2.Driver"); - dataSource.setUsername("username"); - dataSource.setPassword("password"); - dataSource.setUrl("jdbc:h2:mem:" + databaseName); - - try (Connection connection = dataSource.getConnection()) { - connection.createStatement().executeUpdate("RUNSCRIPT FROM '" + scriptPath + "'"); - } - dataSourceMap.put(databaseName, dataSource); - } - - public static Connection getConnection(String database) throws SQLException { - if (dataSourceMap.get(database) != null) { - return dataSourceMap.get(database).getConnection(); - } - throw new RuntimeException("Invalid datasource."); - } - - public static String getFilePath(String fileName) { - if (StringUtils.isNotBlank(fileName)) { - return Paths.get(System.getProperty("user.dir"), "src", "test", "resources", fileName) - .toString(); - } - return null; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/resources/dbScripts/h2.sql b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/resources/dbScripts/h2.sql deleted file mode 100644 index b54f6c72..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/resources/dbScripts/h2.sql +++ /dev/null @@ -1,138 +0,0 @@ --- All the data related to time are stored in unix time stamp and therefore, the data types for the time related data --- are represented in BIGINT. --- Since the database systems does not support adding default unix time to the database columns, the default data --- storing is handled within the database queries. - -CREATE TABLE IF NOT EXISTS OB_CONSENT ( - CONSENT_ID VARCHAR(255) NOT NULL, - RECEIPT CLOB NOT NULL, - CREATED_TIME BIGINT NOT NULL, - UPDATED_TIME BIGINT NOT NULL, - CLIENT_ID VARCHAR(255) NOT NULL, - CONSENT_TYPE VARCHAR(64) NOT NULL, - CURRENT_STATUS VARCHAR(64) NOT NULL, - CONSENT_FREQUENCY INT, - VALIDITY_TIME BIGINT, - RECURRING_INDICATOR BOOLEAN, - PRIMARY KEY (CONSENT_ID) -); - -CREATE TABLE IF NOT EXISTS OB_CONSENT_AUTH_RESOURCE ( - AUTH_ID VARCHAR(255) NOT NULL, - CONSENT_ID VARCHAR(255) NOT NULL, - AUTH_TYPE VARCHAR(255) NOT NULL, - USER_ID VARCHAR(255), - AUTH_STATUS VARCHAR(255) NOT NULL, - UPDATED_TIME BIGINT NOT NULL, - PRIMARY KEY(AUTH_ID), - CONSTRAINT FK_ID_OB_CONSENT_AUTH_RESOURCE FOREIGN KEY (CONSENT_ID) REFERENCES OB_CONSENT (CONSENT_ID) -); - -CREATE TABLE IF NOT EXISTS OB_CONSENT_MAPPING ( - MAPPING_ID VARCHAR(255) NOT NULL, - AUTH_ID VARCHAR(255) NOT NULL, - ACCOUNT_ID VARCHAR(255) NOT NULL, - PERMISSION VARCHAR(255) NOT NULL, - MAPPING_STATUS VARCHAR(255) NOT NULL, - PRIMARY KEY(MAPPING_ID), - CONSTRAINT FK_OB_CONSENT_MAPPING FOREIGN KEY (AUTH_ID) REFERENCES OB_CONSENT_AUTH_RESOURCE (AUTH_ID) -); - -CREATE TABLE IF NOT EXISTS OB_CONSENT_STATUS_AUDIT ( - STATUS_AUDIT_ID VARCHAR(255) NOT NULL, - CONSENT_ID VARCHAR(255) NOT NULL, - CURRENT_STATUS VARCHAR(255) NOT NULL, - ACTION_TIME BIGINT NOT NULL, - REASON VARCHAR(255), - ACTION_BY VARCHAR(255), - PREVIOUS_STATUS VARCHAR(255), - PRIMARY KEY(STATUS_AUDIT_ID), - CONSTRAINT FK_OB_CONSENT_STATUS_AUDIT FOREIGN KEY (CONSENT_ID) REFERENCES OB_CONSENT (CONSENT_ID) -); - -CREATE TABLE IF NOT EXISTS OB_CONSENT_FILE ( - CONSENT_ID VARCHAR(255) NOT NULL, - CONSENT_FILE CLOB, - PRIMARY KEY(CONSENT_ID), - CONSTRAINT FK_OB_CONSENT_FILE FOREIGN KEY (CONSENT_ID) REFERENCES OB_CONSENT (CONSENT_ID) -); - -CREATE TABLE IF NOT EXISTS OB_CONSENT_ATTRIBUTE ( - CONSENT_ID VARCHAR(255) NOT NULL, - ATT_KEY VARCHAR(255) NOT NULL, - ATT_VALUE VARCHAR(1023) NOT NULL, - PRIMARY KEY(CONSENT_ID, ATT_KEY), - CONSTRAINT FK_OB_CONSENT_ATTRIBUTE FOREIGN KEY (CONSENT_ID) REFERENCES OB_CONSENT (CONSENT_ID) -); - -CREATE TABLE IF NOT EXISTS OB_CONSENT_HISTORY ( - TABLE_ID VARCHAR(10) NOT NULL, - RECORD_ID VARCHAR(255) NOT NULL, - HISTORY_ID VARCHAR(255) NOT NULL, - CHANGED_VALUES CLOB NOT NULL, - REASON VARCHAR(255) NOT NULL, - EFFECTIVE_TIMESTAMP BIGINT NOT NULL, - PRIMARY KEY (TABLE_ID,RECORD_ID,HISTORY_ID) -); - -CREATE TABLE IF NOT EXISTS RET_OB_CONSENT ( - CONSENT_ID VARCHAR(255) NOT NULL, - RECEIPT CLOB NOT NULL, - CREATED_TIME BIGINT NOT NULL, - UPDATED_TIME BIGINT NOT NULL, - CLIENT_ID VARCHAR(255) NOT NULL, - CONSENT_TYPE VARCHAR(64) NOT NULL, - CURRENT_STATUS VARCHAR(64) NOT NULL, - CONSENT_FREQUENCY INT, - VALIDITY_TIME BIGINT, - RECURRING_INDICATOR BOOLEAN, - PRIMARY KEY (CONSENT_ID) -); - -CREATE TABLE IF NOT EXISTS RET_OB_CONSENT_AUTH_RESOURCE ( - AUTH_ID VARCHAR(255) NOT NULL, - CONSENT_ID VARCHAR(255) NOT NULL, - AUTH_TYPE VARCHAR(255) NOT NULL, - USER_ID VARCHAR(255), - AUTH_STATUS VARCHAR(255) NOT NULL, - UPDATED_TIME BIGINT NOT NULL, - PRIMARY KEY(AUTH_ID), - CONSTRAINT FK_ID_RET_OB_CONSENT_AUTH_RESOURCE FOREIGN KEY (CONSENT_ID) REFERENCES RET_OB_CONSENT (CONSENT_ID) -); - -CREATE TABLE IF NOT EXISTS RET_OB_CONSENT_MAPPING ( - MAPPING_ID VARCHAR(255) NOT NULL, - AUTH_ID VARCHAR(255) NOT NULL, - ACCOUNT_ID VARCHAR(255) NOT NULL, - PERMISSION VARCHAR(255) NOT NULL, - MAPPING_STATUS VARCHAR(255) NOT NULL, - PRIMARY KEY(MAPPING_ID), - CONSTRAINT FK_RET_OB_CONSENT_MAPPING FOREIGN KEY (AUTH_ID) REFERENCES RET_OB_CONSENT_AUTH_RESOURCE (AUTH_ID) -); - -CREATE TABLE IF NOT EXISTS RET_OB_CONSENT_STATUS_AUDIT ( - STATUS_AUDIT_ID VARCHAR(255) NOT NULL, - CONSENT_ID VARCHAR(255) NOT NULL, - CURRENT_STATUS VARCHAR(255) NOT NULL, - ACTION_TIME BIGINT NOT NULL, - REASON VARCHAR(255), - ACTION_BY VARCHAR(255), - PREVIOUS_STATUS VARCHAR(255), - PRIMARY KEY(STATUS_AUDIT_ID), - CONSTRAINT FK_RET_OB_CONSENT_STATUS_AUDIT FOREIGN KEY (CONSENT_ID) REFERENCES RET_OB_CONSENT (CONSENT_ID) -); - -CREATE TABLE IF NOT EXISTS RET_OB_CONSENT_FILE ( - CONSENT_ID VARCHAR(255) NOT NULL, - CONSENT_FILE CLOB, - PRIMARY KEY(CONSENT_ID), - CONSTRAINT FK_RET_OB_CONSENT_FILE FOREIGN KEY (CONSENT_ID) REFERENCES RET_OB_CONSENT (CONSENT_ID) -); - -CREATE TABLE IF NOT EXISTS OB_CONSENT_ATTRIBUTE ( - CONSENT_ID VARCHAR(255) NOT NULL, - ATT_KEY VARCHAR(255) NOT NULL, - ATT_VALUE VARCHAR(255) NOT NULL, - PRIMARY KEY(CONSENT_ID, ATT_KEY), - CONSTRAINT FK_RET_OB_CONSENT_ATTRIBUTE FOREIGN KEY (CONSENT_ID) REFERENCES RET_OB_CONSENT (CONSENT_ID) -); diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/resources/testng.xml b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/resources/testng.xml deleted file mode 100644 index 7a3f38de..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao/src/test/resources/testng.xml +++ /dev/null @@ -1,27 +0,0 @@ - - - - - - - - - - - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/pom.xml b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/pom.xml deleted file mode 100644 index e6aaeb14..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/pom.xml +++ /dev/null @@ -1,238 +0,0 @@ - - - - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../../pom.xml - - 4.0.0 - - com.wso2.openbanking.accelerator.consent.service - WSO2 Open Banking - Consent Service - WSO2 Open Banking - Consent Service Module - bundle - - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - provided - - - org.wso2.carbon.identity.framework - org.wso2.carbon.identity.application.authentication.framework - provided - - - org.wso2.carbon.identity.inbound.auth.oauth2 - org.wso2.carbon.identity.oauth - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.consent.dao - provided - - - commons-dbcp - commons-dbcp - - - commons-logging - commons-logging - - - org.testng - testng - test - - - com.h2database - h2 - test - - - org.jacoco - org.jacoco.agent - runtime - test - - - org.mockito - mockito-all - test - - - org.powermock - powermock-api-mockito - test - - - org.powermock - powermock-module-testng - test - - - - - - - org.apache.maven.plugins - maven-surefire-plugin - - - src/test/resources/testng.xml - - - target/jacoco.exec - - true - - - - - org.jacoco - jacoco-maven-plugin - ${jacoco.version} - - - - **/*Constants.class - **/*Component.class - **/*DataHolder.class - - - - - default-prepare-agent - - prepare-agent - - - - default-prepare-agent-integration - - prepare-agent-integration - - - - default-report - - report - - - - default-report-integration - - report-integration - - - - default-check - - check - - - - - BUNDLE - - - COMPLEXITY - COVEREDRATIO - 0.76 - - - - - - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - false - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - org.apache.maven.plugins - maven-compiler-plugin - - 1.8 - 1.8 - UTF-8 - - - - org.apache.felix - maven-bundle-plugin - true - - - - ${project.artifactId} - - - com.wso2.openbanking.accelerator.consent.mgt.service.internal - - - org.osgi.framework;version="${osgi.framework.imp.pkg.version.range}", - org.osgi.service.component;version="${osgi.service.component.imp.pkg.version.range}", - org.wso2.carbon.identity.oauth2.*;version="${identity.inbound.auth.oauth.version.range}", - com.wso2.openbanking.accelerator.common.*;version="${project.version}", - org.apache.commons.lang3;version="${commons-lang.version}" - - - !com.wso2.openbanking.accelerator.consent.mgt.service.internal, - com.wso2.openbanking.accelerator.consent.mgt.service.*;version="${project.version}", - - * - <_dsannotations>* - - - - - - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/ConsentCoreService.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/ConsentCoreService.java deleted file mode 100644 index 067c2b70..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/ConsentCoreService.java +++ /dev/null @@ -1,732 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.service; - -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentAttributes; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentFile; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentHistoryResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentMappingResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentStatusAuditRecord; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; - -import java.util.ArrayList; -import java.util.Map; - -/** - * Consent core service interface. - */ -public interface ConsentCoreService { - - /** - * This method is used to create a consent. The following functionality contains in this method. - * - * 1. Creates a consent resource - * 2. If available, stores consent attributes - * 3. Create an audit record for consent creation - * 4. If isImplicitAuth parameter is true, creates an authorization resource - * - * @param consentResource consent resource - * @param userID user ID is optional and used to create the audit record - * @param authStatus authorization status - * @param authType authorization type (eg. authorization, cancellation) - * @param isImplicitAuth flag to determine whether authorization is implicit or not - * @return returns DetailedConsentResource - * @throws ConsentManagementException thrown if any error occur in the process - */ - DetailedConsentResource createAuthorizableConsent(ConsentResource consentResource, String userID, - String authStatus, String authType, - boolean isImplicitAuth) - throws ConsentManagementException; - - /** - * This method is used to create an exclusive consent. The following functionality contains in this method. - * - * 1. Update existing consent statuses as necessary and deactivate their account mappings - * 2. Create audit records for necessary consent updates - * 3. Create a new authorizable consent - * - * @param consentResource consent resource - * @param userID user ID - * @param authStatus authorization status - * @param authType authorization type - * @param applicableExistingConsentsStatus applicable status for existing consents to be updated - * @param newExistingConsentStatus new status that the updated consents should be - * @param isImplicitAuth flag to determine whether authorization is implicit or not - * @return returns DetailedConsentResource - * @throws ConsentManagementException thrown if any error occur in the process - */ - DetailedConsentResource createExclusiveConsent(ConsentResource consentResource, String userID, String authStatus, - String authType, String applicableExistingConsentsStatus, - String newExistingConsentStatus, boolean isImplicitAuth) - throws ConsentManagementException; - - /** - * This method is used to create a consent file. The following functionality contains in this method. - * - * 1. Get the existing consent to validate the status according to the attribute "applicableStatusToFileUpload" - * 2. Create the consent file - * 3. Update the consent status - * 4. Create an audit record for consent update - * - * @param consentFileResource consent file resource - * @param newConsentStatus new consent status - * @param userID user ID (optional) - * @param applicableStatusToFileUpload status that the consent should have to upload a file - * @return true if transaction is a success, throws an exception otherwise - * @throws ConsentManagementException thrown if any error occur in the process - */ - boolean createConsentFile(ConsentFile consentFileResource, String newConsentStatus, String userID, - String applicableStatusToFileUpload) - throws ConsentManagementException; - - /** - * This method is used to revoke a consent. The following functionality contains in this method. - * - * 1. Get existing consent for status validation - * 2. Update existing consent status - * 3. Create an audit record for consent update - * 4. Update account mapping status as inactive - * - * @param consentID ID of the consent - * @param revokedConsentStatus the status of the consent after revoked - * @return true is the transaction is a success, throws an exception otherwise - * @throws ConsentManagementException thrown if any error occur in the process - */ - boolean revokeConsent(String consentID, String revokedConsentStatus) - throws ConsentManagementException; - - /** - * This method is used to revoke a consent by adding a revoke reason. - * The following functionality contains in this method. - * - * 1. Get existing consent for status validation - * 2. Update existing consent status - * 3. Create an audit record for consent update - * 4. Update account mapping status as inactive - * - * @param consentID ID of the consent - * @param revokedConsentStatus the status of the consent after revoked - * @param revokedReason the reason for consent revocation - * @return true is the transaction is a success, throws an exception otherwise - * @throws ConsentManagementException thrown if any error occur in the process - */ - boolean revokeConsentWithReason(String consentID, String revokedConsentStatus, String revokedReason) - throws ConsentManagementException; - - /** - * This method is used to revoke a consent. The following functionality contains in this method. - * - * 1. Get existing consent for status validation - * 2. Update existing consent status - * 3. Create an audit record for consent update - * 4. Update account mapping status as inactive - * - * @param consentID ID of the consent - * @param revokedConsentStatus the status of the consent after revoked - * @param userID user ID - * @return true is the transaction is a success, throws an exception otherwise - * @throws ConsentManagementException thrown if any error occur in the process - */ - boolean revokeConsent(String consentID, String revokedConsentStatus, String userID) - throws ConsentManagementException; - - /** - * This method is used to revoke a consent by adding a revoke reason. - * The following functionality contains in this method. - * - * 1. Get existing consent for status validation - * 2. Update existing consent status - * 3. Create an audit record for consent update - * 4. Update account mapping status as inactive - * - * @param consentID ID of the consent - * @param revokedConsentStatus the status of the consent after revoked - * @param userID user ID - * @param revokedReason the reason for consent revocation - * @return true is the transaction is a success, throws an exception otherwise - * @throws ConsentManagementException thrown if any error occur in the process - */ - boolean revokeConsentWithReason(String consentID, String revokedConsentStatus, String userID, String revokedReason) - throws ConsentManagementException; - - /** - * This method is used to revoke a consent. The following functionality contains in this method. - * - * 1. Get existing consent for status validation - * 2. Update existing consent status - * 3. Create an audit record for consent update - * 4. Update account mapping status as inactive - * 5. Revoke tokens related to the consent if the flag 'shouldRevokeTokens' is true - * - * @param consentID ID of the consent - * @param revokedConsentStatus the status of the consent after revoked - * @param userID user ID - * @param shouldRevokeTokens the check to revoke tokens or not when revoking consent - * @return true is the transaction is a success, throws an exception otherwise - * @throws ConsentManagementException thrown if any error occur in the process - */ - boolean revokeConsent(String consentID, String revokedConsentStatus, String userID, boolean shouldRevokeTokens) - throws ConsentManagementException; - - /** - * This method is used to revoke a consent. The following functionality contains in this method. - * - * 1. Get existing consent for status validation - * 2. Update existing consent status - * 3. Create an audit record for consent update - * 4. Update account mapping status as inactive - * 5. Revoke tokens related to the consent if the flag 'shouldRevokeTokens' is true - * - * @param consentID ID of the consent - * @param revokedConsentStatus the status of the consent after revoked - * @param userID user ID - * @param shouldRevokeTokens the check to revoke tokens or not when revoking consent - * @param revokedReason the reason for consent revocation - * @return true is the transaction is a success, throws an exception otherwise - * @throws ConsentManagementException thrown if any error occur in the process - */ - boolean revokeConsentWithReason(String consentID, String revokedConsentStatus, String userID, - boolean shouldRevokeTokens, String revokedReason) - throws ConsentManagementException; - - /** - * This method is used to revoke existing consents for the given clientID, userID, consent type and status - * combination. Also revokes the tokens related to the consents which are revoked if the flag - * 'shouldRevokeTokens' is true. - * - * @param clientID ID of the client - * @param userID ID of the user - * @param consentType consent type - * @param applicableStatusToRevoke the status that a consent should have for revoking - * @param revokedConsentStatus the status should be updated the consent with after revoking - * @param shouldRevokeTokens the check to revoke tokens or not when revoking consent - * @return returns true if successful - * @throws ConsentManagementException thrown if an error occurs in the process - */ - boolean revokeExistingApplicableConsents(String clientID, String userID, String consentType, - String applicableStatusToRevoke, String revokedConsentStatus, - boolean shouldRevokeTokens) - throws ConsentManagementException; - - /** - * This method is used to get a consent with or without consent attributes. The following functionality contains in - * this method. - * - * 1. Get existing consent for status validation - * 2. Optionally gets consent attributes according to the value of withConsentAttributes flag - * 3. Check whether the retrieved consent involves a file - * - * @param consentID ID of the consent - * @param withConsentAttributes flag to determine the consent should be retrieved with attributes or not - * @return returns ConsentResource - * @throws ConsentManagementException thrown if any error occur in the process - */ - ConsentResource getConsent(String consentID, boolean withConsentAttributes) - throws ConsentManagementException; - - /** - * This method is used to store consent attributes related to a particular consent. - * - * @param consentID consent ID - * @param consentAttributes consent attribute key and values map - * @return a consent attributes resource - * @throws ConsentManagementException thrown if an error occurs in the process - */ - boolean storeConsentAttributes(String consentID, Map consentAttributes) - throws ConsentManagementException; - - /** - * This method is used to get consent attributes for a provided attribute keys list related to a particular consent. - * - * @param consentID consent ID - * @param consentAttributeKeys consent attribute keys list - * @return a consent attributes resource - * @throws ConsentManagementException thrown if an error occurs in the process - */ - ConsentAttributes getConsentAttributes(String consentID, ArrayList consentAttributeKeys) - throws ConsentManagementException; - - /** - * This method is used to get consent attributes related to a particular consent. - * - * @param consentID consent ID - * @return a consent attributes resource - * @throws ConsentManagementException thrown if an error occurs in the process - */ - ConsentAttributes getConsentAttributes(String consentID) throws ConsentManagementException; - - /** - * This method is used to get consent attributes for a provided attribute name. - * - * @param attributeName attribute name - * @return a map with related consent ID and the attribute values - * @throws ConsentManagementException thrown if an error occurs in the process - */ - Map getConsentAttributesByName(String attributeName) throws ConsentManagementException; - - - /** - * This method is used to get consent attributes for a provided attribute name and attribute value. - * - * @param attributeName attribute name - * @param attributeValue attribute value - * @return Consent ID related to the given attribute key and value - * @throws ConsentManagementException thrown if an error occurs in the process - */ - ArrayList getConsentIdByConsentAttributeNameAndValue(String attributeName, String attributeValue) - throws ConsentManagementException; - - /** - * This method is used to delete the provided consent attributes for a particular consent. - * - * @param consentID consen ID - * @param attributeKeysList attributes to delete - * @return true if deletion is successfull - * @throws ConsentManagementException thrown if an error occurs in the process - */ - boolean deleteConsentAttributes(String consentID, ArrayList attributeKeysList) - throws ConsentManagementException; - - /** - * This method is used to retrieve the consent file using the related consent ID. - * - * @param consentID consent ID - * @return the consent file resource - * @throws ConsentManagementException thrown if an error occurs - */ - ConsentFile getConsentFile(String consentID) throws ConsentManagementException; - - /** - * This method is to retrieve an authorization resource using a given authorization ID. - * - * @param authorizationID authorization ID - * @return an authorization resource - * @throws ConsentManagementException thrown if an error occurs in the process - */ - AuthorizationResource getAuthorizationResource(String authorizationID) throws ConsentManagementException; - - /** - * This method is used to search audit records. Useful for auditing purposes. All the input parameters are - * optional. If all parameters are null, all the audit records will be returned. - * - * @param consentID consent ID - * @param status status of the audit records needed - * @param actionBy user who performed the status change - * @param fromTime from time - * @param toTime to time - * @param statusAuditID ID of a specific audit record that need to be searched - * @return a list of consent status audit records - * @throws ConsentManagementException thrown if an error occurs - */ - ArrayList searchConsentStatusAuditRecords(String consentID, String status, - String actionBy, Long fromTime, Long toTime, - String statusAuditID) - throws ConsentManagementException; - - /** - * This method is used in consent re authorization scenarios to update the account mappings according to the - * additional/removed accounts from the new authorization. Also, the consent status is updated with a provided - * status. Also can be used to amend accounts. - * - * @param consentID consent ID - * @param authID authorization ID - * @param userID user ID for creating the audit record - * @param accountIDsMapWithPermissions accounts IDs with relative permissions - * @param currentConsentStatus current status of the consent for creating audit record - * @param newConsentStatus new consent status after re authorization - * @return true if all operations are successful - * @throws ConsentManagementException thrown if any error occurs in the process - */ - boolean reAuthorizeExistingAuthResource(String consentID, String authID, String userID, Map> accountIDsMapWithPermissions, String currentConsentStatus, String newConsentStatus) - throws ConsentManagementException; - - /** - * This method is used in consent re authorization scenarios to update the account mappings according to the - * additional/removed accounts from the new authorization. A new authorization resource will be created when - * re authorizing using this method. Existing authorizations will be updated with a provided status. Also, the - * consent status is updated with a provided status. Also can be used to amend accounts. - * - * @param consentID consent ID - * @param userID user ID - * @param accountIDsMapWithPermissions account IDs with relative permissions - * @param currentConsentStatus current status of the consent for creating audit record - * @param newConsentStatus new consent status after re authorization - * @param newExistingAuthStatus new status of the existing authorizations - * @param newAuthStatus new status of the new authorization - * @param newAuthType new authorization type - * @return true if all operations are successful - * @throws ConsentManagementException thrown if any error occurs in the process - */ - boolean reAuthorizeConsentWithNewAuthResource(String consentID, String userID, Map> accountIDsMapWithPermissions, String currentConsentStatus, String newConsentStatus, - String newExistingAuthStatus, String newAuthStatus, - String newAuthType) - throws ConsentManagementException; - - /** - * This method is used to get a detailed consent for the provided consent ID. The detailed consent includes - * following data if exist in addition to consent resource specific data. - * - * 1. Relative consent authorization data - * 2. Relative consent account mapping data - * 3. Relative consent attributes - * - * @param consentID ID of the consent - * @return a detailed consent resource - * @throws ConsentManagementException thrown if any error occur in the process - */ - DetailedConsentResource getDetailedConsent(String consentID) throws ConsentManagementException; - - /** - * This method is used to create an authorization for a consent. - * - * @param authorizationResource authorization resource - * @return returns AuthorizationResource - * @throws ConsentManagementException thrown if any error occurs in the process - */ - AuthorizationResource createConsentAuthorization(AuthorizationResource authorizationResource) - throws ConsentManagementException; - - /** - * This method is used to create account ID and permission mappings for the relevant authorized user. A map is - * used to represent permissions related to each accountID. - * - * @param authID authorization ID - * @param accountIDsMapWithPermissions account IDs with relative permissions - * @return returns the list of created consent mapping resources - * @throws ConsentManagementException thrown if any error occurs - */ - ArrayList createConsentAccountMappings(String authID, - Map> - accountIDsMapWithPermissions) - throws ConsentManagementException; - - /** - * This method is used to deactivate account bindings of provided account mapping IDs. - * - * @param accountMappingIDs list of account mapping IDs to be deactivated - * @return true is deactivation is a success, false otherwise - * @throws ConsentManagementException thrown if any error occurs - */ - boolean deactivateAccountMappings(ArrayList accountMappingIDs) throws ConsentManagementException; - - /** - * This method is used to update the status of account bindings of provided account mapping IDs. - * - * @param accountMappingIDs list of account mapping IDs to be updated - * @param newMappingStatus new mapping status - * @return true is update is successful, false otherwise - * @throws ConsentManagementException thrown if any error occurs - */ - boolean updateAccountMappingStatus(ArrayList accountMappingIDs, String newMappingStatus) throws - ConsentManagementException; - - /** - * This method is used to update the permissions of the provided account mappings. - * - * @param mappingIDPermissionMap - A map of mapping IDs against new permissions - * @return - true if update is successful - * @throws ConsentManagementException - Thrown if a DAO level error occurs - */ - boolean updateAccountMappingPermission(Map mappingIDPermissionMap) throws - ConsentManagementException; - - /** - * This method is used to search detailed consents for the given lists of parameters. Following optional lists - * can be passed to retrieve detailed consents. The search will be performed according to the provided input. Any - * list can contain any number of elements. The conjunctive result will be returned. If all lists are passed as - * null, all the consents related to other search parameters will be returned. - * - * 1. A list of consent IDs - * 2. A list of client IDs - * 3. A list of consent types - * 4. A list of consent statuses - * 5. A list of user IDs - * - * (All above lists are optional) - * - * @param consentIDs consent IDs optional list - * @param clientIDs client IDs optional list - * @param consentTypes consent types optional list - * @param consentStatuses consent statuses optional list - * @param userIDs user IDs optional list - * @param fromTime from time - * @param toTime to time - * @param limit limit - * @param offset offset - * @return a list of detailed consent resources according to the provided parameters - * @throws ConsentManagementException thrown if any error occur - * @deprecated use {@link #searchDetailedConsents(ArrayList, ArrayList, ArrayList, ArrayList, ArrayList, Long, - * Long, Integer, Integer, boolean)} instead. - */ - @Deprecated - ArrayList searchDetailedConsents(ArrayList consentIDs, ArrayList clientIDs, - ArrayList consentTypes, - ArrayList consentStatuses, - ArrayList userIDs, Long fromTime, Long toTime, - Integer limit, Integer offset) - throws ConsentManagementException; - - - /** - * This method is used to search detailed consents for the given lists of parameters. Following optional lists - * can be passed to retrieve detailed consents. The search will be performed according to the provided input. Any - * list can contain any number of elements. The conjunctive result will be returned. If all lists are passed as - * null, all the consents related to other search parameters will be returned. - * - * 1. A list of consent IDs - * 2. A list of client IDs - * 3. A list of consent types - * 4. A list of consent statuses - * 5. A list of user IDs - * - * (All above lists are optional) - * - * @param consentIDs consent IDs optional list - * @param clientIDs client IDs optional list - * @param consentTypes consent types optional list - * @param consentStatuses consent statuses optional list - * @param userIDs user IDs optional list - * @param fromTime from time - * @param toTime to time - * @param limit limit - * @param offset offset - * @param fetchFromRetentionDatabase flag to enable fetch data from retention database. - * @return a list of detailed consent resources according to the provided parameters - * @throws ConsentManagementException thrown if any error occur - */ - ArrayList searchDetailedConsents(ArrayList consentIDs, ArrayList clientIDs, - ArrayList consentTypes, - ArrayList consentStatuses, - ArrayList userIDs, Long fromTime, Long toTime, - Integer limit, Integer offset, - boolean fetchFromRetentionDatabase) - throws ConsentManagementException; - - /** - * This method is used to bind user and accounts to the consent. - * - * @param consentResource consent resource - * @param userID user ID - * @param authID ID of the authorization resource - * @param accountIDsMapWithPermissions account IDs list with relevant permissions - * @param newAuthStatus new authorization status - * @param newCurrentConsentStatus the new status of the current consent - * @return true if all operations are successful - * @throws ConsentManagementException thrown if an error occurs in the process - */ - boolean bindUserAccountsToConsent(ConsentResource consentResource, String userID, String authID, - Map> accountIDsMapWithPermissions, String newAuthStatus, - String newCurrentConsentStatus) throws ConsentManagementException; - - /** - * This method is used to bind user and accounts to the consent where permissions for each account is not relevant. - * - * @param consentResource consent resource - * @param userID user ID - * @param authID ID of the authorization resource - * @param accountIDs account IDs list - * @param newAuthStatus new authorization status - * @param newCurrentConsentStatus the new status of the current consent - * @return true if all operations are successful - * @throws ConsentManagementException thrown if an error occurs in the process - */ - public boolean bindUserAccountsToConsent(ConsentResource consentResource, String userID, - String authID, ArrayList accountIDs, - String newAuthStatus, - String newCurrentConsentStatus) - throws ConsentManagementException; - - /** - * This method is used to search authorization resources for a given input parameter. Both consent ID and - * user ID are optional. If both are null, all authorization resources will be returned. - * - * @param consentID consent ID - * @param userID user ID - * @return a list of authorization resources - * @throws ConsentManagementException thrown if any error occurs in the process - */ - ArrayList searchAuthorizations(String consentID, String userID) - throws ConsentManagementException; - - /** - * This method is used to search authorization resources for a given consent ID. - * - * @param consentID consent ID - * @return a list of authorization resources - * @throws ConsentManagementException thrown if any error occurs in the process - */ - ArrayList searchAuthorizations(String consentID) - throws ConsentManagementException; - - /** - * This method is used to search authorization resources for a userId. - * - * @param userID user ID - * @return a list of authorization resources - * @throws ConsentManagementException thrown if any error occurs in the process - */ - ArrayList searchAuthorizationsForUser(String userID) - throws ConsentManagementException; - - /** - * This method is used to amend consent receipt or validity period. The consent ID is mandatory. One of consent - * receipt of validity period must be provided. An audit record is created to indicate that the consent is - * amended. But the consent status won't be changed (since when an authorized consent is amended, the status - * remains the same) - * - * @param consentID consent ID - * @param consentReceipt new consent receipt - * @param consentValidityTime new consent validity time - * @param userID user ID to create audit record - * @return the updated consent resource - * @throws ConsentManagementException thrown if any error occurs in the process - */ - ConsentResource amendConsentData(String consentID, String consentReceipt, Long consentValidityTime, String userID) - throws ConsentManagementException; - - /** - * This method is used to update status of the consent for a given consentId and userId. - * @param consentId consent ID - * @param newConsentStatus new consent status - * @return updated consent resource - * @throws ConsentManagementException thrown if any error occurs in the process - */ - ConsentResource updateConsentStatus(String consentId, String newConsentStatus) - throws ConsentManagementException; - - /** - * This method is used to fetch consents which has a expiring time as a consent attribute - * (eligible for expiration). - * @param statusesEligibleForExpiration statuses eligible for expiration - * @return list of consents eligible for expiration - * @throws ConsentManagementException thrown if any error occurs in the process - */ - ArrayList getConsentsEligibleForExpiration(String statusesEligibleForExpiration) - throws ConsentManagementException; - - /** - * This method is used to update the status of an authorization resource by providing the authorization Id and - * the new authorization status. - * - * @param authorizationId the authorization Id of the authorization resource need to be updated - * @param newAuthorizationStatus the new authorization resource - * @return the updated authorization resource - * @throws ConsentManagementException thrown if any error occur while updating - */ - AuthorizationResource updateAuthorizationStatus(String authorizationId, String newAuthorizationStatus) - throws ConsentManagementException; - - /** - * This method is used to update the user of an authorization resource by providing the authorization ID and - * the user ID. - * - * @param authorizationID the authorization ID of the authorization resource that needs to be updated - * @param userID the user of the authorization resource - * @throws ConsentManagementException thrown if any error occurs while updating - */ - void updateAuthorizationUser(String authorizationID, String userID) - throws ConsentManagementException; - - /** - * This method is used to amend the selected properties of the entire detailed consent. The consent ID is mandatory. - * One of consent receipt or validity period must be provided. - * An audit record is created to indicate that the consent is - * amended. But the consent status won't be changed (since when an authorized consent is amended, the status - * remains the same) - * - * @param consentID consent ID - * @param consentReceipt new consent receipt - * @param consentValidityTime new consent validity time - * @param authID authorization ID - * @param accountIDsMapWithPermissions accounts IDs with relative permissions - * @param newConsentStatus new consent status - * @param consentAttributes new consent attributes key and values map - * @param userID user ID to create audit record - * @param additionalAmendmentData A Data Map to pass any additional data that needs to be amended in the consent - * @return the updated detailed consent resource - * @throws ConsentManagementException thrown if any error occurs in the process - */ - DetailedConsentResource amendDetailedConsent(String consentID, String consentReceipt, Long consentValidityTime, - String authID, Map> accountIDsMapWithPermissions, - String newConsentStatus, Map consentAttributes, String userID, - Map additionalAmendmentData) - throws ConsentManagementException; - - /** - * This method is used to store the details of the previous consent when an consent amendment happens. - * The consent ID is mandatory. The detailed consent resource for the previous consent and the amendedTimestamp - * is mandatory to be set in the ConsentHistoryResource. - * - * @param consentID consent ID - * @param consentHistoryResource detailed consent resource and other history parameters of the previous consent - * @param currentConsentResource detailed consent resource of the current (new) consent - * @return true if all operations are successful - * @throws ConsentManagementException thrown if any error occurs in the process - */ - boolean storeConsentAmendmentHistory(String consentID, ConsentHistoryResource consentHistoryResource, - DetailedConsentResource currentConsentResource) throws ConsentManagementException; - - /** - * This method is used to retrieve consent amendment history for a given consentId. Consent ID is mandatory. - * - * @param consentID consent ID - * @return a map of consent history resources - * @throws ConsentManagementException thrown if any error occurs in the process - */ - Map getConsentAmendmentHistoryData(String consentID) - throws ConsentManagementException; - - /** - * This method is used to sync the retention database with purged consents from consent database. - * @return true if the sync is successful - * @throws ConsentManagementException thrown if any error occurs in the process - */ - boolean syncRetentionDatabaseWithPurgedConsent() throws ConsentManagementException; - - /** - * This method is used to retrieve a list of consent status audit records by consent_id. - * - * @param consentIDs list of consentIDs (optional) - * @param limit limit - * @param offset offset - * @param fetchFromRetentionDatabase boolean value to fetch from retention tables (temporary purged data) - * @return returns a list of consent status audit records. - * @throws ConsentManagementException thrown if a database error occurs - */ - ArrayList getConsentStatusAuditRecords(ArrayList consentIDs, - Integer limit, Integer offset, - boolean fetchFromRetentionDatabase) - throws ConsentManagementException; - - /** - * This method is used to retrieve consent file data by consent_id. - * - * @param consentId consentID - * @param fetchFromRetentionDatabase boolean value to fetch from retention tables (temporary purged data) - * @return returns consent file data by consent_id. - * @throws ConsentManagementException thrown if a database error occurs - */ - ConsentFile getConsentFile(String consentId, boolean fetchFromRetentionDatabase) throws ConsentManagementException; - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/constants/ConsentCoreServiceConstants.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/constants/ConsentCoreServiceConstants.java deleted file mode 100644 index 2ab2b14a..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/constants/ConsentCoreServiceConstants.java +++ /dev/null @@ -1,71 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.service.constants; - -/** - * Consent Core Service Constants. - */ -public class ConsentCoreServiceConstants { - - public static final String CONSENT_ATTRIBUTES_DELETE_ERROR_MSG = "Error occurred while deleting consent " + - "attributes in the database"; - public static final String DATA_INSERTION_ROLLBACK_ERROR_MSG = "Error occurred while inserting data. Rolling back" + - " the transaction"; - public static final String DATA_RETRIEVE_ERROR_MSG = "Error occurred while retrieving data"; - public static final String DATA_UPDATE_ROLLBACK_ERROR_MSG = "Error occurred while updating consent data. Rolling " + - "back the transaction"; - public static final String DATA_DELETE_ERROR_MSG = "Error occurred while deleting data"; - public static final String DATA_DELETE_ROLLBACK_ERROR_MSG = "Error occurred while deleting consent data. Rolling " + - "back the transaction"; - public static final String NEW_CONSENT_STATUS_OR_APPLICABLE_STATUS_MISSING_ERROR = "New consent status or " + - "applicable status for file upload is missing. Cannot proceed"; - public static final String CREATE_EXCLUSIVE_CONSENT_MANDATORY_PARAMETER_MISSING_ERROR = "One or more of following" + - " data are missing (Client ID, receipt, consent type, consent status, auth status, auth type, applicable " + - "existing consent status, new existing consent status, new current consent status), cannot proceed"; - - public static final String TRANSACTION_COMMITTED_LOG_MSG = "Transaction committed"; - public static final String DATABASE_CONNECTION_CLOSE_LOG_MSG = "Closing database connection"; - - public static final String CONSENT_REVOKE_FROM_DASHBOARD_REASON = "Revoke the consent from dashboard"; - public static final String CONSENT_REVOKE_REASON = "Revoke the consent"; - public static final String CONSENT_FILE_UPLOAD_REASON = "Upload consent file"; - public static final String CREATE_CONSENT_REASON = "Create consent"; - public static final String CREATE_EXCLUSIVE_AUTHORIZATION_CONSENT_REASON = "Create exclusive authorization consent"; - public static final String USER_ACCOUNTS_BINDING_REASON = "Bind user accounts to consent"; - public static final String CONSENT_REAUTHORIZE_REASON = "Reauthorize consent"; - public static final String CONSENT_AMEND_REASON = "Amend consent"; - public static final String SUBMISSION_RECEIVED_REASON = "Receive submission request for the consent"; - - public static final String ACTIVE_MAPPING_STATUS = "active"; - public static final String INACTIVE_MAPPING_STATUS = "inactive"; - public static final String CONSENT_AMENDED_STATUS = "amended"; - - public static final String CONSENT_RESOURCE = "ConsentResource"; - public static final String DETAILED_CONSENT_RESOURCE = "DetailedConsentResource"; - public static final String CONSENT_AMENDMENT_HISTORY_RESOURCE = "ConsentAmendmentHistory"; - - public static final String ADDITIONAL_AUTHORIZATION_RESOURCES = "AdditionalAuthorizationResources"; - public static final String ADDITIONAL_MAPPING_RESOURCES = "AdditionalMappingResources"; - - public static final String CONSENT_AMENDMENT_TIME = "ConsentAmendmentTime"; - public static final String AMENDMENT_REASON_CONSENT_AMENDMENT_FLOW = "ConsentAmendmentFlow"; - public static final String AMENDMENT_REASON_CONSENT_REVOCATION = "ConsentRevocation"; - public static final String AMENDMENT_REASON_CONSENT_EXPIRATION = "ConsentExpiration"; - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/impl/ConsentCoreServiceImpl.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/impl/ConsentCoreServiceImpl.java deleted file mode 100644 index 0a8f0504..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/impl/ConsentCoreServiceImpl.java +++ /dev/null @@ -1,2848 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.service.impl; - -import com.google.gson.Gson; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.util.DatabaseUtil; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.consent.mgt.dao.ConsentCoreDAO; -import com.wso2.openbanking.accelerator.consent.mgt.dao.constants.ConsentMgtDAOConstants; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataDeletionException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataInsertionException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataRetrievalException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataUpdationException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentAttributes; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentFile; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentHistoryResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentMappingResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentStatusAuditRecord; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.persistence.ConsentStoreInitializer; -import com.wso2.openbanking.accelerator.consent.mgt.service.ConsentCoreService; -import com.wso2.openbanking.accelerator.consent.mgt.service.constants.ConsentCoreServiceConstants; -import com.wso2.openbanking.accelerator.consent.mgt.service.internal.ConsentManagementDataHolder; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.commons.collections.CollectionUtils; -import org.apache.commons.collections.MapUtils; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.oltu.oauth2.common.message.types.GrantType; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedUser; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.OAuth2Service; -import org.wso2.carbon.identity.oauth2.bean.OAuthClientAuthnContext; -import org.wso2.carbon.identity.oauth2.dao.OAuthTokenPersistenceFactory; -import org.wso2.carbon.identity.oauth2.dto.OAuthRevocationRequestDTO; -import org.wso2.carbon.identity.oauth2.dto.OAuthRevocationResponseDTO; -import org.wso2.carbon.identity.oauth2.model.AccessTokenDO; -import org.wso2.carbon.identity.oauth2.util.OAuth2Util; -import org.wso2.carbon.user.core.util.UserCoreUtil; -import org.wso2.carbon.utils.multitenancy.MultitenantUtils; - -import java.sql.Connection; -import java.sql.SQLException; -import java.sql.Savepoint; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.HashSet; -import java.util.LinkedHashMap; -import java.util.List; -import java.util.Map; -import java.util.Set; -import java.util.UUID; - -/** - * Consent core service implementation. - */ -public class ConsentCoreServiceImpl implements ConsentCoreService { - - private static final Log log = LogFactory.getLog(ConsentCoreServiceImpl.class); - - @Override - public DetailedConsentResource createAuthorizableConsent(ConsentResource consentResource, String userID, - String authStatus, String authType, - boolean isImplicitAuth) - throws ConsentManagementException { - - if (StringUtils.isBlank(consentResource.getClientID()) || StringUtils.isBlank(consentResource.getReceipt()) || - StringUtils.isBlank(consentResource.getConsentType()) || - StringUtils.isBlank(consentResource.getCurrentStatus())) { - - log.error("Client ID, receipt, consent type or consent status is missing, cannot proceed"); - throw new ConsentManagementException("Cannot proceed since client ID, receipt, consent type or consent " + - "status is missing."); - } - - if (isImplicitAuth) { - if (StringUtils.isBlank(authStatus) || StringUtils.isBlank(authType)) { - log.error("Authorization status and authorization type is not found for implicit " + - "authorization creation"); - throw new ConsentManagementException("Cannot proceed with implicit authorization creation " + - "without Authorization Status and Authorization Type provided"); - } - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - DetailedConsentResource detailedConsentResource = createAuthorizableConesntWithAuditRecord(connection, - consentCoreDAO, consentResource, userID, authStatus, authType, isImplicitAuth); - DatabaseUtil.commitTransaction(connection); - return detailedConsentResource; - } catch (OBConsentDataInsertionException e) { - log.error(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public DetailedConsentResource createExclusiveConsent(ConsentResource consentResource, String userID, - String authStatus, String authType, - String applicableExistingConsentsStatus, - String newExistingConsentStatus, - boolean isImplicitAuth) - throws ConsentManagementException { - - if (StringUtils.isBlank(consentResource.getClientID()) || StringUtils.isBlank(consentResource.getReceipt()) || - StringUtils.isBlank(consentResource.getConsentType()) || - StringUtils.isBlank(consentResource.getCurrentStatus()) || StringUtils.isBlank(userID) - || StringUtils.isBlank(applicableExistingConsentsStatus) - || StringUtils.isBlank(newExistingConsentStatus)) { - - log.error(ConsentCoreServiceConstants.CREATE_EXCLUSIVE_CONSENT_MANDATORY_PARAMETER_MISSING_ERROR); - throw new ConsentManagementException(ConsentCoreServiceConstants - .CREATE_EXCLUSIVE_CONSENT_MANDATORY_PARAMETER_MISSING_ERROR); - } - - if (isImplicitAuth) { - if (StringUtils.isBlank(authStatus) || StringUtils.isBlank(authType)) { - log.error("Authorization status and authorization type is not found for implicit " + - "authorization creation"); - throw new ConsentManagementException("Cannot proceed with implicit authorization creation " + - "without Authorization Status and Authorization Type provided"); - } - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - - // Update existing consent statuses and revoke their account mappings - updateExistingConsentStatusesAndRevokeAccountMappings(connection, consentCoreDAO, consentResource, - userID, applicableExistingConsentsStatus, newExistingConsentStatus); - - // Create a new consent, audit record and authorization resource if allowed - DetailedConsentResource storedDetailedConsentResource = - createAuthorizableConesntWithAuditRecord(connection, consentCoreDAO, consentResource, userID, - authStatus, authType, isImplicitAuth); - - // Commit the transaction - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return storedDetailedConsentResource; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG); - } catch (OBConsentDataInsertionException e) { - log.error(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - } catch (OBConsentDataUpdationException e) { - log.error(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public boolean createConsentFile(ConsentFile consentFileResource, String newConsentStatus, String userID, - String applicableStatusToFileUpload) - throws ConsentManagementException { - - if (StringUtils.isBlank(consentFileResource.getConsentID()) || - StringUtils.isBlank(consentFileResource.getConsentFile())) { - - log.error("Consent ID or Consent File content is missing. Cannot proceed."); - throw new ConsentManagementException("Cannot proceed without consent ID and file content."); - } - - String consentID = consentFileResource.getConsentID(); - - if (StringUtils.isBlank(newConsentStatus) || StringUtils.isBlank(applicableStatusToFileUpload)) { - log.error(ConsentCoreServiceConstants.NEW_CONSENT_STATUS_OR_APPLICABLE_STATUS_MISSING_ERROR); - throw new ConsentManagementException(ConsentCoreServiceConstants - .NEW_CONSENT_STATUS_OR_APPLICABLE_STATUS_MISSING_ERROR); - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - // Get the existing consent to validate status - if (log.isDebugEnabled()) { - log.debug("Retrieving the consent for ID:" + consentID.replaceAll("[\r\n]", "") + - " to validate status"); - } - ConsentResource existingConsentResource = consentCoreDAO.getConsentResource(connection, consentID); - - String existingConsentStatus = existingConsentResource.getCurrentStatus(); - - // Validate status of the consent - if (!applicableStatusToFileUpload.equalsIgnoreCase(existingConsentResource.getCurrentStatus())) { - log.error("The consent is not in required state to proceed"); - throw new ConsentManagementException("The consent should be in the required state in order to " + - "proceed"); - } - - // Store the consent file - if (log.isDebugEnabled()) { - log.debug("Creating the consent file for the consent of ID:" + - consentID.replaceAll("[\r\n]", "")); - } - consentCoreDAO.storeConsentFile(connection, consentFileResource); - - // Update consent status with new status - if (log.isDebugEnabled()) { - log.debug("Updating the status of the consent for ID:" + consentID.replaceAll("[\r\n]", "")); - } - consentCoreDAO.updateConsentStatus(connection, consentID, newConsentStatus); - - // Create audit record and execute state change listener - HashMap consentDataMap = new HashMap<>(); - consentDataMap.put(ConsentCoreServiceConstants.CONSENT_RESOURCE, existingConsentResource); - postStateChange(connection, consentCoreDAO, consentID, userID, newConsentStatus, - existingConsentStatus, ConsentCoreServiceConstants.CONSENT_FILE_UPLOAD_REASON, - existingConsentResource.getClientID(), consentDataMap); - - // Commit transaction - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return true; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } catch (OBConsentDataInsertionException e) { - log.error(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - } catch (OBConsentDataUpdationException e) { - log.error(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public boolean revokeConsent(String consentID, String revokedConsentStatus) - throws ConsentManagementException { - return revokeConsentWithReason(consentID, revokedConsentStatus, null, true, - ConsentCoreServiceConstants.CONSENT_REVOKE_REASON); - } - - @Override - public boolean revokeConsentWithReason(String consentID, String revokedConsentStatus, String revokedReason) - throws ConsentManagementException { - return revokeConsentWithReason(consentID, revokedConsentStatus, null, true, revokedReason); - } - - @Override - public boolean revokeConsent(String consentID, String revokedConsentStatus, String userID) - throws ConsentManagementException { - return revokeConsentWithReason(consentID, revokedConsentStatus, userID, true, - ConsentCoreServiceConstants.CONSENT_REVOKE_REASON); - } - - @Override - public boolean revokeConsentWithReason(String consentID, String revokedConsentStatus, String userID, - String revokedReason) - throws ConsentManagementException { - return revokeConsentWithReason(consentID, revokedConsentStatus, userID, true, revokedReason); - } - - @Override - public boolean revokeConsent(String consentID, String revokedConsentStatus, String userID, - boolean shouldRevokeTokens) - throws ConsentManagementException { - return revokeConsentWithReason(consentID, revokedConsentStatus, userID, shouldRevokeTokens, - ConsentCoreServiceConstants.CONSENT_REVOKE_REASON); - } - - - @Override - public boolean revokeConsentWithReason(String consentID, String revokedConsentStatus, String userID, - boolean shouldRevokeTokens, String revokedReason) - throws ConsentManagementException { - - if (StringUtils.isBlank(consentID) || StringUtils.isBlank(revokedConsentStatus)) { - log.error("Consent ID or new consent status is missing, cannot proceed"); - throw new ConsentManagementException("Consent ID or new consent status is missing, cannot " + - "proceed"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - // Get existing detailed consent - if (log.isDebugEnabled()) { - log.debug("Retrieving existing consent of ID: " + consentID.replaceAll("[\r\n]", "") + - " for status validation"); - } - DetailedConsentResource retrievedDetailedConsentResource = consentCoreDAO - .getDetailedConsentResource(connection, consentID, false); - String previousConsentStatus = retrievedDetailedConsentResource.getCurrentStatus(); - - // Update consent status as revoked - if (log.isDebugEnabled()) { - log.debug("Updating the status of the consent of ID: " + consentID.replaceAll("[\r\n]", "")); - } - consentCoreDAO.updateConsentStatus(connection, consentID, revokedConsentStatus); - - if (shouldRevokeTokens) { - // Extract userId from authorizationResources - ArrayList authorizationResources = retrievedDetailedConsentResource - .getAuthorizationResources(); - - // Get all users of the consent - Set consentUserIDSet = new HashSet<>(); - if (authorizationResources != null && !authorizationResources.isEmpty()) { - for (AuthorizationResource authorizationResource : authorizationResources) { - consentUserIDSet.add(authorizationResource.getUserID()); - } - } - - if (consentUserIDSet.isEmpty()) { - log.error("User ID is required for token revocation, cannot proceed"); - throw new ConsentManagementException("User ID is required for token revocation, cannot " + - "proceed"); - } - - if (!isValidUserID(userID, consentUserIDSet)) { - final String errorMsg = "Requested UserID and Consent UserID do not match, cannot proceed."; - log.error(errorMsg + ", request UserID: " + userID.replaceAll("[\r\n]", "") + - " is not a member of the consent user list"); - throw new ConsentManagementException(errorMsg); - } - revokeTokens(retrievedDetailedConsentResource, userID); - } - - ArrayList consentMappingResources = retrievedDetailedConsentResource - .getConsentMappingResources(); - ArrayList mappingIDs = new ArrayList<>(); - - if (!consentMappingResources.isEmpty()) { - for (ConsentMappingResource resource : consentMappingResources) { - mappingIDs.add(resource.getMappingID()); - } - - // Update account mapping status as inactive - if (log.isDebugEnabled()) { - log.debug("Updating the account mappings of consent ID: " + - consentID.replaceAll("[\r\n]", "") + " as inactive"); - } - consentCoreDAO.updateConsentMappingStatus(connection, mappingIDs, - ConsentCoreServiceConstants.INACTIVE_MAPPING_STATUS); - } - - HashMap consentDataMap = new HashMap<>(); - // Get detailed consent status after the updates - DetailedConsentResource newDetailedConsentResource = - consentCoreDAO.getDetailedConsentResource(connection, consentID, false); - consentDataMap.put(ConsentCoreServiceConstants.DETAILED_CONSENT_RESOURCE, - newDetailedConsentResource); - - // Pass the previous status consent to persist as consent history - consentDataMap.put(ConsentCoreServiceConstants.CONSENT_AMENDMENT_HISTORY_RESOURCE, - retrievedDetailedConsentResource); - consentDataMap.put(ConsentCoreServiceConstants.CONSENT_AMENDMENT_TIME, System.currentTimeMillis()); - - // Create an audit record execute state change listener - postStateChange(connection, consentCoreDAO, consentID, userID, revokedConsentStatus, - previousConsentStatus, revokedReason, - retrievedDetailedConsentResource.getClientID(), consentDataMap); - - //Commit transaction - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } catch (OBConsentDataInsertionException e) { - log.error(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - } catch (OBConsentDataUpdationException e) { - log.error(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - } catch (IdentityOAuth2Exception e) { - log.error("Error while revoking tokens for the consent ID: " - + consentID.replaceAll("[\r\n]", ""), e); - throw new ConsentManagementException("Error occurred while revoking tokens for the consent ID: " - + consentID); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - return true; - } - - @Override - public boolean revokeExistingApplicableConsents(String clientID, String userID, String consentType, - String applicableStatusToRevoke, - String revokedConsentStatus, boolean shouldRevokeTokens) - throws ConsentManagementException { - - if (StringUtils.isBlank(clientID) || StringUtils.isBlank(revokedConsentStatus) || StringUtils.isBlank(userID) - || StringUtils.isBlank(applicableStatusToRevoke) || StringUtils.isBlank(consentType)) { - log.error("Client ID, new consent status, consent type, user ID or applicable consent status to revoke is" + - " missing, cannot proceed"); - throw new ConsentManagementException("Client ID, new consent status, consent type, user ID or applicable " + - "consent status to revoke is missing, cannot proceed"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - - ArrayList accountMappingIDsList = new ArrayList<>(); - ArrayList clientIDsList = new ArrayList<>(); - clientIDsList.add(clientID); - ArrayList userIDsList = new ArrayList<>(); - userIDsList.add(userID); - ArrayList consentTypesList = new ArrayList<>(); - consentTypesList.add(consentType); - ArrayList consentStatusesList = new ArrayList<>(); - consentStatusesList.add(applicableStatusToRevoke); - - // Get existing consents - log.debug("Retrieving existing consents"); - - // Only parameters needed for the search are provided, others are made null - ArrayList retrievedDetailedConsentResources = consentCoreDAO - .searchConsents(connection, null, clientIDsList, consentTypesList, - consentStatusesList, userIDsList, null, null, null, null); - - // Revoke existing consents and create audit records - for (DetailedConsentResource resource : retrievedDetailedConsentResources) { - String previousConsentStatus = resource.getCurrentStatus(); - - // Update consent status - if (log.isDebugEnabled()) { - log.debug("Updating consent status for consent ID: " + - resource.getConsentID().replaceAll("[\r\n]", "")); - } - consentCoreDAO.updateConsentStatus(connection, resource.getConsentID(), revokedConsentStatus); - - if (shouldRevokeTokens) { - revokeTokens(resource, userID); - } - - // Create an audit record for consent update - if (log.isDebugEnabled()) { - log.debug("Creating audit record for the status change of consent ID: " - + resource.getConsentID().replaceAll("[\r\n]", "")); - } - // Create an audit record execute state change listener - HashMap consentDataMap = new HashMap<>(); - consentDataMap.put(ConsentCoreServiceConstants.DETAILED_CONSENT_RESOURCE, resource); - postStateChange(connection, consentCoreDAO, resource.getConsentID(), userID, - revokedConsentStatus, previousConsentStatus, - ConsentCoreServiceConstants.CONSENT_REVOKE_REASON, resource.getClientID(), consentDataMap); - - // Extract account mapping IDs for retrieved applicable consents - if (log.isDebugEnabled()) { - log.debug("Extracting account mapping IDs from consent ID: " + - resource.getConsentID().replaceAll("[\r\n]", "")); - } - for (ConsentMappingResource mappingResource : resource.getConsentMappingResources()) { - accountMappingIDsList.add(mappingResource.getMappingID()); - } - } - - // Update account mappings as inactive - log.debug("Deactivating account mappings"); - if (accountMappingIDsList.size() > 0) { - consentCoreDAO.updateConsentMappingStatus(connection, accountMappingIDsList, - ConsentCoreServiceConstants.INACTIVE_MAPPING_STATUS); - } - //Commit transaction - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return true; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } catch (OBConsentDataInsertionException e) { - log.error(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - } catch (OBConsentDataUpdationException e) { - log.error(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - } catch (IdentityOAuth2Exception e) { - log.error("Error while revoking tokens for existing consents", e); - throw new ConsentManagementException("Error occurred while revoking tokens for existing consents"); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public ConsentResource getConsent(String consentID, boolean withAttributes) - throws ConsentManagementException { - - if (StringUtils.isBlank(consentID)) { - log.error("Consent ID is missing, cannot proceed"); - throw new ConsentManagementException("Consent ID is missing, cannot proceed"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - ConsentResource retrievedConsentResource; - - // Get consent attributes if needed - if (!withAttributes) { - if (log.isDebugEnabled()) { - log.debug("Retrieving consent for consent ID: " + consentID.replaceAll("[\r\n]", "")); - } - retrievedConsentResource = consentCoreDAO.getConsentResource(connection, consentID); - } else { - if (log.isDebugEnabled()) { - log.debug("Retrieving consent with consent attributes for consent ID: " + - consentID.replaceAll("[\r\n]", "")); - } - retrievedConsentResource = consentCoreDAO.getConsentResourceWithAttributes(connection, consentID); - } - - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return retrievedConsentResource; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public boolean storeConsentAttributes(String consentID, Map consentAttributes) - throws ConsentManagementException { - - boolean isConsentAttributesStored; - - if (StringUtils.isBlank(consentID) || consentAttributes == null || consentAttributes.isEmpty()) { - - log.error("consentID or consentAttributes is missing, cannot proceed"); - throw new ConsentManagementException("Cannot proceed since consentID or consentAttributes is missing."); - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - ConsentAttributes consentAttributesObject = new ConsentAttributes(); - consentAttributesObject.setConsentID(consentID); - consentAttributesObject.setConsentAttributes(consentAttributes); - - if (log.isDebugEnabled()) { - log.debug("Storing consent attributes for the consent of ID: " + - consentID.replaceAll("[\r\n]", "")); - } - isConsentAttributesStored = consentCoreDAO.storeConsentAttributes(connection, consentAttributesObject); - DatabaseUtil.commitTransaction(connection); - } catch (OBConsentDataInsertionException e) { - log.error(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - - return isConsentAttributesStored; - - } - - @Override - public ConsentAttributes getConsentAttributes(String consentID, ArrayList consentAttributeKeys) - throws ConsentManagementException { - - if (StringUtils.isBlank(consentID) || CollectionUtils.isEmpty(consentAttributeKeys)) { - log.error("Consent ID or consent attributes keys are missing, cannot proceed"); - throw new ConsentManagementException("Consent ID or consent attribute keys are missing, cannot proceed"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - ConsentAttributes retrievedConsentAttributes; - if (log.isDebugEnabled()) { - log.debug("Retrieving consent attributes for given keys for consent ID: " + - consentID.replaceAll("[\r\n]", "")); - } - retrievedConsentAttributes = consentCoreDAO.getConsentAttributes(connection, consentID, - consentAttributeKeys); - - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return retrievedConsentAttributes; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public ConsentAttributes getConsentAttributes(String consentID) - throws ConsentManagementException { - - if (StringUtils.isBlank(consentID)) { - log.error("Consent ID is missing, cannot proceed"); - throw new ConsentManagementException("Consent ID is missing, cannot proceed"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - ConsentAttributes retrievedConsentAttributes; - if (log.isDebugEnabled()) { - log.debug("Retrieving consent attributes for consent ID: " + - consentID.replaceAll("[\r\n]", "")); - } - retrievedConsentAttributes = consentCoreDAO.getConsentAttributes(connection, consentID); - - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return retrievedConsentAttributes; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public Map getConsentAttributesByName(String attributeName) throws ConsentManagementException { - - if (StringUtils.isBlank(attributeName)) { - log.error("Attribute name is not provided, cannot proceed"); - throw new ConsentManagementException("Attribute name is not provided, cannot proceed"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - Map retrievedAttributeValuesMap; - if (log.isDebugEnabled()) { - log.debug("Retrieving attribute values for the provided attribute key: " + - attributeName.replaceAll("[\r\n]", "")); - } - retrievedAttributeValuesMap = consentCoreDAO.getConsentAttributesByName(connection, attributeName); - - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return retrievedAttributeValuesMap; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public ArrayList getConsentIdByConsentAttributeNameAndValue(String attributeName, String attributeValue) - throws ConsentManagementException { - - if (StringUtils.isBlank(attributeName) || StringUtils.isBlank(attributeValue)) { - log.error("Attribute name or value is not provided, cannot proceed"); - throw new ConsentManagementException("Attribute name or value is not provided, cannot proceed"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - ArrayList retrievedConsentIdList; - if (log.isDebugEnabled()) { - log.debug("Retrieving consent Id for the provided attribute key : " + - attributeName.replaceAll("[\r\n]", "") + " and " + - "attribute value : " + attributeValue.replaceAll("[\r\n]", "")); - } - retrievedConsentIdList = consentCoreDAO.getConsentIdByConsentAttributeNameAndValue(connection, - attributeName, attributeValue); - - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return retrievedConsentIdList; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public boolean deleteConsentAttributes(String consentID, ArrayList attributeKeysList) - throws ConsentManagementException { - - if (StringUtils.isBlank(consentID) || CollectionUtils.isEmpty(attributeKeysList)) { - log.error("Consent ID or attributes list is not provided, cannot proceed"); - throw new ConsentManagementException("Consent ID or attributes list is not provided, cannot proceed"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - if (log.isDebugEnabled()) { - log.debug("Deleting attributes for the consent ID: " + consentID.replaceAll("[\r\n]", "")); - } - consentCoreDAO.deleteConsentAttributes(connection, consentID, attributeKeysList); - - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return true; - } catch (OBConsentDataDeletionException e) { - log.error(ConsentCoreServiceConstants.DATA_DELETE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.CONSENT_ATTRIBUTES_DELETE_ERROR_MSG); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public ConsentFile getConsentFile(String consentID) throws ConsentManagementException { - - if (StringUtils.isBlank(consentID)) { - log.error("Consent ID is missing, cannot proceed"); - throw new ConsentManagementException("Consent ID is missing, cannot proceed"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - ConsentFile retrievedConsentFileResource; - - // Get consent file - if (log.isDebugEnabled()) { - log.debug("Retrieving consent file resource for consent ID: " + - consentID.replaceAll("[\r\n]", "")); - } - retrievedConsentFileResource = consentCoreDAO.getConsentFile(connection, consentID, false); - - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return retrievedConsentFileResource; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public AuthorizationResource getAuthorizationResource(String authorizationID) throws ConsentManagementException { - - if (StringUtils.isBlank(authorizationID)) { - log.error("Authorization ID is missing, cannot proceed"); - throw new ConsentManagementException("Authorization ID is missing, cannot proceed"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - AuthorizationResource retrievedAuthorizationResource; - - // Get consent file - if (log.isDebugEnabled()) { - log.debug("Retrieving authorization resource for authorization ID: " + - authorizationID.replaceAll("[\r\n]", "")); - } - retrievedAuthorizationResource = consentCoreDAO.getAuthorizationResource(connection, authorizationID); - - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return retrievedAuthorizationResource; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public ArrayList searchConsentStatusAuditRecords(String consentID, String status, - String actionBy, Long fromTime, - Long toTime, String statusAuditID) - throws ConsentManagementException { - - ArrayList auditRecords; - Connection connection = DatabaseUtil.getDBConnection(); - - try { - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - - log.debug("Searching audit records"); - auditRecords = consentCoreDAO.getConsentStatusAuditRecords(connection, consentID, status, actionBy, - fromTime, toTime, statusAuditID, false); - - } catch (OBConsentDataRetrievalException e) { - log.error("Error occurred while searching audit records"); - throw new ConsentManagementException("Error occurred while searching audit records", e); - } - - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - return auditRecords; - } - - @Override - public boolean reAuthorizeExistingAuthResource(String consentID, String authID, String userID, - Map> accountIDsMapWithPermissions, - String currentConsentStatus, String newConsentStatus) - throws ConsentManagementException { - - if (StringUtils.isBlank(consentID) || StringUtils.isBlank(authID) || StringUtils.isBlank(userID) - || MapUtils.isEmpty(accountIDsMapWithPermissions) || StringUtils.isBlank(newConsentStatus) - || StringUtils.isBlank(currentConsentStatus)) { - log.error("Consent ID, auth ID, user ID, account permissions map, applicable consent status, new consent " + - "status or current consent status is not present, cannot proceed"); - throw new ConsentManagementException("Consent ID, auth ID, user ID, account permissions map, applicable " + - "consent status, new consent status or current consent status is not present, cannot proceed"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - - // Get detailed consent to retrieve account mappings - DetailedConsentResource detailedConsentResource = - consentCoreDAO.getDetailedConsentResource(connection, consentID, false); - - // Update accounts if required - updateAccounts(connection, consentCoreDAO, authID, accountIDsMapWithPermissions, - detailedConsentResource, false); - - // Update consent status - consentCoreDAO.updateConsentStatus(connection, consentID, newConsentStatus); - - // Create an audit record execute state change listener - HashMap consentDataMap = new HashMap<>(); - consentDataMap.put(ConsentCoreServiceConstants.DETAILED_CONSENT_RESOURCE, detailedConsentResource); - postStateChange(connection, consentCoreDAO, consentID, userID, newConsentStatus, - currentConsentStatus, ConsentCoreServiceConstants.CONSENT_REAUTHORIZE_REASON, - detailedConsentResource.getClientID(), consentDataMap); - - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return true; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } catch (OBConsentDataInsertionException e) { - log.error(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - } catch (OBConsentDataUpdationException e) { - log.error(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public boolean reAuthorizeConsentWithNewAuthResource(String consentID, String userID, Map> accountIDsMapWithPermissions, String currentConsentStatus, String newConsentStatus, - String newExistingAuthStatus, String newAuthStatus, - String newAuthType) - throws ConsentManagementException { - - if (StringUtils.isBlank(consentID) || StringUtils.isBlank(userID) - || MapUtils.isEmpty(accountIDsMapWithPermissions) || StringUtils.isBlank(newConsentStatus) - || StringUtils.isBlank(currentConsentStatus) || StringUtils.isBlank(newExistingAuthStatus) - || StringUtils.isBlank(newAuthStatus) || StringUtils.isBlank(newAuthType)) { - log.error("Consent ID, user ID, account permissions map, current consent status, new consent " + - "status, new existing auth status, new auth status or new auth type is not present, cannot " + - "proceed"); - throw new ConsentManagementException("Consent ID, user ID, account permissions map, current consent " + - "status, new consent status, new existing auth status, new auth status or new auth type is not " + - "present, cannot proceed"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - - // Get authorizations related to current consent to revoke - ArrayList authorizationResources = - consentCoreDAO.searchConsentAuthorizations(connection, consentID, userID); - - ArrayList mappingResourcesToDeactivate - = new ArrayList(); - for (AuthorizationResource resource : authorizationResources) { - // Update existing authorizations - consentCoreDAO.updateAuthorizationStatus(connection, resource.getAuthorizationID(), - newExistingAuthStatus); - mappingResourcesToDeactivate.addAll(consentCoreDAO.getConsentMappingResources(connection, - resource.getAuthorizationID())); - } - - // Deactivate account mappings of old auth resource. - ArrayList mappingIdsToDeactivate = new ArrayList<>(); - for (ConsentMappingResource resource : mappingResourcesToDeactivate) { - mappingIdsToDeactivate.add(resource.getMappingID()); - } - consentCoreDAO.updateConsentMappingStatus(connection, mappingIdsToDeactivate, - ConsentCoreServiceConstants.INACTIVE_MAPPING_STATUS); - - // Create a new authorization resource for the consent - AuthorizationResource newAuthorizationResource = new AuthorizationResource(); - newAuthorizationResource.setConsentID(consentID); - newAuthorizationResource.setAuthorizationType(newAuthType); - newAuthorizationResource.setAuthorizationStatus(newAuthStatus); - newAuthorizationResource.setUserID(userID); - consentCoreDAO.storeAuthorizationResource(connection, newAuthorizationResource); - - // Retrieve the detailed consent for obtaining relative account mappings - DetailedConsentResource detailedConsentResource = - consentCoreDAO.getDetailedConsentResource(connection, consentID, false); - - // Update accounts if required - updateAccounts(connection, consentCoreDAO, newAuthorizationResource.getAuthorizationID(), - accountIDsMapWithPermissions, detailedConsentResource, true); - - // Update consent status - consentCoreDAO.updateConsentStatus(connection, consentID, newConsentStatus); - - // Create an audit record execute state change listener - HashMap consentDataMap = new HashMap<>(); - consentDataMap.put(ConsentCoreServiceConstants.DETAILED_CONSENT_RESOURCE, detailedConsentResource); - postStateChange(connection, consentCoreDAO, consentID, userID, newConsentStatus, - currentConsentStatus, ConsentCoreServiceConstants.CONSENT_REAUTHORIZE_REASON, - detailedConsentResource.getClientID(), consentDataMap); - - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return true; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } catch (OBConsentDataInsertionException e) { - log.error(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - } catch (OBConsentDataUpdationException e) { - log.error(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public DetailedConsentResource getDetailedConsent(String consentID) throws ConsentManagementException { - - if (StringUtils.isBlank(consentID)) { - log.error("Consent ID is missing, cannot proceed"); - throw new ConsentManagementException("Consent ID is missing, cannot proceed"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - DetailedConsentResource retrievedDetailedConsentResource; - - // Retrieve the detailed consent resource - if (log.isDebugEnabled()) { - log.debug("Retrieving detailed consent for consent ID: " + consentID.replaceAll("[\r\n]", "")); - } - retrievedDetailedConsentResource = consentCoreDAO.getDetailedConsentResource(connection, consentID, - false); - - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return retrievedDetailedConsentResource; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public AuthorizationResource createConsentAuthorization(AuthorizationResource authorizationResource) - throws ConsentManagementException { - - if (StringUtils.isBlank(authorizationResource.getConsentID()) || - StringUtils.isBlank(authorizationResource.getAuthorizationType()) || - StringUtils.isBlank(authorizationResource.getAuthorizationStatus())) { - - log.error("Consent ID, authorization type, user ID or authorization status is missing, cannot proceed"); - throw new ConsentManagementException("Cannot proceed since consent ID, authorization type, user ID or " + - "authorization status is missing"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - // Create authorization resource - if (log.isDebugEnabled()) { - log.debug("Creating authorization resource for the consent of ID: " + authorizationResource - .getConsentID().replaceAll("[\r\n]", "")); - } - AuthorizationResource storedAuthorizationResource = - consentCoreDAO.storeAuthorizationResource(connection, authorizationResource); - - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return storedAuthorizationResource; - } catch (OBConsentDataInsertionException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public ArrayList createConsentAccountMappings(String authID, Map> accountIDsMapWithPermissions) throws ConsentManagementException { - - if (StringUtils.isBlank(authID) || MapUtils.isEmpty(accountIDsMapWithPermissions)) { - log.error("Authorization ID, accountID/permission map is not found, cannot " + - "proceed"); - throw new ConsentManagementException("Authorization ID, accountID/permission map " + - "is not found, cannot proceed"); - } - - ArrayList storedConsentMappingResources = new ArrayList<>(); - Connection connection = DatabaseUtil.getDBConnection(); - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - // Create account mapping resources - if (log.isDebugEnabled()) { - log.debug("Creating consent account mapping resources for authorization ID: " + - authID.replaceAll("[\r\n]", "")); - } - for (Map.Entry> entry : accountIDsMapWithPermissions.entrySet()) { - String accountID = entry.getKey(); - for (String value : entry.getValue()) { - ConsentMappingResource consentMappingResource = new ConsentMappingResource(); - consentMappingResource.setAccountID(accountID); - consentMappingResource.setPermission(value); - consentMappingResource.setAuthorizationID(authID); - consentMappingResource.setMappingStatus(ConsentCoreServiceConstants.ACTIVE_MAPPING_STATUS); - storedConsentMappingResources.add(consentCoreDAO.storeConsentMappingResource(connection, - consentMappingResource)); - } - } - - // Commit transaction - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return storedConsentMappingResources; - } catch (OBConsentDataInsertionException e) { - log.error(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public boolean deactivateAccountMappings(ArrayList accountMappingIDs) throws ConsentManagementException { - - if (accountMappingIDs.isEmpty()) { - log.error("Account mapping IDs are not provided, cannot proceed"); - throw new ConsentManagementException("Cannot proceed since account mapping IDs are not provided"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - // Deactivate account mapping resources - log.debug("Deactivating consent account mapping resources for given mapping IDs"); - - consentCoreDAO.updateConsentMappingStatus(connection, accountMappingIDs, - ConsentCoreServiceConstants.INACTIVE_MAPPING_STATUS); - - // Commit transaction - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return true; - } catch (OBConsentDataUpdationException e) { - log.error(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - @Override - public boolean updateAccountMappingStatus(ArrayList accountMappingIDs, String newMappingStatus) throws - ConsentManagementException { - - if (accountMappingIDs.isEmpty()) { - log.error("Account mapping IDs are not provided, cannot proceed"); - throw new ConsentManagementException("Cannot proceed since account mapping IDs are not provided"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - // update account mapping resources - log.debug("Deactivating consent account mapping resources for given mapping IDs"); - - consentCoreDAO.updateConsentMappingStatus(connection, accountMappingIDs, - newMappingStatus); - - // Commit transaction - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return true; - } catch (OBConsentDataUpdationException e) { - log.error(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public boolean updateAccountMappingPermission(Map mappingIDPermissionMap) throws - ConsentManagementException { - - if (mappingIDPermissionMap.isEmpty()) { - log.error("Account mapping IDs are not provided, cannot proceed"); - throw new ConsentManagementException("Cannot proceed since account mapping IDs are not provided"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - log.debug("Updating consent account mapping permissions for given mapping IDs"); - consentCoreDAO.updateConsentMappingPermission(connection, mappingIDPermissionMap); - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return true; - } catch (OBConsentDataUpdationException e) { - log.error(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public ArrayList searchDetailedConsents(ArrayList consentIDs, - ArrayList clientIDs, - ArrayList consentTypes, - ArrayList consentStatuses, - ArrayList userIDs, Long fromTime, - Long toTime, - Integer limit, Integer offset) - throws ConsentManagementException { - - // Input parameters except limit and offset are not validated since they are validated in the DAO method - ArrayList detailedConsentResources; - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - - log.debug("Searching detailed consents"); - detailedConsentResources = consentCoreDAO.searchConsents(connection, consentIDs, clientIDs, - consentTypes, consentStatuses, userIDs, fromTime, toTime, limit, offset); - - } catch (OBConsentDataRetrievalException e) { - log.error("Error occurred while searching detailed consents", e); - throw new ConsentManagementException("Error occurred while searching detailed consents", e); - } - - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - return detailedConsentResources; - } - - @Override - public ArrayList searchDetailedConsents(ArrayList consentIDs, - ArrayList clientIDs, - ArrayList consentTypes, - ArrayList consentStatuses, - ArrayList userIDs, Long fromTime, - Long toTime, Integer limit, Integer offset, - boolean fetchFromRetentionDatabase) - throws ConsentManagementException { - - // Input parameters except limit and offset are not validated since they are validated in the DAO method - ArrayList detailedConsentResources; - - Connection connection; - if (fetchFromRetentionDatabase) { - connection = DatabaseUtil.getRetentionDBConnection(); - } else { - connection = DatabaseUtil.getDBConnection(); - } - - try { - try { - ConsentCoreDAO consentCoreDAO; - if (fetchFromRetentionDatabase) { - consentCoreDAO = ConsentStoreInitializer.getInitializedConsentRetentionDAOImpl(); - } else { - consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - } - - log.debug("Searching detailed consents"); - detailedConsentResources = consentCoreDAO.searchConsents(connection, consentIDs, clientIDs, - consentTypes, consentStatuses, userIDs, fromTime, toTime, limit, offset); - - } catch (OBConsentDataRetrievalException e) { - log.error("Error occurred while searching detailed consents", e); - throw new ConsentManagementException("Error occurred while searching detailed consents", e); - } - - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - return detailedConsentResources; - } - - @Override - public boolean bindUserAccountsToConsent(ConsentResource consentResource, String userID, - String authID, ArrayList accountIDs, - String newAuthStatus, - String newCurrentConsentStatus) - throws ConsentManagementException { - - Map> accountIDsMapWithPermissions = new HashMap<>(); - ArrayList permissionsDefault = new ArrayList<>(); - permissionsDefault.add("n/a"); - - for (String accountId : accountIDs) { - accountIDsMapWithPermissions.put(accountId, permissionsDefault); - } - - return bindUserAccountsToConsent(consentResource, userID, authID, accountIDsMapWithPermissions, newAuthStatus, - newCurrentConsentStatus); - } - - @Override - public boolean bindUserAccountsToConsent(ConsentResource consentResource, String userID, - String authID, Map> accountIDsMapWithPermissions, - String newAuthStatus, - String newCurrentConsentStatus) - throws ConsentManagementException { - - String consentID = consentResource.getConsentID(); - String clientID = consentResource.getClientID(); - String consentType = consentResource.getConsentType(); - - if (StringUtils.isBlank(consentID) || StringUtils.isBlank(clientID) || StringUtils.isBlank(consentType) - || StringUtils.isBlank(userID) || StringUtils.isBlank(authID) || StringUtils.isBlank(newAuthStatus) - || StringUtils.isBlank(newCurrentConsentStatus)) { - log.error("Consent ID, client ID, consent type, user ID, authorization ID, new authorization status or " + - "new consent status is " + - "missing, cannot proceed."); - throw new ConsentManagementException("Consent ID, client ID, consent type, user ID, authorization ID, new" + - " authorization status or new consent status is missing, " + - "cannot proceed"); - } - - if (MapUtils.isEmpty(accountIDsMapWithPermissions)) { - log.error("Account IDs and relative permissions are not present, cannot proceed"); - throw new ConsentManagementException("Account IDs and relative permissions are not present, cannot " + - "proceed"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - try { - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - - // Update authorization resource of current consent - if (log.isDebugEnabled()) { - log.debug("Update authorization status and authorization user for current consent ID: " - + consentID.replaceAll("[\r\n]", "")); - } - consentCoreDAO.updateAuthorizationUser(connection, authID, userID); - consentCoreDAO.updateAuthorizationStatus(connection, authID, newAuthStatus); - - // Create account mappings for current consent - if (log.isDebugEnabled()) { - log.debug("Creating account mappings for current consent ID: " + - consentID.replaceAll("[\r\n]", "")); - } - for (Map.Entry> entry : accountIDsMapWithPermissions.entrySet()) { - String accountID = entry.getKey(); - for (String value : entry.getValue()) { - ConsentMappingResource consentMappingResource = new ConsentMappingResource(); - consentMappingResource.setAccountID(accountID); - consentMappingResource.setPermission(value); - consentMappingResource.setAuthorizationID(authID); - consentMappingResource.setMappingStatus(ConsentCoreServiceConstants.ACTIVE_MAPPING_STATUS); - consentCoreDAO.storeConsentMappingResource(connection, consentMappingResource); - } - } - - // Update current consent status - if (log.isDebugEnabled()) { - log.debug("Update the status of the current consent ID: " + consentID.replaceAll("[\r\n]", "")); - } - consentCoreDAO.updateConsentStatus(connection, consentID, newCurrentConsentStatus); - - // Create audit record for the consent status update and execute the state change listener - HashMap consentDataMap = new HashMap<>(); - consentDataMap.put(ConsentCoreServiceConstants.CONSENT_RESOURCE, consentResource); - postStateChange(connection, consentCoreDAO, consentID, userID, newCurrentConsentStatus, - consentResource.getCurrentStatus(), ConsentCoreServiceConstants.USER_ACCOUNTS_BINDING_REASON, - clientID, consentDataMap); - - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return true; - } catch (OBConsentDataInsertionException e) { - log.error(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - } catch (OBConsentDataUpdationException e) { - log.error(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public ArrayList searchAuthorizations(String consentID) - throws ConsentManagementException { - return searchAuthorizations(consentID, null); - } - - @Override - public ArrayList searchAuthorizationsForUser(String userID) - throws ConsentManagementException { - return searchAuthorizations(null, userID); - } - - @Override - public ArrayList searchAuthorizations(String consentID, String userID) - throws ConsentManagementException { - - ArrayList authorizationResources; - Connection connection = DatabaseUtil.getDBConnection(); - - try { - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - - log.debug("Searching authorization resources"); - authorizationResources = consentCoreDAO.searchConsentAuthorizations(connection, consentID, userID); - - } catch (OBConsentDataRetrievalException e) { - log.error("Error occurred while searching authorization resources", e); - throw new ConsentManagementException("Error occurred while searching authorization resources", e); - } - - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - return authorizationResources; - } - - private void createAuditRecord(Connection connection, ConsentCoreDAO consentCoreDAO, String consentID, - String userID, String newConsentStatus, String previousConsentStatus, String reason) - throws OBConsentDataInsertionException { - - // Create an audit record - ConsentStatusAuditRecord consentStatusAuditRecord = new ConsentStatusAuditRecord(); - consentStatusAuditRecord.setConsentID(consentID); - consentStatusAuditRecord.setCurrentStatus(newConsentStatus); - consentStatusAuditRecord.setReason(reason); - if (StringUtils.isNotEmpty(userID)) { - consentStatusAuditRecord.setActionBy(userID); - } else { - consentStatusAuditRecord.setActionBy(null); - } - consentStatusAuditRecord.setPreviousStatus(previousConsentStatus); - - if (log.isDebugEnabled()) { - log.debug(("Storing audit record for consent of ID: " + - consentStatusAuditRecord.getConsentID()).replaceAll("[\r\n]", "")); - } - consentCoreDAO.storeConsentStatusAuditRecord(connection, consentStatusAuditRecord); - } - - private DetailedConsentResource createAuthorizableConesntWithAuditRecord(Connection connection, - ConsentCoreDAO consentCoreDAO, - ConsentResource consentResource, - String userID, String authStatus, - String authType, - boolean isImplicitAuthorization) - throws OBConsentDataInsertionException, ConsentManagementException { - - boolean isConsentAttributesStored = false; - AuthorizationResource storedAuthorizationResource = null; - - // Create consent - if (log.isDebugEnabled()) { - log.debug(("Creating the consent for ID:" + consentResource.getConsentID()).replaceAll("[\r\n]", "")); - } - ConsentResource storedConsentResource = consentCoreDAO.storeConsentResource(connection, - consentResource); - String consentID = storedConsentResource.getConsentID(); - - // Store consent attributes if available - if (MapUtils.isNotEmpty(consentResource.getConsentAttributes())) { - ConsentAttributes consentAttributes = new ConsentAttributes(); - consentAttributes.setConsentID(consentID); - consentAttributes.setConsentAttributes(consentResource.getConsentAttributes()); - - if (log.isDebugEnabled()) { - log.debug("Storing consent attributes for the consent of ID: " + consentAttributes - .getConsentID().replaceAll("[\r\n]", "")); - } - isConsentAttributesStored = consentCoreDAO.storeConsentAttributes(connection, consentAttributes); - } - - /* Create audit record, setting previous consent status as null since this is the first time the - consent is created and execute state change listener */ - HashMap consentDataMap = new HashMap<>(); - consentDataMap.put(ConsentCoreServiceConstants.CONSENT_RESOURCE, consentResource); - postStateChange(connection, consentCoreDAO, consentID, userID, consentResource.getCurrentStatus(), - null, ConsentCoreServiceConstants.CREATE_CONSENT_REASON, - consentResource.getClientID(), consentDataMap); - - // Create an authorization resource if isImplicitAuth parameter is true - if (isImplicitAuthorization) { - AuthorizationResource authorizationResource = new AuthorizationResource(); - authorizationResource.setConsentID(consentID); - authorizationResource.setAuthorizationStatus(authStatus); - authorizationResource.setAuthorizationType(authType); - if (StringUtils.isNotBlank(userID)) { - authorizationResource.setUserID(userID); - } else { - /* Setting userID as null since at this point, there is no userID in this flow. User ID can be - updated in authorization flow */ - authorizationResource.setUserID(null); - } - - if (log.isDebugEnabled()) { - log.debug(("Storing authorization resource for consent of ID: " + authorizationResource - .getConsentID()).replaceAll("[\r\n]", "")); - } - - storedAuthorizationResource = consentCoreDAO.storeAuthorizationResource(connection, - authorizationResource); - } - - DetailedConsentResource detailedConsentResource = new DetailedConsentResource(); - detailedConsentResource.setConsentID(consentID); - detailedConsentResource.setClientID(storedConsentResource.getClientID()); - detailedConsentResource.setReceipt(storedConsentResource.getReceipt()); - detailedConsentResource.setConsentType(storedConsentResource.getConsentType()); - detailedConsentResource.setCurrentStatus(storedConsentResource.getCurrentStatus()); - detailedConsentResource.setConsentFrequency(storedConsentResource.getConsentFrequency()); - detailedConsentResource.setValidityPeriod(storedConsentResource.getValidityPeriod()); - detailedConsentResource.setCreatedTime(storedConsentResource.getCreatedTime()); - detailedConsentResource.setRecurringIndicator(storedConsentResource.isRecurringIndicator()); - detailedConsentResource.setUpdatedTime(storedConsentResource.getUpdatedTime()); - - if (isConsentAttributesStored) { - detailedConsentResource.setConsentAttributes(consentResource.getConsentAttributes()); - } - if (isImplicitAuthorization) { - ArrayList authorizationResources = new ArrayList<>(); - authorizationResources.add(storedAuthorizationResource); - detailedConsentResource.setAuthorizationResources(authorizationResources); - } - return detailedConsentResource; - } - - private void updateExistingConsentStatusesAndRevokeAccountMappings(Connection connection, - ConsentCoreDAO consentCoreDAO, - ConsentResource consentResource, String userID, - String applicableExistingConsentsStatus, - String newExistingConsentStatus) - throws OBConsentDataRetrievalException, OBConsentDataUpdationException, OBConsentDataInsertionException, - ConsentManagementException { - - ArrayList accountMappingIDsList = new ArrayList<>(); - ArrayList clientIDsList = new ArrayList<>(); - clientIDsList.add(consentResource.getClientID()); - ArrayList userIDsList = new ArrayList<>(); - userIDsList.add(userID); - ArrayList consentTypesList = new ArrayList<>(); - consentTypesList.add(consentResource.getConsentType()); - ArrayList consentStatusesList = new ArrayList<>(); - consentStatusesList.add(applicableExistingConsentsStatus); - - // Get existing applicable consents - log.debug("Retrieving existing authorized consents"); - ArrayList retrievedExistingAuthorizedConsentsList = - consentCoreDAO.searchConsents(connection, null, clientIDsList, consentTypesList, - consentStatusesList, userIDsList, null, null, null, - null); - - for (DetailedConsentResource resource : retrievedExistingAuthorizedConsentsList) { - - String previousConsentStatus = resource.getCurrentStatus(); - - // Update existing consents as necessary - if (log.isDebugEnabled()) { - log.debug(("Updating existing consent statuses with the new status provided for consent ID: " - + resource.getConsentID()).replaceAll("[\r\n]", "")); - } - consentCoreDAO.updateConsentStatus(connection, resource.getConsentID(), - newExistingConsentStatus); - - // Create audit record for each consent update - if (log.isDebugEnabled()) { - log.debug(("Creating audit record for the consent update of consent ID: " - + resource.getConsentID()).replaceAll("[\r\n]", "")); - } - // Create an audit record execute state change listener - HashMap consentDataMap = new HashMap<>(); - consentDataMap.put(ConsentCoreServiceConstants.DETAILED_CONSENT_RESOURCE, resource); - postStateChange(connection, consentCoreDAO, resource.getConsentID(), userID, - newExistingConsentStatus, previousConsentStatus, - ConsentCoreServiceConstants.CREATE_EXCLUSIVE_AUTHORIZATION_CONSENT_REASON, resource.getClientID(), - consentDataMap); - - // Extract account mapping IDs for retrieved applicable consents - if (log.isDebugEnabled()) { - log.debug(("Extracting account mapping IDs from consent ID: " + - resource.getConsentID()).replaceAll("[\r\n]", "")); - } - for (ConsentMappingResource mappingResource : resource.getConsentMappingResources()) { - accountMappingIDsList.add(mappingResource.getMappingID()); - } - } - - // Update account mappings as inactive - log.debug("Deactivating account mappings"); - consentCoreDAO.updateConsentMappingStatus(connection, accountMappingIDsList, - ConsentCoreServiceConstants.INACTIVE_MAPPING_STATUS); - } - - private void updateAccounts(Connection connection, - ConsentCoreDAO consentCoreDAO, String authID, - Map> accountIDsMapWithPermissions, - DetailedConsentResource detailedConsentResource, boolean isNewAuthResource) - throws OBConsentDataInsertionException, OBConsentDataUpdationException { - - // Get existing consent account mappings - log.debug("Retrieve existing active account mappings"); - ArrayList existingAccountMappings = - detailedConsentResource.getConsentMappingResources(); - - // Determine unique account IDs - HashSet existingAccountIDs = new HashSet<>(); - for (ConsentMappingResource resource : existingAccountMappings) { - existingAccountIDs.add(resource.getAccountID()); - } - - ArrayList existingAccountIDsList = new ArrayList<>(existingAccountIDs); - - ArrayList reAuthorizedAccounts = new ArrayList<>(); - for (Map.Entry> entry : accountIDsMapWithPermissions.entrySet()) { - String accountID = entry.getKey(); - reAuthorizedAccounts.add(accountID); - } - - // Determine whether the account should be removed or added - ArrayList accountsToRevoke = new ArrayList<>(existingAccountIDsList); - accountsToRevoke.removeAll(reAuthorizedAccounts); - - ArrayList accountsToAdd = new ArrayList<>(reAuthorizedAccounts); - - if (isNewAuthResource) { - ArrayList commonAccountsFromReAuth = new ArrayList<>(existingAccountIDs); - commonAccountsFromReAuth.retainAll(accountsToAdd); - accountsToAdd.removeAll(existingAccountIDs); - accountsToAdd.addAll(commonAccountsFromReAuth); - } else { - accountsToAdd.removeAll(existingAccountIDs); - } - - if (!accountsToAdd.isEmpty()) { - // Store accounts as consent account mappings - log.debug("Add extra accounts as account mappings"); - for (String accountID : accountsToAdd) { - ArrayList permissions = accountIDsMapWithPermissions.get(accountID); - for (String permission : permissions) { - ConsentMappingResource consentMappingResource = new ConsentMappingResource(); - consentMappingResource.setAuthorizationID(authID); - consentMappingResource.setAccountID(accountID); - consentMappingResource.setPermission(permission); - consentMappingResource.setMappingStatus(ConsentCoreServiceConstants.ACTIVE_MAPPING_STATUS); - consentCoreDAO.storeConsentMappingResource(connection, consentMappingResource); - } - } - } - if (!accountsToRevoke.isEmpty()) { - // Update mapping statuses of revoking accounts to inactive - log.debug("Deactivate unwanted account mappings"); - ArrayList mappingIDsToUpdate = new ArrayList<>(); - for (String accountID : accountsToRevoke) { - for (ConsentMappingResource resource : existingAccountMappings) { - if (accountID.equalsIgnoreCase(resource.getAccountID())) { - mappingIDsToUpdate.add(resource.getMappingID()); - } - } - } - consentCoreDAO.updateConsentMappingStatus(connection, mappingIDsToUpdate, - ConsentCoreServiceConstants.INACTIVE_MAPPING_STATUS); - } - } - - @Override - public ConsentResource amendConsentData(String consentID, String consentReceipt, Long consentValidityTime, - String userID) - throws ConsentManagementException { - - if (StringUtils.isBlank(consentID) || - (StringUtils.isBlank(consentReceipt) && (consentValidityTime == null))) { - log.error("Consent ID or both consent receipt and consent validity period are not provided, cannot " + - "proceed"); - throw new ConsentManagementException("Consent ID or both consent receipt and consent validity period are " + - "not provided, cannot proceed"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - // Update receipt and validity time accordingly - if (StringUtils.isNotBlank(consentReceipt) && (consentValidityTime != null)) { - // update receipt - consentCoreDAO.updateConsentReceipt(connection, consentID, consentReceipt); - // update validity period - consentCoreDAO.updateConsentValidityTime(connection, consentID, consentValidityTime); - } else { - if (StringUtils.isBlank(consentReceipt) && (consentValidityTime != null)) { - // update receipt - consentCoreDAO.updateConsentValidityTime(connection, consentID, consentValidityTime); - } else { - // update receipt - consentCoreDAO.updateConsentReceipt(connection, consentID, consentReceipt); - } - } - - // Get consent after update - ConsentResource consentResource = consentCoreDAO.getConsentResource(connection, consentID); - - /* Even if the consent is amended, the status remains same as Authorized. For tracking purposes, an - audit record is created as the consent status of "amended". But still the real consent status will - remain as it is */ - HashMap consentDataMap = new HashMap<>(); - consentDataMap.put(ConsentCoreServiceConstants.CONSENT_RESOURCE, consentResource); - postStateChange(connection, consentCoreDAO, consentID, userID, - ConsentCoreServiceConstants.CONSENT_AMENDED_STATUS, consentResource.getCurrentStatus(), - ConsentCoreServiceConstants.CONSENT_AMEND_REASON, consentResource.getClientID(), - consentDataMap); - - // Commit transaction - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return consentResource; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } catch (OBConsentDataInsertionException e) { - log.error(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - } catch (OBConsentDataUpdationException e) { - log.error(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - public DetailedConsentResource amendDetailedConsent(String consentID, String consentReceipt, - Long consentValidityTime, String authID, - Map> accountIDsMapWithPermissions, - String newConsentStatus, Map consentAttributes, String userID, - Map additionalAmendmentData) - throws ConsentManagementException { - - if (StringUtils.isBlank(consentID) || - (StringUtils.isBlank(consentReceipt) && (consentValidityTime == null))) { - log.error("Consent ID or both consent receipt and consent validity period are not provided, cannot " + - "proceed"); - throw new ConsentManagementException("Consent ID or both consent receipt and consent validity period are " + - "not provided, cannot proceed"); - } - - if (StringUtils.isBlank(authID) || StringUtils.isBlank(userID) - || MapUtils.isEmpty(accountIDsMapWithPermissions) || StringUtils.isBlank(newConsentStatus) - || consentAttributes == null) { - log.error("Auth ID, user ID, account permissions map, new consent status or new consent attributes " + - "is not present, cannot proceed"); - throw new ConsentManagementException("Auth ID, user ID, account permissions map, new consent status or " + - "new consent attributes is not present, cannot proceed"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - // Retrieve the current detailed consent before the amendment for the consent amendment history persistence - DetailedConsentResource detailedConsentResource = - consentCoreDAO.getDetailedConsentResource(connection, consentID, false); - - // Update receipt and validity time - if (StringUtils.isNotBlank(consentReceipt)) { - consentCoreDAO.updateConsentReceipt(connection, consentID, consentReceipt); - } - if (consentValidityTime != null) { - consentCoreDAO.updateConsentValidityTime(connection, consentID, consentValidityTime); - } - - // Update consent status and record the updated time - consentCoreDAO.updateConsentStatus(connection, consentID, newConsentStatus); - - // Update accounts if required - updateAccounts(connection, consentCoreDAO, authID, accountIDsMapWithPermissions, - detailedConsentResource, false); - - // Update consent attributes - updateConsentAttributes(connection, consentCoreDAO, consentID, consentAttributes); - - // Update consent accordingly if additional amendment data passed - if (!additionalAmendmentData.isEmpty()) { - processAdditionalConsentAmendmentData(connection, consentCoreDAO, additionalAmendmentData); - } - - // Get detailed consent status after update - DetailedConsentResource newDetailedConsentResource = - consentCoreDAO.getDetailedConsentResource(connection, consentID, false); - - /* Even if the consent is amended, the status remains same as Authorized. For tracking purposes, an - audit record is created as the consent status of "amended". But still the real consent status will - remain as it is */ - HashMap consentDataMap = new HashMap<>(); - consentDataMap.put(ConsentCoreServiceConstants.DETAILED_CONSENT_RESOURCE, newDetailedConsentResource); - - // Pass the previous consent to persist as consent amendment history - consentDataMap.put(ConsentCoreServiceConstants.CONSENT_AMENDMENT_HISTORY_RESOURCE, detailedConsentResource); - consentDataMap.put(ConsentCoreServiceConstants.CONSENT_AMENDMENT_TIME, System.currentTimeMillis()); - - postStateChange(connection, consentCoreDAO, consentID, userID, - ConsentCoreServiceConstants.CONSENT_AMENDED_STATUS, detailedConsentResource.getCurrentStatus(), - ConsentCoreServiceConstants.CONSENT_AMEND_REASON, detailedConsentResource.getClientID(), - consentDataMap); - - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return newDetailedConsentResource; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } catch (OBConsentDataInsertionException e) { - log.error(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - } catch (OBConsentDataUpdationException e) { - log.error(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - } catch (OBConsentDataDeletionException e) { - log.error(ConsentCoreServiceConstants.DATA_DELETE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.CONSENT_ATTRIBUTES_DELETE_ERROR_MSG); - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - private void updateConsentAttributes(Connection connection, ConsentCoreDAO consentCoreDAO, - String consentID, Map consentAttributes) - throws OBConsentDataDeletionException, OBConsentDataInsertionException { - - // delete existing consent attributes - if (log.isDebugEnabled()) { - log.debug("Deleting attributes for the consent ID: " + consentID.replaceAll("[\r\n]", "")); - } - consentCoreDAO.deleteConsentAttributes(connection, consentID, - new ArrayList<>(consentAttributes.keySet())); - - // store new set of consent attributes - ConsentAttributes consentAttributesObject = new ConsentAttributes(); - consentAttributesObject.setConsentID(consentID); - consentAttributesObject.setConsentAttributes(consentAttributes); - if (log.isDebugEnabled()) { - log.debug("Storing consent attributes for the consent of ID: " + consentID.replaceAll("[\r\n]", "")); - } - consentCoreDAO.storeConsentAttributes(connection, consentAttributesObject); - } - - private void processAdditionalConsentAmendmentData(Connection connection, ConsentCoreDAO consentCoreDAO, - Map additionalAmendmentData) - throws ConsentManagementException, OBConsentDataInsertionException { - - Map newAuthResources; - Map> newMappingResources; - - if (additionalAmendmentData.containsKey(ConsentCoreServiceConstants.ADDITIONAL_AUTHORIZATION_RESOURCES) && - additionalAmendmentData.containsKey(ConsentCoreServiceConstants.ADDITIONAL_MAPPING_RESOURCES)) { - - newAuthResources = (Map) additionalAmendmentData - .get(ConsentCoreServiceConstants.ADDITIONAL_AUTHORIZATION_RESOURCES); - newMappingResources = (Map>) additionalAmendmentData - .get(ConsentCoreServiceConstants.ADDITIONAL_MAPPING_RESOURCES); - - for (Map.Entry authResourceEntry : newAuthResources.entrySet()) { - String userId = authResourceEntry.getKey(); - AuthorizationResource authResource = authResourceEntry.getValue(); - - if (StringUtils.isBlank(authResource.getConsentID()) || - StringUtils.isBlank(authResource.getAuthorizationType()) || - StringUtils.isBlank(authResource.getAuthorizationStatus())) { - log.error("Consent ID, authorization type or authorization status is missing, cannot proceed"); - throw new ConsentManagementException("Cannot proceed since consent ID, authorization type or " + - "authorization status is missing"); - } - // create authorization resource - AuthorizationResource authorizationResource = - consentCoreDAO.storeAuthorizationResource(connection, authResource); - ArrayList mappingResources = newMappingResources.get(userId); - - for (ConsentMappingResource mappingResource : mappingResources) { - - if (StringUtils.isBlank(mappingResource.getAccountID()) || - StringUtils.isBlank(mappingResource.getMappingStatus())) { - log.error("Account ID or Mapping Status is not found, cannot proceed"); - throw new ConsentManagementException("Account ID or Mapping Status is not found, " + - "cannot proceed"); - } - mappingResource.setAuthorizationID(authorizationResource.getAuthorizationID()); - // create mapping resource - consentCoreDAO.storeConsentMappingResource(connection, mappingResource); - } - } - } - } - - @Override - public boolean storeConsentAmendmentHistory(String consentID, ConsentHistoryResource consentHistoryResource, - DetailedConsentResource detailedCurrentConsent) throws ConsentManagementException { - - if (StringUtils.isBlank(consentID) || consentHistoryResource == null || - StringUtils.isBlank(consentHistoryResource.getReason()) || consentHistoryResource.getTimestamp() == 0) { - log.error("Consent ID or detailed consent resource or amendment reason or amended timestamp " + - "in consent history resource is empty/zero"); - throw new ConsentManagementException("Consent ID or detailed consent resource or amendment reason or " + - "amended timestamp in consent resource history is empty/zero, cannot proceed"); - } - - String historyID = consentHistoryResource.getHistoryID(); - if (StringUtils.isBlank(historyID)) { - historyID = String.valueOf(UUID.randomUUID()); - } - long amendedTimestamp = consentHistoryResource.getTimestamp(); - String amendmentReason = consentHistoryResource.getReason(); - - Connection connection = DatabaseUtil.getDBConnection(); - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - if (detailedCurrentConsent == null) { - detailedCurrentConsent = consentCoreDAO.getDetailedConsentResource(connection, consentID, false); - } - - DetailedConsentResource detailedHistoryConsent = consentHistoryResource.getDetailedConsentResource(); - // store only the changes in basic consent data to CA history - JSONObject changedConsentDataJson = getChangedBasicConsentDataJSON(detailedCurrentConsent, - detailedHistoryConsent); - if (!changedConsentDataJson.isEmpty()) { - consentCoreDAO.storeConsentAmendmentHistory(connection, historyID, amendedTimestamp, consentID, - ConsentMgtDAOConstants.TYPE_CONSENT_BASIC_DATA, String.valueOf(changedConsentDataJson), - amendmentReason); - } - - // store only the changes in consent attributes to CA history - JSONObject changedConsentAttributesJson = getChangedConsentAttributesDataJSON( - detailedCurrentConsent.getConsentAttributes(), detailedHistoryConsent.getConsentAttributes()); - if (!changedConsentAttributesJson.isEmpty()) { - consentCoreDAO.storeConsentAmendmentHistory(connection, historyID, amendedTimestamp, - consentID, ConsentMgtDAOConstants.TYPE_CONSENT_ATTRIBUTES_DATA, - String.valueOf(changedConsentAttributesJson), amendmentReason); - } - - // store only the changes in consent mappings to CA history - Map changedConsentMappingsJsonDataMap = - getChangedConsentMappingDataJSONMap(detailedCurrentConsent.getConsentMappingResources(), - detailedHistoryConsent.getConsentMappingResources()); - for (Map.Entry changedConsentMapping : changedConsentMappingsJsonDataMap.entrySet()) { - consentCoreDAO.storeConsentAmendmentHistory(connection, historyID, amendedTimestamp, - changedConsentMapping.getKey(), ConsentMgtDAOConstants.TYPE_CONSENT_MAPPING_DATA, - String.valueOf(changedConsentMapping.getValue()), amendmentReason); - } - - // store only the changes in consent Auth Resources to CA history - Map changedConsentAuthResourcesJsonDataMap = - getChangedConsentAuthResourcesDataJSONMap(detailedCurrentConsent.getAuthorizationResources(), - detailedHistoryConsent.getAuthorizationResources()); - for (Map.Entry changedConsentAuthResource : - changedConsentAuthResourcesJsonDataMap.entrySet()) { - consentCoreDAO.storeConsentAmendmentHistory(connection, historyID, amendedTimestamp, - changedConsentAuthResource.getKey(), ConsentMgtDAOConstants.TYPE_CONSENT_AUTH_RESOURCE_DATA, - String.valueOf(changedConsentAuthResource.getValue()), amendmentReason); - } - - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return true; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } catch (OBConsentDataInsertionException e) { - log.error(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - private JSONObject getChangedBasicConsentDataJSON(DetailedConsentResource newConsentResource, - DetailedConsentResource oldConsentResource) { - - JSONObject changedConsentDataJson = new JSONObject(); - if (!newConsentResource.getReceipt().equalsIgnoreCase(oldConsentResource.getReceipt())) { - changedConsentDataJson.put(ConsentMgtDAOConstants.RECEIPT, oldConsentResource.getReceipt()); - } - if (newConsentResource.getValidityPeriod() != oldConsentResource.getValidityPeriod()) { - changedConsentDataJson.put(ConsentMgtDAOConstants.VALIDITY_TIME, - String.valueOf(oldConsentResource.getValidityPeriod())); - } - if (newConsentResource.getUpdatedTime() != oldConsentResource.getUpdatedTime()) { - changedConsentDataJson.put(ConsentMgtDAOConstants.UPDATED_TIME, - String.valueOf(oldConsentResource.getUpdatedTime())); - } - if (!newConsentResource.getCurrentStatus().equalsIgnoreCase(oldConsentResource.getCurrentStatus())) { - changedConsentDataJson.put(ConsentMgtDAOConstants.CURRENT_STATUS, - String.valueOf(oldConsentResource.getCurrentStatus())); - } - return changedConsentDataJson; - } - - private JSONObject getChangedConsentAttributesDataJSON(Map newConsentAttributes, - Map oldConsentAttributes) { - - JSONObject changedConsentAttributesJson = new JSONObject(); - for (Map.Entry consentAttribute : oldConsentAttributes.entrySet()) { - String attributeKey = consentAttribute.getKey(); - if (!newConsentAttributes.containsKey(attributeKey) - || !newConsentAttributes.get(attributeKey).equalsIgnoreCase(consentAttribute.getValue())) { - //store only the consent attributes with a changed value to the consent amendment history - changedConsentAttributesJson.put(attributeKey, consentAttribute.getValue()); - } - } - for (Map.Entry newConsentAttribute : newConsentAttributes.entrySet()) { - String attributeKey = newConsentAttribute.getKey(); - if (!oldConsentAttributes.containsKey(newConsentAttribute.getKey())) { - //store any new consent attribute in current consent to the changedConsentAttributesJson of - //the immediate past consent amendment history with a null value - changedConsentAttributesJson.put(attributeKey, null); - } - } - return changedConsentAttributesJson; - } - - private Map getChangedConsentMappingDataJSONMap(ArrayList - newConsentMappings, ArrayList oldConsentMappings) { - - Map changedConsentMappingsJsonDataMap = new HashMap<>(); - ArrayList existingConsentMappingIds = new ArrayList<>(); - for (ConsentMappingResource newMapping : newConsentMappings) { - JSONObject changedConsentMappingJson = new JSONObject(); - for (ConsentMappingResource oldMapping : oldConsentMappings) { - if (newMapping.getMappingID().equalsIgnoreCase(oldMapping.getMappingID())) { - existingConsentMappingIds.add(newMapping.getMappingID()); - if (!newMapping.getMappingStatus().equalsIgnoreCase(oldMapping.getMappingStatus())) { - //store only the mapping-ids with a changed Mapping Status to the consent amendment history - changedConsentMappingJson.put(ConsentMgtDAOConstants.MAPPING_STATUS, - oldMapping.getMappingStatus()); - } - break; - } - } - if (!changedConsentMappingJson.isEmpty()) { - changedConsentMappingsJsonDataMap.put(newMapping.getMappingID(), changedConsentMappingJson); - } - // store any new mapping-ids in current consent to the immediate past consent amendment history with - // 'null' value - if (!existingConsentMappingIds.contains(newMapping.getMappingID())) { - changedConsentMappingsJsonDataMap.put(newMapping.getMappingID(), null); - } - } - return changedConsentMappingsJsonDataMap; - } - - private Map getChangedConsentAuthResourcesDataJSONMap(ArrayList - newConsentAuthResources, ArrayList oldConsentAuthResources) { - - Map changedConsentAuthResourcesJsonDataMap = new HashMap<>(); - - ArrayList existingConsentAuthResourceIds = new ArrayList<>(); - for (AuthorizationResource newAuthResource : newConsentAuthResources) { - for (AuthorizationResource oldAuthResource : oldConsentAuthResources) { - if (newAuthResource.getAuthorizationID().equalsIgnoreCase(oldAuthResource.getAuthorizationID())) { - existingConsentAuthResourceIds.add(newAuthResource.getAuthorizationID()); - break; - } - } - // store any new authorization-ids in current consent (an Auth Resource not available in previous consent, - // but newly added in current consent) to the immediate past consent amendment history with 'null' value - if (!existingConsentAuthResourceIds.contains(newAuthResource.getAuthorizationID())) { - changedConsentAuthResourcesJsonDataMap.put(newAuthResource.getAuthorizationID(), null); - } - } - return changedConsentAuthResourcesJsonDataMap; - } - - @Override - public Map getConsentAmendmentHistoryData(String consentID) - throws ConsentManagementException { - - if (StringUtils.isBlank(consentID)) { - log.error("Consent ID is empty"); - throw new ConsentManagementException("Consent ID is empty, cannot proceed"); - } - - Connection connection = DatabaseUtil.getDBConnection(); - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - //Retrieve the current detailed consent to build the detailed consent amendment history resources - DetailedConsentResource currentConsentResource = - consentCoreDAO.getDetailedConsentResource(connection, consentID, false); - - Map consentAmendmentHistoryRetrievalResult = - consentCoreDAO.retrieveConsentAmendmentHistory(connection, - getRecordIdListForConsentHistoryRetrieval(currentConsentResource)); - - Map consentAmendmentHistory = new LinkedHashMap<>(); - if (!consentAmendmentHistoryRetrievalResult.isEmpty()) { - consentAmendmentHistory = processConsentAmendmentHistoryData( - consentAmendmentHistoryRetrievalResult, currentConsentResource); - } - DatabaseUtil.commitTransaction(connection); - return consentAmendmentHistory; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public boolean syncRetentionDatabaseWithPurgedConsent() throws ConsentManagementException { - - if (!OpenBankingConfigParser.getInstance().isConsentDataRetentionEnabled()) { - log.error("Consent data retention is not enabled, Hence data sync is not possible at the moment"); - throw new ConsentManagementException("Consent data retention is not enabled, " + - "Hence data sync is not possible at the moment"); - } - - Connection consentDBConnection = DatabaseUtil.getDBConnection(); - Connection retentionDBConnection = DatabaseUtil.getRetentionDBConnection(); - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - ConsentCoreDAO consentRetentionDAO = ConsentStoreInitializer.getInitializedConsentRetentionDAOImpl(); - - // Fetch list of consent_id's to sync from temporary retention tables in consent DB. - ArrayList listOfConsentIds = consentCoreDAO.getListOfConsentIds(consentDBConnection, true); - - // Fetch consent data from temporary retention tables in consent DB. - log.debug("Fetching consent data from temporary retention tables in consent DB"); - for (String consentId : listOfConsentIds) { - Savepoint retentionDBSavepoint = retentionDBConnection.setSavepoint(); - Savepoint consentDBSavepoint = consentDBConnection.setSavepoint(); - try { - // Fetching detailed consent. - DetailedConsentResource detailedConsent = - consentCoreDAO.getDetailedConsentResource(consentDBConnection, consentId, true); - ConsentResource consentResource = new ConsentResource(detailedConsent.getConsentID(), - detailedConsent.getClientID(), detailedConsent.getReceipt(), - detailedConsent.getConsentType(), detailedConsent.getConsentFrequency(), - detailedConsent.getValidityPeriod(), detailedConsent.isRecurringIndicator(), - detailedConsent.getCurrentStatus(), detailedConsent.getCreatedTime(), - detailedConsent.getUpdatedTime()); - - ConsentFile consentFile = null; - ArrayList consentStatusAuditRecords = null; - try { - // Fetching consent file. - consentFile = consentCoreDAO.getConsentFile(consentDBConnection, consentId, true); - } catch (OBConsentDataRetrievalException e) { - log.error(String.format("Error occurred fetching consent file for consent_id : %s , " + - "Ignoring this as null consent file for given consent_id", - consentId.replaceAll("[\r\n]", ""))); - } - try { - // Fetching consent audit records. - ArrayList consentIds = new ArrayList<>(); - consentIds.add(consentId); - consentStatusAuditRecords = - consentCoreDAO.getConsentStatusAuditRecordsByConsentId(consentDBConnection, consentIds, - null, null, true); - } catch (OBConsentDataRetrievalException e) { - log.error(String.format("Error occurred fetching consent audit records for consent_id : %s , " + - "Ignoring this as null consent audit records for given consent_id", - consentId.replaceAll("[\r\n]", ""))); - } - - // Inserting to retention datasource - ConsentResource insertedConsentResources = - consentRetentionDAO.storeConsentResource(retentionDBConnection, consentResource); - if (insertedConsentResources == null) { - throw new OBConsentDataInsertionException(ConsentCoreServiceConstants. - DATA_INSERTION_ROLLBACK_ERROR_MSG + " for consent resource"); - } - for (AuthorizationResource authResource : detailedConsent.getAuthorizationResources()) { - if (authResource.getAuthorizationID() != null) { - AuthorizationResource storeAuthorizationResource = - consentRetentionDAO.storeAuthorizationResource(retentionDBConnection, authResource); - if (storeAuthorizationResource == null) { - throw new OBConsentDataInsertionException(ConsentCoreServiceConstants. - DATA_INSERTION_ROLLBACK_ERROR_MSG + " for authorization resources"); - } - } - } - for (ConsentMappingResource mappingResource : detailedConsent.getConsentMappingResources()) { - ConsentMappingResource storeConsentMappingResource = - consentRetentionDAO.storeConsentMappingResource(retentionDBConnection, mappingResource); - if (storeConsentMappingResource == null) { - throw new OBConsentDataInsertionException(ConsentCoreServiceConstants. - DATA_INSERTION_ROLLBACK_ERROR_MSG + " for mapping resources"); - } - } - if (!detailedConsent.getConsentAttributes().isEmpty()) { - ConsentAttributes consentAttributes = new ConsentAttributes(consentId, - detailedConsent.getConsentAttributes()); - if (!consentRetentionDAO.storeConsentAttributes(retentionDBConnection, consentAttributes)) { - throw new OBConsentDataInsertionException(ConsentCoreServiceConstants. - DATA_INSERTION_ROLLBACK_ERROR_MSG + " for consent attributes"); - } - } - if (consentFile != null) { - if (!consentRetentionDAO.storeConsentFile(retentionDBConnection, consentFile)) { - throw new OBConsentDataInsertionException(ConsentCoreServiceConstants. - DATA_INSERTION_ROLLBACK_ERROR_MSG + " for consent file"); - } - } - if (consentStatusAuditRecords != null) { - for (ConsentStatusAuditRecord auditRecords : consentStatusAuditRecords) { - ConsentStatusAuditRecord storeConsentStatusAuditRecord = consentRetentionDAO - .storeConsentStatusAuditRecord(retentionDBConnection, auditRecords); - if (storeConsentStatusAuditRecord == null) { - throw new OBConsentDataInsertionException(ConsentCoreServiceConstants. - DATA_INSERTION_ROLLBACK_ERROR_MSG + " for consent audit records"); - } - } - } - - // Removing consent data from temporary retention table in consent database - boolean consentDeleted = consentCoreDAO.deleteConsentData(consentDBConnection, consentId, true); - if (!consentDeleted) { - throw new OBConsentDataDeletionException(ConsentCoreServiceConstants. - DATA_DELETE_ROLLBACK_ERROR_MSG + " for consent data deletion"); - } - // Commit transactions - DatabaseUtil.commitTransaction(retentionDBConnection); - DatabaseUtil.commitTransaction(consentDBConnection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - } catch (OBConsentDataRetrievalException | OBConsentDataInsertionException | - OBConsentDataDeletionException e) { - log.error(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - consentDBConnection.rollback(consentDBSavepoint); - retentionDBConnection.rollback(retentionDBSavepoint); - } - } - return true; - } catch (OBConsentDataRetrievalException | SQLException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException("Error occurred while syncing the retention data in consent " + - "database to retention database", e); - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(consentDBConnection); - DatabaseUtil.closeConnection(retentionDBConnection); - } - } - - @Override - public ArrayList getConsentStatusAuditRecords(ArrayList consentIDs, - Integer limit, Integer offset, - boolean fetchFromRetentionDatabase) - throws ConsentManagementException { - - if (!OpenBankingConfigParser.getInstance().isConsentDataRetentionEnabled() && fetchFromRetentionDatabase) { - log.error("Consent data retention is not enabled."); - throw new ConsentManagementException("Consent data retention is not enabled."); - } - - Connection connection; - if (fetchFromRetentionDatabase) { - connection = DatabaseUtil.getRetentionDBConnection(); - } else { - connection = DatabaseUtil.getDBConnection(); - } - - ConsentCoreDAO consentCoreDAO; - if (fetchFromRetentionDatabase) { - consentCoreDAO = ConsentStoreInitializer.getInitializedConsentRetentionDAOImpl(); - log.debug("Fetching consent status audit records from retention datasource"); - } else { - consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - } - - try { - //Retrieve consent status audit records. - return consentCoreDAO.getConsentStatusAuditRecordsByConsentId(connection, consentIDs, limit, offset, - false); - - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public ConsentFile getConsentFile(String consentId, boolean fetchFromRetentionDatabase) - throws ConsentManagementException { - - if (StringUtils.isBlank(consentId)) { - log.error("Consent ID is empty"); - throw new ConsentManagementException("Consent ID is empty, cannot proceed"); - } - - if (!OpenBankingConfigParser.getInstance().isConsentDataRetentionEnabled() && fetchFromRetentionDatabase) { - log.error("Consent data retention is not enabled."); - throw new ConsentManagementException("Consent data retention is not enabled."); - } - - Connection connection; - if (fetchFromRetentionDatabase) { - connection = DatabaseUtil.getRetentionDBConnection(); - } else { - connection = DatabaseUtil.getDBConnection(); - } - - ConsentCoreDAO consentCoreDAO; - if (fetchFromRetentionDatabase) { - consentCoreDAO = ConsentStoreInitializer.getInitializedConsentRetentionDAOImpl(); - log.debug("Fetching consent file from retention datasource"); - } else { - consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - } - - try { - //Retrieve consent status audit records. - return consentCoreDAO.getConsentFile(connection, consentId, false); - - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - private List getRecordIdListForConsentHistoryRetrieval(DetailedConsentResource detailedConsentResource) { - - List recordIdsList = new ArrayList<>(); - recordIdsList.add(detailedConsentResource.getConsentID()); - - for (ConsentMappingResource mappingResource : detailedConsentResource.getConsentMappingResources()) { - recordIdsList.add(mappingResource.getMappingID()); - } - for (AuthorizationResource authResource : detailedConsentResource.getAuthorizationResources()) { - recordIdsList.add(authResource.getAuthorizationID()); - } - return recordIdsList; - } - - private Map processConsentAmendmentHistoryData( - Map consentAmendmentHistoryRetrievalResult, - DetailedConsentResource currentConsentResource) throws ConsentManagementException { - - Gson gson = new Gson(); - Map consentAmendmentHistoryDataMap = new LinkedHashMap<>(); - DetailedConsentResource detailedConsentHistoryResource = gson.fromJson(gson.toJson(currentConsentResource), - DetailedConsentResource.class); - - for (Map.Entry consentHistoryDataEntry : - consentAmendmentHistoryRetrievalResult.entrySet()) { - String historyId = consentHistoryDataEntry.getKey(); - ConsentHistoryResource consentHistoryResource = consentHistoryDataEntry.getValue(); - - for (Map.Entry consentHistoryDataTypeEntry : - consentHistoryResource.getChangedAttributesJsonDataMap().entrySet()) { - String consentDataType = consentHistoryDataTypeEntry.getKey(); - Object changedAttributes = consentHistoryDataTypeEntry.getValue(); - - if (ConsentMgtDAOConstants.TYPE_CONSENT_BASIC_DATA.equalsIgnoreCase(consentDataType)) { - JSONObject changedValuesJSON = parseChangedAttributeJsonString(changedAttributes.toString()); - if (changedValuesJSON.containsKey(ConsentMgtDAOConstants.RECEIPT)) { - detailedConsentHistoryResource.setReceipt( - (String) changedValuesJSON.get(ConsentMgtDAOConstants.RECEIPT)); - } - if (changedValuesJSON.containsKey(ConsentMgtDAOConstants.VALIDITY_TIME)) { - detailedConsentHistoryResource.setValidityPeriod(Long.parseLong((String) - changedValuesJSON.get(ConsentMgtDAOConstants.VALIDITY_TIME))); - } - if (changedValuesJSON.containsKey(ConsentMgtDAOConstants.UPDATED_TIME)) { - detailedConsentHistoryResource.setUpdatedTime(Long.parseLong((String) - changedValuesJSON.get(ConsentMgtDAOConstants.UPDATED_TIME))); - } - if (changedValuesJSON.containsKey(ConsentMgtDAOConstants.CURRENT_STATUS)) { - detailedConsentHistoryResource.setCurrentStatus((String) - changedValuesJSON.get(ConsentMgtDAOConstants.CURRENT_STATUS)); - } - - } else if (ConsentMgtDAOConstants.TYPE_CONSENT_ATTRIBUTES_DATA.equalsIgnoreCase(consentDataType)) { - JSONObject changedValuesJSON = parseChangedAttributeJsonString(changedAttributes.toString()); - for (Map.Entry attribute : changedValuesJSON.entrySet()) { - Object attributeValue = attribute.getValue(); - if (attributeValue == null) { - //Ignore the consent attribute from the consent history if it's value is stored as null - detailedConsentHistoryResource.getConsentAttributes().remove(attribute.getKey()); - } else { - detailedConsentHistoryResource.getConsentAttributes().put(attribute.getKey(), - attributeValue.toString()); - } - } - - } else if (ConsentMgtDAOConstants.TYPE_CONSENT_MAPPING_DATA.equalsIgnoreCase(consentDataType)) { - Map changedConsentMappingsDataMap = (Map) changedAttributes; - ArrayList consentMappings = - detailedConsentHistoryResource.getConsentMappingResources(); - ArrayList consentMappingsHistory = new ArrayList<>(); - for (ConsentMappingResource mapping : consentMappings) { - String mappingID = mapping.getMappingID(); - if (changedConsentMappingsDataMap.containsKey(mappingID)) { - JSONObject changedValuesJSON = parseChangedAttributeJsonString( - changedConsentMappingsDataMap.get(mappingID).toString()); - if (changedValuesJSON.isEmpty()) { - //Skip setting the mapping to consent history if the value is null - continue; - } - //set the value available in the history as the mapping status - mapping.setMappingStatus( - changedValuesJSON.get(ConsentMgtDAOConstants.MAPPING_STATUS).toString()); - } - consentMappingsHistory.add(gson.fromJson(gson.toJson(mapping), ConsentMappingResource.class)); - } - detailedConsentHistoryResource.setConsentMappingResources(consentMappingsHistory); - - } else if (ConsentMgtDAOConstants.TYPE_CONSENT_AUTH_RESOURCE_DATA.equalsIgnoreCase(consentDataType)) { - Map changedConsentAuthResourceDataMap = (Map) changedAttributes; - ArrayList consentAuthResources = detailedConsentHistoryResource - .getAuthorizationResources(); - ArrayList consentAuthResourceHistory = new ArrayList<>(); - for (AuthorizationResource authResource : consentAuthResources) { - String authID = authResource.getAuthorizationID(); - if (changedConsentAuthResourceDataMap.containsKey(authID)) { - JSONObject changedValuesJSON = parseChangedAttributeJsonString( - changedConsentAuthResourceDataMap.get(authID).toString()); - if (changedValuesJSON.isEmpty()) { - //Skip setting the auth resource to consent history if the value is null - continue; - } - } - consentAuthResourceHistory.add(gson.fromJson(gson.toJson(authResource), - AuthorizationResource.class)); - } - detailedConsentHistoryResource.setAuthorizationResources(consentAuthResourceHistory); - } - } - consentHistoryResource.setDetailedConsentResource(gson.fromJson(gson.toJson(detailedConsentHistoryResource), - DetailedConsentResource.class)); - consentAmendmentHistoryDataMap.put(historyId, consentHistoryResource); - } - return consentAmendmentHistoryDataMap; - } - - private JSONObject parseChangedAttributeJsonString(String changedAttributes) throws ConsentManagementException { - - Object changedValues; - try { - changedValues = new JSONParser(JSONParser.MODE_PERMISSIVE).parse(changedAttributes); - } catch (ParseException e) { - throw new ConsentManagementException("Changed Values is not a valid JSON object", e); - } - if (changedValues == null) { - return new JSONObject(); - } - return (JSONObject) changedValues; - - } - - public void revokeTokens(DetailedConsentResource detailedConsentResource, String userID) - throws IdentityOAuth2Exception { - - OAuth2Service oAuth2Service = getOAuth2Service(); - String clientId = detailedConsentResource.getClientID(); - String consentId = detailedConsentResource.getConsentID(); - AuthenticatedUser authenticatedUser = getAuthenticatedUser(userID); - Set accessTokenDOSet = getAccessTokenDOSet(detailedConsentResource, authenticatedUser); - - String consentIdClaim = OpenBankingConfigParser.getInstance().getConfiguration() - .get(OpenBankingConstants.CONSENT_ID_CLAIM_NAME).toString(); - - if (!accessTokenDOSet.isEmpty()) { - Set activeTokens = new HashSet<>(); - // Get tokens to revoke to an array - for (AccessTokenDO accessTokenDO : accessTokenDOSet) { - // Filter tokens by consent ID claim - if (Arrays.asList(accessTokenDO.getScope()).contains(consentIdClaim + - detailedConsentResource.getConsentID())) { - activeTokens.add(accessTokenDO.getAccessToken()); - } - } - - if (!activeTokens.isEmpty()) { - // set authorization context details for the given user - OAuthClientAuthnContext oAuthClientAuthnContext = new OAuthClientAuthnContext(); - oAuthClientAuthnContext.setAuthenticated(true); - oAuthClientAuthnContext.setClientId(clientId); - oAuthClientAuthnContext.addParameter(OpenBankingConstants.IS_CONSENT_REVOCATION_FLOW, true); - - // set common properties of token revocation request - OAuthRevocationRequestDTO revokeRequestDTO = new OAuthRevocationRequestDTO(); - revokeRequestDTO.setOauthClientAuthnContext(oAuthClientAuthnContext); - revokeRequestDTO.setConsumerKey(clientId); - revokeRequestDTO.setTokenType(GrantType.REFRESH_TOKEN.toString()); - - for (String activeToken : activeTokens) { - // set access token to be revoked - revokeRequestDTO.setToken(activeToken); - OAuthRevocationResponseDTO oAuthRevocationResponseDTO = - revokeTokenByClient(oAuth2Service, revokeRequestDTO); - - if (oAuthRevocationResponseDTO.isError()) { - log.error("Error while revoking access token for consent ID: " - + consentId.replaceAll("[\r\n]", "")); - throw new IdentityOAuth2Exception( - String.format("Error while revoking access token for consent ID: %s. Caused by, %s", - consentId, oAuthRevocationResponseDTO.getErrorMsg())); - } - } - } - } - } - - private boolean isValidUserID(String requestUserID, Set consentUserIDSet) { - if (StringUtils.isEmpty(requestUserID)) { - // userId not present in request query parameters, can use consentUserID to revoke tokens - return true; - } - return consentUserIDSet.contains(requestUserID); - } - - @Generated(message = "Excluded from code coverage since used for testing purposes") - OAuth2Service getOAuth2Service() { - - return ConsentManagementDataHolder.getInstance().getOAuth2Service(); - } - - @Generated(message = "Excluded from code coverage since used for testing purposes") - AuthenticatedUser getAuthenticatedUser(String userID) throws IdentityOAuth2Exception { - // set domain name - if (UserCoreUtil.getDomainFromThreadLocal() == null) { - UserCoreUtil.setDomainInThreadLocal(UserCoreUtil.extractDomainFromName(userID)); - } - if (OpenBankingConfigParser.getInstance().isPSUFederated()) { - AuthenticatedUser authenticatedUser = - AuthenticatedUser.createFederateAuthenticatedUserFromSubjectIdentifier(userID); - authenticatedUser.setUserStoreDomain(OAuth2Util.getUserStoreForFederatedUser(authenticatedUser)); - authenticatedUser.setTenantDomain(MultitenantUtils.getTenantDomain(userID)); - authenticatedUser.setFederatedIdPName(OpenBankingConfigParser.getInstance().getFederatedIDPName()); - authenticatedUser.setUserName(MultitenantUtils.getTenantAwareUsername(userID)); - return authenticatedUser; - } else { - return AuthenticatedUser.createLocalAuthenticatedUserFromSubjectIdentifier(userID); - } - } - - @Generated(message = "Excluded from code coverage since used for testing purposes") - Set getAccessTokenDOSet(DetailedConsentResource detailedConsentResource, - AuthenticatedUser authenticatedUser) throws IdentityOAuth2Exception { - - return OAuthTokenPersistenceFactory.getInstance().getAccessTokenDAO() - .getAccessTokens(detailedConsentResource.getClientID(), authenticatedUser, - authenticatedUser.getUserStoreDomain(), false); - } - - @Generated(message = "Excluded from code coverage since used for testing purposes") - OAuthRevocationResponseDTO revokeTokenByClient(OAuth2Service oAuth2Service, - OAuthRevocationRequestDTO revocationRequestDTO) { - - return oAuth2Service.revokeTokenByOAuthClient(revocationRequestDTO); - } - - @Override - public ConsentResource updateConsentStatus(String consentId, String newConsentStatus) - throws ConsentManagementException { - - if (StringUtils.isBlank(consentId) || StringUtils.isBlank(newConsentStatus)) { - - log.error("Consent ID, userID or newConsentStatus is missing. Cannot proceed."); - throw new ConsentManagementException("Cannot proceed without Consent ID, userID or newConsentStatus."); - } - - Connection connection = DatabaseUtil.getDBConnection(); - ConsentResource updatedConsentResource; - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - // Get the existing consent to validate status - if (log.isDebugEnabled()) { - log.debug("Retrieving the consent for ID:" + consentId.replaceAll("[\r\n]", "") - + " to validate status"); - } - - // Update consent status with new status - if (log.isDebugEnabled()) { - log.debug("Updating the status of the consent for ID:" + consentId.replaceAll("[\r\n]", "")); - } - - DetailedConsentResource existingConsentResource = consentCoreDAO - .getDetailedConsentResource(connection, consentId, false); - String existingConsentStatus = existingConsentResource.getCurrentStatus(); - ArrayList authResources = existingConsentResource.getAuthorizationResources(); - - updatedConsentResource = consentCoreDAO.updateConsentStatus(connection, consentId, newConsentStatus); - - // Previous consent status is not added in reason because it can be null - String auditMessage = "Consent status updated to " + newConsentStatus; - for (AuthorizationResource authResource : authResources) { - // Create an audit record execute state change listener - HashMap consentDataMap = new HashMap<>(); - consentDataMap.put(ConsentCoreServiceConstants.DETAILED_CONSENT_RESOURCE, existingConsentResource); - postStateChange(connection, consentCoreDAO, consentId, authResource.getUserID(), newConsentStatus, - existingConsentStatus, auditMessage, existingConsentResource.getClientID(), consentDataMap); - } - - // Commit transaction - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return updatedConsentResource; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } catch (OBConsentDataInsertionException e) { - log.error(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - } catch (OBConsentDataUpdationException e) { - log.error(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public ArrayList getConsentsEligibleForExpiration(String statusesEligibleForExpiration) - throws ConsentManagementException { - - Connection connection = DatabaseUtil.getDBConnection(); - ArrayList detailedConsentResources; - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - log.debug("Retrieving consents which has expiration time attribute."); - detailedConsentResources = consentCoreDAO.getExpiringConsents(connection, - statusesEligibleForExpiration); - // Commit transactions - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return detailedConsentResources; - } catch (OBConsentDataRetrievalException e) { - log.error(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - private void postStateChange(Connection connection, ConsentCoreDAO consentCoreDAO, String consentID, - String userID, String newConsentStatus, String previousConsentStatus, String reason, - String clientId, Map consentDataMap) - throws OBConsentDataInsertionException, ConsentManagementException { - - createAuditRecord(connection, consentCoreDAO, consentID, userID, newConsentStatus, previousConsentStatus, - reason); - ConsentStateChangeListenerImpl.getInstance().onStateChange(consentID, userID, newConsentStatus, - previousConsentStatus, reason, clientId, consentDataMap); - } - - public AuthorizationResource updateAuthorizationStatus(String authorizationId, String newAuthorizationStatus) - throws ConsentManagementException { - - if (StringUtils.isBlank(authorizationId) || StringUtils.isBlank(newAuthorizationStatus)) { - - log.error("Authorization ID or newAuthorizationStatus is missing. Cannot proceed."); - throw new ConsentManagementException("Cannot proceed without Authorization ID or newAuthorizationStatus" + - "."); - } - - Connection connection = DatabaseUtil.getDBConnection(); - AuthorizationResource updatedAuthorizationResource; - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - // Update authorization status with new status - if (log.isDebugEnabled()) { - log.debug("Updating the status of the authorization for ID:" + - authorizationId.replaceAll("[\r\n]", "")); - } - updatedAuthorizationResource = consentCoreDAO.updateAuthorizationStatus(connection, authorizationId, - newAuthorizationStatus); - - // Commit transaction - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return updatedAuthorizationResource; - } catch (OBConsentDataUpdationException e) { - log.error(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - public void updateAuthorizationUser(String authorizationID, String userID) - throws ConsentManagementException { - - if (StringUtils.isBlank(authorizationID) || StringUtils.isBlank(userID)) { - - log.error("Authorization ID or user ID is missing. Cannot proceed."); - throw new ConsentManagementException("Cannot proceed without Authorization ID or UserID."); - } - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - ConsentCoreDAO consentCoreDAO = ConsentStoreInitializer.getInitializedConsentCoreDAOImpl(); - try { - // Updating the authorized user - if (log.isDebugEnabled()) { - log.debug("Updating the status of the user for authorization ID:" + - authorizationID.replaceAll("[\r\n]", "")); - } - consentCoreDAO.updateAuthorizationUser(connection, authorizationID, - userID); - - // Commit transaction - DatabaseUtil.commitTransaction(connection); - log.debug(ConsentCoreServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return; - } catch (OBConsentDataUpdationException e) { - log.error(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new ConsentManagementException(ConsentCoreServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - } - } finally { - log.debug(ConsentCoreServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/impl/ConsentStateChangeListenerImpl.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/impl/ConsentStateChangeListenerImpl.java deleted file mode 100644 index fa9ce72f..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/impl/ConsentStateChangeListenerImpl.java +++ /dev/null @@ -1,71 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.service.impl; - -import com.wso2.openbanking.accelerator.common.event.executor.OBEventQueue; -import com.wso2.openbanking.accelerator.common.event.executor.model.OBEvent; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.consent.mgt.service.internal.ConsentManagementDataHolder; -import com.wso2.openbanking.accelerator.consent.mgt.service.listener.ConsentStateChangeListener; - -import java.util.HashMap; -import java.util.Map; - -/** - * Consent state change listener implementation. - */ -public class ConsentStateChangeListenerImpl implements ConsentStateChangeListener { - - private static volatile ConsentStateChangeListenerImpl instance; - - private ConsentStateChangeListenerImpl() { - - } - - public static ConsentStateChangeListenerImpl getInstance() { - - if (instance == null) { - synchronized (ConsentStateChangeListenerImpl.class) { - if (instance == null) { - instance = new ConsentStateChangeListenerImpl(); - } - } - } - return instance; - } - - @Override - public void onStateChange(String consentID, String userID, String newConsentStatus, String previousConsentStatus, - String reason, String clientId, Map consentDataMap) - throws ConsentManagementException { - - OBEventQueue obEventQueue = ConsentManagementDataHolder.getInstance().getOBEventQueue(); - - Map eventData = new HashMap<>(); - eventData.put("ConsentId", consentID); - eventData.put("UserId", userID); - eventData.put("PreviousConsentStatus", previousConsentStatus); - eventData.put("Reason", reason); - eventData.put("ClientId", clientId); - eventData.put("ConsentDataMap", consentDataMap); - - obEventQueue.put(new OBEvent(newConsentStatus, eventData)); - - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/internal/ConsentManagementDataHolder.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/internal/ConsentManagementDataHolder.java deleted file mode 100644 index a50ed80d..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/internal/ConsentManagementDataHolder.java +++ /dev/null @@ -1,74 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.service.internal; - -import com.wso2.openbanking.accelerator.common.event.executor.OBEventQueue; -import org.wso2.carbon.identity.oauth2.OAuth2Service; -import org.wso2.carbon.identity.oauth2.dao.AccessTokenDAOImpl; - -/** - * Data holder for consent management service. - */ -public class ConsentManagementDataHolder { - - private OAuth2Service oAuth2Service; - private static volatile ConsentManagementDataHolder instance; - private OBEventQueue obEventQueue; - private final AccessTokenDAOImpl accessTokenDAO; - - private ConsentManagementDataHolder() { - this.accessTokenDAO = new AccessTokenDAOImpl(); - } - - public static ConsentManagementDataHolder getInstance() { - - if (instance == null) { - synchronized (ConsentManagementDataHolder.class) { - if (instance == null) { - instance = new ConsentManagementDataHolder(); - } - } - } - return instance; - } - - public OAuth2Service getOAuth2Service() { - - return oAuth2Service; - } - - public void setOAuth2Service(OAuth2Service oAuth2Service) { - - this.oAuth2Service = oAuth2Service; - } - - public void setOBEventQueue(OBEventQueue obEventQueue) { - - this.obEventQueue = obEventQueue; - } - - public OBEventQueue getOBEventQueue() { - - return obEventQueue; - } - - public AccessTokenDAOImpl getAccessTokenDAO() { - return accessTokenDAO; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/internal/ConsentManagementServiceComponent.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/internal/ConsentManagementServiceComponent.java deleted file mode 100644 index 2a7e4cff..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/internal/ConsentManagementServiceComponent.java +++ /dev/null @@ -1,130 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.service.internal; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.event.executor.OBEventQueue; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingRuntimeException; -import com.wso2.openbanking.accelerator.common.persistence.JDBCPersistenceManager; -import com.wso2.openbanking.accelerator.consent.mgt.service.ConsentCoreService; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.osgi.service.component.ComponentContext; -import org.osgi.service.component.annotations.Activate; -import org.osgi.service.component.annotations.Component; -import org.osgi.service.component.annotations.Deactivate; -import org.osgi.service.component.annotations.Reference; -import org.osgi.service.component.annotations.ReferenceCardinality; -import org.osgi.service.component.annotations.ReferencePolicy; -import org.wso2.carbon.identity.oauth2.OAuth2Service; - -import java.sql.SQLException; - -/** - * Consent Management Core Service Component. - */ -@Component(name = "com.wso2.openbanking.accelerator.consent.mgt.service.ConsentManagementServiceComponent", - immediate = true) -public class ConsentManagementServiceComponent { - - private static Log log = LogFactory.getLog(ConsentManagementServiceComponent.class); - - @Activate - protected void activate(ComponentContext context) { - - ConsentCoreService consentCoreService = new ConsentCoreServiceImpl(); - - // Verify Open Banking consent database connection when the server starts up - try { - boolean isConnectionActive = JDBCPersistenceManager.getInstance().getDBConnection() - .isValid(OpenBankingConfigParser.getInstance().getConnectionVerificationTimeout()); - - if (!isConnectionActive) { - log.error("The connection is not active"); - throw new OpenBankingRuntimeException("The connection is not active"); - } - } catch (SQLException e) { - log.error("Database connection is not active, cannot proceed"); - throw new OpenBankingRuntimeException("Database connection is not active, cannot proceed"); - } - - // Verify Open Banking retention database connection when the server starts up - if (OpenBankingConfigParser.getInstance().isConsentDataRetentionEnabled()) { - try { - boolean isConnectionActive = JDBCPersistenceManager.getInstance().getDBConnection().isValid( - OpenBankingConfigParser.getInstance().getRetentionDataSourceConnectionVerificationTimeout()); - - if (!isConnectionActive) { - log.error("The connection is not active for retention datasource"); - throw new OpenBankingRuntimeException("The connection is not active for retention datasource"); - } - } catch (SQLException e) { - log.error("Database connection is not active for retention datasource, cannot proceed"); - throw new OpenBankingRuntimeException("Database connection is not active for retention datasource, " + - "cannot proceed"); - } - } - - context.getBundleContext().registerService(ConsentCoreService.class.getName(), consentCoreService, null); - log.debug("Consent Management Service is registered successfully."); - } - - @Deactivate - protected void deactivate(ComponentContext ctxt) { - log.debug("Consent Management Service is deactivated"); - } - - @Reference( - name = "identity.oauth.service", - service = OAuth2Service.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetOAuth2Service" - ) - protected void setOAuth2Service(OAuth2Service oAuth2Service) { - - ConsentManagementDataHolder.getInstance().setOAuth2Service(oAuth2Service); - log.debug("OAuth2Service is activated"); - } - - protected void unsetOAuth2Service(OAuth2Service oAuth2Service) { - - ConsentManagementDataHolder.getInstance().setOAuth2Service(oAuth2Service); - } - - @Reference( - service = OBEventQueue.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetOBEventQueue" - ) - - protected void setOBEventQueue(OBEventQueue obEventQueue) { - - ConsentManagementDataHolder.getInstance().setOBEventQueue(obEventQueue); - } - - protected void unsetOBEventQueue(OBEventQueue obEventQueue) { - - ConsentManagementDataHolder.getInstance().setOBEventQueue(null); - } - - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/listener/ConsentStateChangeListener.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/listener/ConsentStateChangeListener.java deleted file mode 100644 index e2fab0a1..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/java/com/wso2/openbanking/accelerator/consent/mgt/service/listener/ConsentStateChangeListener.java +++ /dev/null @@ -1,45 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.service.listener; - -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; - -import java.util.Map; - -/** - * Consent state change listener interface. - */ -public interface ConsentStateChangeListener { - - /** - * This method is used to put events to OBEventQueue related to different consent state changes. - * - * @param consentID consent ID - * @param userID user ID - * @param newConsentStatus new consent status after state change - * @param previousConsentStatus previous consent status - * @param reason reason for changing consent state - * @param clientId client ID - * @param consentDataMap consent data map holding different consent related data - * @throws ConsentManagementException thrown if an error occurs - */ - public void onStateChange(String consentID, String userID, String newConsentStatus, - String previousConsentStatus, String reason, String clientId, - Map consentDataMap) throws ConsentManagementException; -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/resources/findbugs-include.xml b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/resources/findbugs-include.xml deleted file mode 100644 index 0882a6d1..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,17 +0,0 @@ - diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/service/impl/OBConsentMgtCoreServiceTests.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/service/impl/OBConsentMgtCoreServiceTests.java deleted file mode 100644 index 360eafe6..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/service/impl/OBConsentMgtCoreServiceTests.java +++ /dev/null @@ -1,3401 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.service.impl; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.event.executor.OBEventQueue; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.util.DatabaseUtil; -import com.wso2.openbanking.accelerator.consent.mgt.dao.ConsentCoreDAO; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataDeletionException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataInsertionException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataRetrievalException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.exceptions.OBConsentDataUpdationException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentAttributes; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentFile; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentHistoryResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentMappingResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentStatusAuditRecord; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.persistence.ConsentStoreInitializer; -import com.wso2.openbanking.accelerator.consent.mgt.service.constants.ConsentCoreServiceConstants; -import com.wso2.openbanking.accelerator.consent.mgt.service.internal.ConsentManagementDataHolder; -import com.wso2.openbanking.accelerator.consent.mgt.service.util.ConsentMgtServiceTestData; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedUser; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.OAuth2Service; -import org.wso2.carbon.identity.oauth2.dao.AccessTokenDAOImpl; -import org.wso2.carbon.identity.oauth2.dto.OAuthRevocationRequestDTO; -import org.wso2.carbon.identity.oauth2.dto.OAuthRevocationResponseDTO; -import org.wso2.carbon.identity.oauth2.model.AccessTokenDO; - -import java.sql.Connection; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.HashSet; -import java.util.Map; -import java.util.Set; -import java.util.UUID; - -import static org.mockito.Matchers.any; - -/** - * Test for Open Banking consent management core service. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({DatabaseUtil.class, ConsentStoreInitializer.class, ConsentManagementDataHolder.class, - OpenBankingConfigParser.class}) -public class OBConsentMgtCoreServiceTests { - - private ConsentCoreServiceImpl consentCoreServiceImpl; - private ConsentCoreDAO mockedConsentCoreDAO; - private String sampleID; - private ConsentManagementDataHolder consentManagementDataHolderMock; - private OBEventQueue obEventQueueMock; - - @BeforeClass - public void initTest() { - - consentCoreServiceImpl = new ConsentCoreServiceImpl(); - mockedConsentCoreDAO = Mockito.mock(ConsentCoreDAO.class); - consentManagementDataHolderMock = Mockito.mock(ConsentManagementDataHolder.class); - obEventQueueMock = Mockito.mock(OBEventQueue.class); - } - - @BeforeMethod - public void mock() throws ConsentManagementException, IdentityOAuth2Exception { - - sampleID = UUID.randomUUID().toString(); - mockStaticClasses(); - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - - @Test (priority = 2) - public void testCreateAuthorizableConsent() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()).when(mockedConsentCoreDAO) - .storeConsentResource(Mockito.any(), Mockito.any()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleStoredTestAuthorizationResource()).when(mockedConsentCoreDAO) - .storeAuthorizationResource(Mockito.any(), Mockito.any()); - - DetailedConsentResource detailedConsentResource = - consentCoreServiceImpl.createAuthorizableConsent(ConsentMgtServiceTestData - .getSampleTestConsentResource(), ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_TYPE, true); - - Assert.assertNotNull(detailedConsentResource); - Assert.assertNotNull(detailedConsentResource.getConsentID()); - Assert.assertNotNull(detailedConsentResource.getClientID()); - Assert.assertNotNull(detailedConsentResource.getReceipt()); - Assert.assertNotNull(detailedConsentResource.getConsentType()); - Assert.assertNotNull(detailedConsentResource.getCurrentStatus()); - } - - @Test (priority = 2) - public void testCreateAuthorizableConsentWithIsImplicitAuthFalse() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()).when(mockedConsentCoreDAO) - .storeConsentResource(Mockito.any(), Mockito.any()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleStoredTestAuthorizationResource()).when(mockedConsentCoreDAO) - .storeAuthorizationResource(Mockito.any(), Mockito.any()); - - DetailedConsentResource detailedConsentResource = - consentCoreServiceImpl.createAuthorizableConsent(ConsentMgtServiceTestData - .getSampleTestConsentResource(), ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_TYPE, false); - - Assert.assertNotNull(detailedConsentResource); - Assert.assertNotNull(detailedConsentResource.getConsentID()); - Assert.assertNotNull(detailedConsentResource.getClientID()); - Assert.assertNotNull(detailedConsentResource.getReceipt()); - Assert.assertNotNull(detailedConsentResource.getConsentType()); - Assert.assertNotNull(detailedConsentResource.getCurrentStatus()); - } - - @Test (priority = 2) - public void testCreateAuthorizableConsentWithAttributes() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()).when(mockedConsentCoreDAO) - .storeConsentResource(Mockito.any(), Mockito.any()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestAuthorizationResource()) - .when(mockedConsentCoreDAO).storeAuthorizationResource(Mockito.any(), Mockito.any()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).storeConsentAttributes(Mockito.any(), - Mockito.any()); - - DetailedConsentResource detailedConsentResource = - consentCoreServiceImpl.createAuthorizableConsent(ConsentMgtServiceTestData - .getSampleStoredTestConsentResourceWithAttributes(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_TYPE, true); - - Assert.assertNotNull(detailedConsentResource); - Assert.assertNotNull(detailedConsentResource.getConsentID()); - Assert.assertNotNull(detailedConsentResource.getClientID()); - Assert.assertNotNull(detailedConsentResource.getReceipt()); - Assert.assertNotNull(detailedConsentResource.getConsentType()); - Assert.assertNotNull(detailedConsentResource.getCurrentStatus()); - Assert.assertNotNull(detailedConsentResource.getConsentAttributes()); - } - - @Test (priority = 2) - public void testCreateAuthorizableConsentWithoutUserID() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()).when(mockedConsentCoreDAO) - .storeConsentResource(Mockito.any(), Mockito.any()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestAuthorizationResource()) - .when(mockedConsentCoreDAO).storeAuthorizationResource(Mockito.any(), Mockito.any()); - - DetailedConsentResource detailedConsentResource = - consentCoreServiceImpl.createAuthorizableConsent(ConsentMgtServiceTestData - .getSampleStoredTestConsentResourceWithAttributes(), null, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_TYPE, true); - - Assert.assertNotNull(detailedConsentResource); - Assert.assertNotNull(detailedConsentResource.getConsentID()); - Assert.assertNotNull(detailedConsentResource.getClientID()); - Assert.assertNotNull(detailedConsentResource.getReceipt()); - Assert.assertNotNull(detailedConsentResource.getConsentType()); - Assert.assertNotNull(detailedConsentResource.getCurrentStatus()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateAuthorizableConsentWithoutClientID() throws Exception { - - ConsentResource consentResource = ConsentMgtServiceTestData.getSampleTestConsentResource(); - consentResource.setClientID(null); - - consentCoreServiceImpl.createAuthorizableConsent(consentResource, Mockito.anyString(), Mockito.anyString(), - Mockito.anyString(), Mockito.anyBoolean()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateAuthorizableConsentWithoutReceipt() throws Exception { - - ConsentResource consentResource = ConsentMgtServiceTestData.getSampleTestConsentResource(); - consentResource.setReceipt(null); - - consentCoreServiceImpl.createAuthorizableConsent(consentResource, Mockito.anyString(), Mockito.anyString(), - Mockito.anyString(), Mockito.anyBoolean()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateAuthorizableConsentWithoutConsentType() throws Exception { - - ConsentResource consentResource = ConsentMgtServiceTestData.getSampleTestConsentResource(); - consentResource.setConsentType(null); - - consentCoreServiceImpl.createAuthorizableConsent(consentResource, Mockito.anyString(), Mockito.anyString(), - Mockito.anyString(), Mockito.anyBoolean()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateAuthorizableConsentWithoutCurrentStatus() throws Exception { - - ConsentResource consentResource = ConsentMgtServiceTestData.getSampleTestConsentResource(); - consentResource.setCurrentStatus(null); - - consentCoreServiceImpl.createAuthorizableConsent(consentResource, Mockito.anyString(), Mockito.anyString(), - Mockito.anyString(), Mockito.anyBoolean()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateAuthorizableConsentWithImplicitAndNoAuthStatus() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()).when(mockedConsentCoreDAO) - .storeConsentResource(Mockito.any(), Mockito.any()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestAuthorizationResource()) - .when(mockedConsentCoreDAO).storeAuthorizationResource(Mockito.any(), Mockito.any()); - - consentCoreServiceImpl.createAuthorizableConsent(ConsentMgtServiceTestData - .getSampleStoredTestConsentResourceWithAttributes(), null, null, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_TYPE, true); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateAuthorizableConsentWithImplicitAndNoAuthType() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()).when(mockedConsentCoreDAO) - .storeConsentResource(Mockito.any(), Mockito.any()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestAuthorizationResource()) - .when(mockedConsentCoreDAO).storeAuthorizationResource(Mockito.any(), Mockito.any()); - - consentCoreServiceImpl.createAuthorizableConsent(ConsentMgtServiceTestData - .getSampleStoredTestConsentResourceWithAttributes(), null, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, null, true); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateAuthorizableConsentRollback() throws Exception { - - Mockito.doThrow(OBConsentDataInsertionException.class).when(mockedConsentCoreDAO) - .storeConsentResource(Mockito.any(), Mockito.any()); - - consentCoreServiceImpl.createAuthorizableConsent(ConsentMgtServiceTestData - .getSampleTestConsentResource(), ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_TYPE, true); - } - - @Test - public void testCreateExclusiveConsent() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResourcesList()) - .when(mockedConsentCoreDAO).searchConsents(Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.any(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyInt(), - Mockito.anyInt()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleTestConsentResource()) - .when(mockedConsentCoreDAO).updateConsentStatus(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentStatusAuditRecord(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.anyObject(), Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleTestConsentResource()) - .when(mockedConsentCoreDAO).storeConsentResource(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).storeConsentAttributes(Mockito.any(), - Mockito.anyObject()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestAuthorizationResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID)) - .when(mockedConsentCoreDAO).storeAuthorizationResource(Mockito.any(), Mockito.anyObject()); - - DetailedConsentResource exclusiveConsent = - consentCoreServiceImpl.createExclusiveConsent(ConsentMgtServiceTestData - .getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, true); - Assert.assertNotNull(exclusiveConsent); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateExclusiveConsentDataRetrieveError() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).searchConsents(Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.any(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyInt(), - Mockito.anyInt()); - - consentCoreServiceImpl.createExclusiveConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, true); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateExclusiveConsentDataUpdateError() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResourcesList()) - .when(mockedConsentCoreDAO).searchConsents(Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.any(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyInt(), - Mockito.anyInt()); - Mockito.doThrow(OBConsentDataUpdationException.class) - .when(mockedConsentCoreDAO).updateConsentStatus(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - - consentCoreServiceImpl.createExclusiveConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, true); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateExclusiveConsentDataInsertError() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResourcesList()) - .when(mockedConsentCoreDAO).searchConsents(Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.any(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyInt(), - Mockito.anyInt()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleTestConsentResource()) - .when(mockedConsentCoreDAO).updateConsentStatus(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - Mockito.doThrow(OBConsentDataInsertionException.class) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - - consentCoreServiceImpl.createExclusiveConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, true); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateExclusiveConsentWithoutClientID() throws Exception { - - ConsentResource sampleConsentResource = ConsentMgtServiceTestData.getSampleStoredTestConsentResource(); - sampleConsentResource.setClientID(null); - - consentCoreServiceImpl.createExclusiveConsent(sampleConsentResource, - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, true); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateExclusiveConsentWithoutReceipt() throws Exception { - - ConsentResource sampleConsentResource = ConsentMgtServiceTestData.getSampleStoredTestConsentResource(); - sampleConsentResource.setReceipt(null); - - consentCoreServiceImpl.createExclusiveConsent(sampleConsentResource, - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, true); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateExclusiveConsentWithoutConsentType() throws Exception { - - ConsentResource sampleConsentResource = ConsentMgtServiceTestData.getSampleStoredTestConsentResource(); - sampleConsentResource.setConsentType(null); - - consentCoreServiceImpl.createExclusiveConsent(sampleConsentResource, - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, true); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateExclusiveConsentWithoutConsentStatus() throws Exception { - - ConsentResource sampleConsentResource = ConsentMgtServiceTestData.getSampleStoredTestConsentResource(); - sampleConsentResource.setCurrentStatus(null); - - consentCoreServiceImpl.createExclusiveConsent(sampleConsentResource, - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, true); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateExclusiveConsentWithoutUserID() throws Exception { - - consentCoreServiceImpl.createExclusiveConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - null, ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, true); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateExclusiveConsentWithouAuthStatus() throws Exception { - - consentCoreServiceImpl.createExclusiveConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, null, - ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, true); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateExclusiveConsentWithouAuthType() throws Exception { - - consentCoreServiceImpl.createExclusiveConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - null, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, true); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateExclusiveConsentWithImplicitAuthFalse() throws Exception { - - consentCoreServiceImpl.createExclusiveConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - null, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, false); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateExclusiveConsentWithouApplicableExistingConsentStatus() throws Exception { - - consentCoreServiceImpl.createExclusiveConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, null, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, true); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateExclusiveConsentWithouNewExistingConsentStatus() throws Exception { - - consentCoreServiceImpl.createExclusiveConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - null, true); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateExclusiveConsentWithouNewCurrentConsentStatus() throws Exception { - - consentCoreServiceImpl.createExclusiveConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - null, true); - } - - @Test - public void testRevokeConsent() throws Exception { - - DetailedConsentResource retrievedDetailedConsentResource = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - - Mockito.doReturn(retrievedDetailedConsentResource).when(mockedConsentCoreDAO) - .getDetailedConsentResource(Mockito.any(), Mockito.any(), Mockito.anyBoolean()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentResource()).when(mockedConsentCoreDAO) - .updateConsentStatus(Mockito.any(), Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleTestConsentStatusAuditRecord( - retrievedDetailedConsentResource.getConsentID(), retrievedDetailedConsentResource.getCurrentStatus())) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.any(), Mockito.anyString()); - - boolean isConsentRevoked = new MockConsentCoreServiceImpl() - .revokeConsentWithReason(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_USER_ID, - false, ConsentCoreServiceConstants.CONSENT_REVOKE_REASON); - - Assert.assertTrue(isConsentRevoked); - } - - @Test - public void testRevokeConsentAndTokens() throws Exception { - - DetailedConsentResource retrievedDetailedConsentResource = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - - Mockito.doReturn(retrievedDetailedConsentResource).when(mockedConsentCoreDAO) - .getDetailedConsentResource(Mockito.any(), Mockito.any(), Mockito.anyBoolean()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentResource()).when(mockedConsentCoreDAO) - .updateConsentStatus(Mockito.any(), Mockito.anyString(), Mockito.anyString()); - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleTestConsentStatusAuditRecord( - retrievedDetailedConsentResource.getConsentID(), retrievedDetailedConsentResource.getCurrentStatus())) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.any(), Mockito.anyString()); - - boolean isConsentRevoked = new MockConsentCoreServiceImpl() - .revokeConsent(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_USER_ID, - true); - - - Assert.assertTrue(isConsentRevoked); - } - - @Test - public void testRevokeConsentAndTokensTokenRevokeError() throws Exception { - - DetailedConsentResource retrievedDetailedConsentResource = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - - Mockito.doReturn(retrievedDetailedConsentResource).when(mockedConsentCoreDAO) - .getDetailedConsentResource(Mockito.any(), Mockito.any(), Mockito.anyBoolean()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentResource()).when(mockedConsentCoreDAO) - .updateConsentStatus(Mockito.any(), Mockito.anyString(), Mockito.anyString()); - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleTestConsentStatusAuditRecord( - retrievedDetailedConsentResource.getConsentID(), retrievedDetailedConsentResource.getCurrentStatus())) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.any(), Mockito.anyString()); - try { - boolean isConsentRevoked = new MockConsentCoreServiceImplTokenError() - .revokeConsentWithReason(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_USER_ID, - true, ConsentCoreServiceConstants.CONSENT_REVOKE_REASON); - Assert.assertTrue(isConsentRevoked); - } catch (Exception e) { - Assert.assertTrue(e instanceof ConsentManagementException); - } - - } - - @Test - public void testRevokeConsentWithoutConsentAttributes() throws Exception { - - DetailedConsentResource retrievedDetailedConsentResource = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - retrievedDetailedConsentResource.setConsentAttributes(null); - - Mockito.doReturn(retrievedDetailedConsentResource).when(mockedConsentCoreDAO) - .getDetailedConsentResource(Mockito.any(), Mockito.any(), Mockito.anyBoolean()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentResource()).when(mockedConsentCoreDAO) - .updateConsentStatus(Mockito.any(), Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleTestConsentStatusAuditRecord( - retrievedDetailedConsentResource.getConsentID(), retrievedDetailedConsentResource.getCurrentStatus())) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.any(), Mockito.anyString()); - - boolean isConsentRevoked = new MockConsentCoreServiceImpl() - .revokeConsent(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_USER_ID, - false); - - Assert.assertTrue(isConsentRevoked); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testRevokeConsentWithoutConsentID() throws Exception { - - consentCoreServiceImpl.revokeConsentWithReason(null, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_USER_ID, - false, ConsentCoreServiceConstants.CONSENT_REVOKE_REASON); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testRevokeConsentWithoutNewConsentStatus() throws Exception { - - consentCoreServiceImpl.revokeConsent(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - null, ConsentMgtServiceTestData.SAMPLE_USER_ID, false); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testRevokeConsentWithoutApplicableStatusToRevoke() throws Exception { - - consentCoreServiceImpl.revokeConsent(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_USER_ID, - false); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testRevokeConsentWithInvalidApplicableStatus() throws Exception { - - DetailedConsentResource retrievedDetailedConsentResource = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - - Mockito.doReturn(retrievedDetailedConsentResource).when(mockedConsentCoreDAO) - .getDetailedConsentResource(Mockito.any(), Mockito.any(), Mockito.anyBoolean()); - - consentCoreServiceImpl.revokeConsent(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_USER_ID, - false); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testRevokeConsentDataRetrievalError() throws Exception { - - DetailedConsentResource retrievedDetailedConsentResource = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - - Mockito.doThrow(OBConsentDataRetrievalException.class).when(mockedConsentCoreDAO) - .getDetailedConsentResource(Mockito.any(), Mockito.any(), Mockito.anyBoolean()); - - consentCoreServiceImpl.revokeConsentWithReason(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_USER_ID, - false, ConsentCoreServiceConstants.CONSENT_REVOKE_REASON); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testRevokeConsentDataInsertionError() throws Exception { - - DetailedConsentResource retrievedDetailedConsentResource = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - - Mockito.doReturn(retrievedDetailedConsentResource).when(mockedConsentCoreDAO) - .getDetailedConsentResource(Mockito.any(), Mockito.any(), Mockito.anyBoolean()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentResource()).when(mockedConsentCoreDAO) - .updateConsentStatus(Mockito.any(), Mockito.anyString(), Mockito.anyString()); - Mockito.doThrow(OBConsentDataInsertionException.class) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - - new MockConsentCoreServiceImpl().revokeConsent(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_USER_ID, - false); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testRevokeConsentDataUpdationError() throws Exception { - - DetailedConsentResource retrievedDetailedConsentResource = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - - Mockito.doReturn(retrievedDetailedConsentResource).when(mockedConsentCoreDAO) - .getDetailedConsentResource(Mockito.any(), Mockito.any(), Mockito.anyBoolean()); - Mockito.doThrow(OBConsentDataUpdationException.class).when(mockedConsentCoreDAO) - .updateConsentStatus(Mockito.any(), Mockito.anyString(), Mockito.anyString()); - - consentCoreServiceImpl.revokeConsentWithReason(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_USER_ID, - false, ConsentCoreServiceConstants.CONSENT_REVOKE_REASON); - } - - @Test - public void testRevokeExistingApplicableConsents() throws Exception { - - ArrayList detailedConsentResources = new ArrayList<>(); - detailedConsentResources.add(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()); - - Mockito.doReturn(detailedConsentResources).when(mockedConsentCoreDAO) - .searchConsents(Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyInt(), Mockito.anyInt()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()).when(mockedConsentCoreDAO) - .updateConsentStatus(Mockito.any(), Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentStatusAuditRecord(sampleID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)).when(mockedConsentCoreDAO) - .storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.any(), Mockito.any()); - - Assert.assertTrue(consentCoreServiceImpl.revokeExistingApplicableConsents(sampleID, - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - false)); - } - - @Test - public void testRevokeExistingApplicableConsentsWithTokens() throws Exception { - - ArrayList detailedConsentResources = new ArrayList<>(); - detailedConsentResources.add(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()); - - Mockito.doReturn(detailedConsentResources).when(mockedConsentCoreDAO) - .searchConsents(Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyInt(), Mockito.anyInt()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()).when(mockedConsentCoreDAO) - .updateConsentStatus(Mockito.any(), Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentStatusAuditRecord(sampleID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)).when(mockedConsentCoreDAO) - .storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.any(), Mockito.any()); - - Assert.assertTrue(new MockConsentCoreServiceImpl().revokeExistingApplicableConsents(sampleID, - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - true)); - } - - @Test - public void testRevokeExistingApplicableConsentsWithConsentsWithNoAttributes() throws Exception { - - DetailedConsentResource detailedConsentResource = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - detailedConsentResource.setConsentAttributes(null); - - ArrayList detailedConsentResources = new ArrayList<>(); - detailedConsentResources.add(detailedConsentResource); - - Mockito.doReturn(detailedConsentResources).when(mockedConsentCoreDAO) - .searchConsents(Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyInt(), Mockito.anyInt()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()).when(mockedConsentCoreDAO) - .updateConsentStatus(Mockito.any(), Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentStatusAuditRecord(sampleID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)).when(mockedConsentCoreDAO) - .storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.any(), Mockito.any()); - - consentCoreServiceImpl.revokeExistingApplicableConsents(sampleID, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, false); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testRevokeExistingApplicableConsentsRetrieveError() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class).when(mockedConsentCoreDAO) - .searchConsents(Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyInt(), Mockito.anyInt()); - - consentCoreServiceImpl.revokeExistingApplicableConsents(sampleID, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, false); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testRevokeExistingApplicableConsentsUpdateError() throws Exception { - - ArrayList detailedConsentResources = new ArrayList<>(); - detailedConsentResources.add(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()); - - Mockito.doReturn(detailedConsentResources).when(mockedConsentCoreDAO) - .searchConsents(Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyInt(), Mockito.anyInt()); - Mockito.doThrow(OBConsentDataUpdationException.class).when(mockedConsentCoreDAO) - .updateConsentStatus(Mockito.any(), Mockito.anyString(), Mockito.anyString()); - - consentCoreServiceImpl.revokeExistingApplicableConsents(sampleID, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, false); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testRevokeExistingApplicableConsentsInsertionError() throws Exception { - - ArrayList detailedConsentResources = new ArrayList<>(); - detailedConsentResources.add(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()); - - Mockito.doReturn(detailedConsentResources).when(mockedConsentCoreDAO) - .searchConsents(Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyInt(), Mockito.anyInt()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()).when(mockedConsentCoreDAO) - .updateConsentStatus(Mockito.any(), Mockito.anyString(), Mockito.anyString()); - Mockito.doThrow(OBConsentDataInsertionException.class).when(mockedConsentCoreDAO) - .storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - - consentCoreServiceImpl.revokeExistingApplicableConsents(sampleID, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, false); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testRevokeExistingApplicableConsentsWithoutClientID() throws Exception { - - consentCoreServiceImpl.revokeExistingApplicableConsents(null, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, false); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testRevokeExistingApplicableConsentsWithoutRevokedConsentStatus() throws Exception { - - consentCoreServiceImpl.revokeExistingApplicableConsents(ConsentMgtServiceTestData.SAMPLE_CLIENT_ID, - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, null, false); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testRevokeExistingApplicableConsentsWithoutUserID() throws Exception { - - consentCoreServiceImpl.revokeExistingApplicableConsents(ConsentMgtServiceTestData.SAMPLE_CLIENT_ID, - null, ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS - , false); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testRevokeExistingApplicableConsentsWithoutConsentType() throws Exception { - - consentCoreServiceImpl.revokeExistingApplicableConsents(ConsentMgtServiceTestData.SAMPLE_CLIENT_ID, - ConsentMgtServiceTestData.SAMPLE_USER_ID, null, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS - , false); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testRevokeExistingApplicableConsentsWithoutApplicableStatusToRevoke() throws Exception { - - consentCoreServiceImpl.revokeExistingApplicableConsents(ConsentMgtServiceTestData.SAMPLE_CLIENT_ID, - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE, - null, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS - , false); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateConsentFileErrorWhenRetrieval() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class).when(mockedConsentCoreDAO) - .getConsentResource(Mockito.any(), Mockito.anyString()); - - consentCoreServiceImpl.createConsentFile(ConsentMgtServiceTestData - .getSampleConsentFileObject(ConsentMgtServiceTestData.SAMPLE_CONSENT_FILE), - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.AWAITING_UPLOAD_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateConsentFileRollBackWhenCreation() throws Exception { - - ConsentResource storedConsentResource = ConsentMgtServiceTestData.getSampleStoredTestConsentResource(); - - Mockito.doReturn(storedConsentResource).when(mockedConsentCoreDAO) - .getConsentResource(Mockito.any(), Mockito.anyString()); - Mockito.doThrow(OBConsentDataInsertionException.class).when(mockedConsentCoreDAO) - .storeConsentFile(Mockito.any(), Mockito.any()); - - consentCoreServiceImpl.createConsentFile(ConsentMgtServiceTestData - .getSampleConsentFileObject(ConsentMgtServiceTestData.SAMPLE_CONSENT_FILE), - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_USER_ID, - Mockito.anyString()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateConsentFileRollBackWhenUpdating() throws Exception { - - ConsentResource storedConsentResource = ConsentMgtServiceTestData.getSampleStoredTestConsentResource(); - - Mockito.doReturn(storedConsentResource).when(mockedConsentCoreDAO) - .getConsentResource(Mockito.any(), Mockito.anyString()); - Mockito.doThrow(OBConsentDataUpdationException.class).when(mockedConsentCoreDAO) - .updateConsentStatus(Mockito.any(), Mockito.anyString(), Mockito.anyString()); - - consentCoreServiceImpl.createConsentFile(ConsentMgtServiceTestData - .getSampleConsentFileObject(ConsentMgtServiceTestData.SAMPLE_CONSENT_FILE), - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_USER_ID, - storedConsentResource.getCurrentStatus()); - } - - @Test(expectedExceptions = ConsentManagementException.class) - public void testCreateConsentFileWithInvalidStatus() - throws Exception { - - ConsentResource consentResource = ConsentMgtServiceTestData.getSampleTestConsentResource(); - consentResource.setCurrentStatus(ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - - Mockito.doReturn(consentResource).when(mockedConsentCoreDAO).getConsentResource(Mockito.any(), - Mockito.anyString()); - - // Create consent file - consentCoreServiceImpl.createConsentFile(ConsentMgtServiceTestData - .getSampleConsentFileObject(ConsentMgtServiceTestData.SAMPLE_CONSENT_FILE), - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.AWAITING_UPLOAD_STATUS); - } - - @Test(expectedExceptions = ConsentManagementException.class) - public void testCreateConsentFileWithoutFileContent() throws Exception { - - consentCoreServiceImpl.createConsentFile(ConsentMgtServiceTestData - .getSampleConsentFileObject(null), ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_USER_ID, ConsentMgtServiceTestData.AWAITING_UPLOAD_STATUS); - } - - @Test(expectedExceptions = ConsentManagementException.class) - public void testCreateConsentFileWithoutConsentID() throws Exception { - - ConsentFile sampleConsentFile = - ConsentMgtServiceTestData.getSampleConsentFileObject(ConsentMgtServiceTestData.SAMPLE_CONSENT_FILE); - - sampleConsentFile.setConsentID(null); - consentCoreServiceImpl.createConsentFile(sampleConsentFile, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.AWAITING_UPLOAD_STATUS); - } - - @Test(expectedExceptions = ConsentManagementException.class) - public void testCreateConsentFileWithoutNewConsentStatus() - throws Exception { - - consentCoreServiceImpl.createConsentFile(ConsentMgtServiceTestData - .getSampleConsentFileObject(ConsentMgtServiceTestData.SAMPLE_CONSENT_FILE), - null, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.AWAITING_UPLOAD_STATUS); - } - - @Test(expectedExceptions = ConsentManagementException.class) - public void testCreateConsentFileWithoutApplicableStatusForFileUpload() - throws Exception { - - consentCoreServiceImpl.createConsentFile(ConsentMgtServiceTestData - .getSampleConsentFileObject(ConsentMgtServiceTestData.SAMPLE_CONSENT_FILE), - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_USER_ID, - null); - } - - @Test - public void testGetConsent() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()).when(mockedConsentCoreDAO) - .getConsentResource(Mockito.any(), Mockito.anyString()); - - // Get consent - ConsentResource retrievedConsentResource = consentCoreServiceImpl.getConsent(ConsentMgtServiceTestData - .getSampleStoredTestConsentResource().getConsentID(), false); - - Assert.assertNotNull(retrievedConsentResource); - } - - @Test - public void testGetConsentWithAttributes() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResourceWithAttributes()) - .when(mockedConsentCoreDAO).getConsentResourceWithAttributes(Mockito.any(), Mockito.anyString()); - - // Get consent - ConsentResource retrievedConsentResource = consentCoreServiceImpl.getConsent(ConsentMgtServiceTestData - .getSampleStoredTestConsentResource().getConsentID(), true); - - Assert.assertNotNull(retrievedConsentResource); - Assert.assertNotNull(retrievedConsentResource.getConsentAttributes()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentRollBackWhenRetrieve() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class).when(mockedConsentCoreDAO) - .getConsentResource(Mockito.any(), Mockito.anyString()); - - // Get consent - consentCoreServiceImpl.getConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource().getConsentID(), - false); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentWithoutConsentID() throws Exception { - - consentCoreServiceImpl.getConsent(null, false); - } - - @Test - public void testGetDetailedConsent() throws Exception { - - DetailedConsentResource detailedConsentResource = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - - Mockito.doReturn(detailedConsentResource).when(mockedConsentCoreDAO) - .getDetailedConsentResource(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - - // Get consent - DetailedConsentResource retrievedConsentResource = - consentCoreServiceImpl.getDetailedConsent(detailedConsentResource.getConsentID()); - - Assert.assertNotNull(retrievedConsentResource); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetDetailedConsentWithoutConsentID() throws Exception { - - consentCoreServiceImpl.getDetailedConsent(null); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetDetailedConsentRetrieveError() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class).when(mockedConsentCoreDAO) - .getDetailedConsentResource(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - consentCoreServiceImpl.getDetailedConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource() - .getConsentID()); - } - - @Test - public void testCreateConsentAuthorization() throws Exception { - - AuthorizationResource sampleAuthorizationResource = - ConsentMgtServiceTestData.getSampleTestAuthorizationResource(sampleID); - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestAuthorizationResource()) - .when(mockedConsentCoreDAO).storeAuthorizationResource(Mockito.any(), Mockito.any()); - - //Create a consent authorization resource - AuthorizationResource storedAuthorizationResource = - consentCoreServiceImpl.createConsentAuthorization(sampleAuthorizationResource); - - Assert.assertNotNull(storedAuthorizationResource); - Assert.assertNotNull(storedAuthorizationResource.getAuthorizationID()); - Assert.assertNotNull(storedAuthorizationResource.getConsentID()); - Assert.assertNotNull(storedAuthorizationResource.getAuthorizationType()); - Assert.assertNotNull(storedAuthorizationResource.getUserID()); - Assert.assertNotNull(storedAuthorizationResource.getAuthorizationStatus()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateConsentAuthorizationRollbackWhenCreation() throws Exception { - - AuthorizationResource sampleAuthorizationResource = - ConsentMgtServiceTestData.getSampleTestAuthorizationResource(sampleID); - - Mockito.doThrow(OBConsentDataInsertionException.class).when(mockedConsentCoreDAO) - .storeAuthorizationResource(Mockito.any(), Mockito.any()); - - // Get consent - consentCoreServiceImpl.createConsentAuthorization(sampleAuthorizationResource); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateConsentAuthorizationWithoutConsentID() throws Exception { - - AuthorizationResource sampleAuthorizationResource = - ConsentMgtServiceTestData.getSampleTestAuthorizationResource(null); - - //Create a consent authorization resource - consentCoreServiceImpl.createConsentAuthorization(sampleAuthorizationResource); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateConsentAuthorizationWithoutAuthorizationStatus() throws Exception { - - AuthorizationResource sampleAuthorizationResource = - ConsentMgtServiceTestData.getSampleTestAuthorizationResource(sampleID); - - // Explicitly setting authorization status to null - sampleAuthorizationResource.setAuthorizationStatus(null); - - //Create a consent authorization resource - consentCoreServiceImpl.createConsentAuthorization(sampleAuthorizationResource); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateConsentAuthorizationWithoutAuthorizationType() throws Exception { - - AuthorizationResource sampleAuthorizationResource = - ConsentMgtServiceTestData.getSampleTestAuthorizationResource(sampleID); - sampleAuthorizationResource.setAuthorizationType(null); - - //Create a consent authorization resource - consentCoreServiceImpl.createConsentAuthorization(sampleAuthorizationResource); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateConsentAuthorizationWithoutAuthorizationUserID() throws Exception { - - AuthorizationResource sampleAuthorizationResource = - ConsentMgtServiceTestData.getSampleTestAuthorizationResource(sampleID); - sampleAuthorizationResource.setUserID(null); - - //Create a consent authorization resource - consentCoreServiceImpl.createConsentAuthorization(sampleAuthorizationResource); - } - - @Test - public void testCreateConsentAccountMapping() throws Exception { - - AuthorizationResource storedAuthorizationResource = - ConsentMgtServiceTestData.getSampleStoredTestAuthorizationResource(); - - ConsentMappingResource storedConsentMappingResource = - ConsentMgtServiceTestData.getSampleStoredTestConsentMappingResource(sampleID); - - Mockito.doReturn(storedConsentMappingResource).when(mockedConsentCoreDAO) - .storeConsentMappingResource(Mockito.any(), Mockito.any()); - - ArrayList storedConsentMappingResources = - consentCoreServiceImpl.createConsentAccountMappings(storedAuthorizationResource.getAuthorizationID(), - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP); - - Assert.assertNotNull(storedConsentMappingResources); - for (ConsentMappingResource resource : storedConsentMappingResources) { - Assert.assertNotNull(resource.getAccountID()); - Assert.assertNotNull(resource.getPermission()); - Assert.assertNotNull(resource.getAuthorizationID()); - Assert.assertEquals(resource.getMappingStatus(), ConsentCoreServiceConstants.ACTIVE_MAPPING_STATUS); - } - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateConsentAccountMappingRollBackWhenCreation() throws Exception { - - AuthorizationResource storedAuthorizationResource = - ConsentMgtServiceTestData.getSampleStoredTestAuthorizationResource(); - - Mockito.doThrow(OBConsentDataInsertionException.class).when(mockedConsentCoreDAO) - .storeConsentMappingResource(Mockito.any(), Mockito.any()); - - consentCoreServiceImpl.createConsentAccountMappings(storedAuthorizationResource.getAuthorizationID(), - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateConsentAccountMappingWithoutAuthID() throws Exception { - - consentCoreServiceImpl.createConsentAccountMappings(null, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testCreateConsentAccountMappingWithoutAccountAndPermissionsMap() throws Exception { - - consentCoreServiceImpl.createConsentAccountMappings(sampleID, new HashMap<>()); - } - - @Test - public void testDeactivateAccountMappings() throws Exception { - - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.any(), Mockito.any()); - Assert.assertTrue(consentCoreServiceImpl - .deactivateAccountMappings(ConsentMgtServiceTestData.UNMATCHED_MAPPING_IDS)); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testDeactivateAccountMappingsWithEmptyMappingIDList() throws Exception { - - consentCoreServiceImpl.deactivateAccountMappings(new ArrayList<>()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testDeactivateAccountMappingsRollback() throws Exception { - - Mockito.doThrow(OBConsentDataUpdationException.class).when(mockedConsentCoreDAO) - .updateConsentMappingStatus(Mockito.any(), Mockito.any(), Mockito.any()); - consentCoreServiceImpl.deactivateAccountMappings(ConsentMgtServiceTestData.UNMATCHED_MAPPING_IDS); - } - - @Test - public void testUpdateAccountMappingPermissionWithEmptyMap() { - - try { - consentCoreServiceImpl.updateAccountMappingPermission(new HashMap<>()); - Assert.fail("Expected ConsentManagementException to be thrown"); - } catch (ConsentManagementException e) { - Assert.assertEquals(e.getMessage(), "Cannot proceed since account mapping IDs are not provided"); - } - } - - @Test - public void testUpdateAccountMappingPermission() throws Exception { - - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingPermission(Mockito.any(), - Mockito.any()); - Assert.assertTrue(consentCoreServiceImpl - .updateAccountMappingPermission(ConsentMgtServiceTestData.SAMPLE_MAPPING_ID_PERMISSION_MAP)); - } - - @Test - public void testSearchConsents() throws Exception { - - ArrayList detailedConsentResources = new ArrayList<>(); - detailedConsentResources.add(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()); - - Mockito.doReturn(detailedConsentResources) - .when(mockedConsentCoreDAO).searchConsents(Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.any(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyInt(), Mockito.anyInt()); - - consentCoreServiceImpl.searchDetailedConsents(ConsentMgtServiceTestData.SAMPLE_CLIENT_IDS_LIST, - ConsentMgtServiceTestData.SAMPLE_CLIENT_IDS_LIST, ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPES_LIST, - ConsentMgtServiceTestData.SAMPLE_CONSENT_STATUSES_LIST, ConsentMgtServiceTestData.SAMPLE_USER_IDS_LIST, - 12345L, 23456L, null, null); - } - - @Test - public void testSearchConsentsInRetention() throws Exception { - - ArrayList detailedConsentResources = new ArrayList<>(); - detailedConsentResources.add(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()); - - Mockito.doReturn(detailedConsentResources) - .when(mockedConsentCoreDAO).searchConsents(Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.any(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyInt(), - Mockito.anyInt()); - - consentCoreServiceImpl.searchDetailedConsents(ConsentMgtServiceTestData.SAMPLE_CLIENT_IDS_LIST, - ConsentMgtServiceTestData.SAMPLE_CLIENT_IDS_LIST, ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPES_LIST, - ConsentMgtServiceTestData.SAMPLE_CONSENT_STATUSES_LIST, ConsentMgtServiceTestData.SAMPLE_USER_IDS_LIST, - 12345L, 23456L, null, null, false); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testSearchConsentsRetrieveError() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).searchConsents(Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.any(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyInt(), Mockito.anyInt()); - - consentCoreServiceImpl.searchDetailedConsents(ConsentMgtServiceTestData.SAMPLE_CLIENT_IDS_LIST, - ConsentMgtServiceTestData.SAMPLE_CLIENT_IDS_LIST, ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPES_LIST, - ConsentMgtServiceTestData.SAMPLE_CONSENT_STATUSES_LIST, ConsentMgtServiceTestData.SAMPLE_USER_IDS_LIST, - 12345L, 23456L, null, null); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testSearchConsentsRetrieveErrorInRetention() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).searchConsents(Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.any(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyInt(), - Mockito.anyInt()); - - consentCoreServiceImpl.searchDetailedConsents(ConsentMgtServiceTestData.SAMPLE_CLIENT_IDS_LIST, - ConsentMgtServiceTestData.SAMPLE_CLIENT_IDS_LIST, ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPES_LIST, - ConsentMgtServiceTestData.SAMPLE_CONSENT_STATUSES_LIST, ConsentMgtServiceTestData.SAMPLE_USER_IDS_LIST, - 12345L, 23456L, null, null, true); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testSearchConsentsWithLimits() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).searchConsents(Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.any(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyInt(), Mockito.anyInt()); - - consentCoreServiceImpl.searchDetailedConsents(ConsentMgtServiceTestData.SAMPLE_CLIENT_IDS_LIST, - ConsentMgtServiceTestData.SAMPLE_CLIENT_IDS_LIST, ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPES_LIST, - ConsentMgtServiceTestData.SAMPLE_CONSENT_STATUSES_LIST, ConsentMgtServiceTestData.SAMPLE_USER_IDS_LIST, - 12345L, 23456L, 1, 0); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testSearchConsentsInRetentionDBWithLimits() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).searchConsents(Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.any(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyInt(), - Mockito.anyInt()); - - consentCoreServiceImpl.searchDetailedConsents(ConsentMgtServiceTestData.SAMPLE_CLIENT_IDS_LIST, - ConsentMgtServiceTestData.SAMPLE_CLIENT_IDS_LIST, ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPES_LIST, - ConsentMgtServiceTestData.SAMPLE_CONSENT_STATUSES_LIST, ConsentMgtServiceTestData.SAMPLE_USER_IDS_LIST, - 12345L, 23456L, 1, 0, true); - } - - @Test - public void testBindUserAccountsToConsentWithAccountIdList() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestAuthorizationResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID)) - .when(mockedConsentCoreDAO).updateAuthorizationUser(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestAuthorizationResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID)) - .when(mockedConsentCoreDAO).updateAuthorizationStatus(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentMappingResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID)) - .when(mockedConsentCoreDAO).storeConsentMappingResource(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentResource()) - .when(mockedConsentCoreDAO).updateConsentStatus(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentStatusAuditRecord(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - - Assert.assertTrue(consentCoreServiceImpl - .bindUserAccountsToConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, "authID", - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_ID_LIST, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)); - } - - @Test - public void testBindUserAccountsToConsent() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestAuthorizationResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID)) - .when(mockedConsentCoreDAO).updateAuthorizationUser(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestAuthorizationResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID)) - .when(mockedConsentCoreDAO).updateAuthorizationStatus(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentMappingResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID)) - .when(mockedConsentCoreDAO).storeConsentMappingResource(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentResource()) - .when(mockedConsentCoreDAO).updateConsentStatus(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentStatusAuditRecord(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - - Assert.assertTrue(consentCoreServiceImpl - .bindUserAccountsToConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, "authID", - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testBindUserAccountsToConsentWithoutNewCurrentConsentStatus() throws Exception { - - consentCoreServiceImpl.bindUserAccountsToConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, "authID", - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, null); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testBindUserAccountsToConsentWithoutConsentID() throws Exception { - - ConsentResource consentResource = ConsentMgtServiceTestData.getSampleStoredTestConsentResource(); - consentResource.setConsentID(null); - - consentCoreServiceImpl.bindUserAccountsToConsent(consentResource, - ConsentMgtServiceTestData.SAMPLE_USER_ID, "authID", - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testBindUserAccountsToConsentWithoutClientID() throws Exception { - - ConsentResource consentResource = ConsentMgtServiceTestData.getSampleStoredTestConsentResource(); - consentResource.setClientID(null); - - consentCoreServiceImpl.bindUserAccountsToConsent(consentResource, - ConsentMgtServiceTestData.SAMPLE_USER_ID, "authID", - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testBindUserAccountsToConsentWithotConsentType() throws Exception { - - ConsentResource consentResource = ConsentMgtServiceTestData.getSampleStoredTestConsentResource(); - consentResource.setConsentType(null); - - consentCoreServiceImpl.bindUserAccountsToConsent(consentResource, - ConsentMgtServiceTestData.SAMPLE_USER_ID, "authID", - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testBindUserAccountsToConsentWithoutUserID() throws Exception { - - consentCoreServiceImpl.bindUserAccountsToConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - null, "authID", ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testBindUserAccountsToConsentWithoutAuthID() throws Exception { - - consentCoreServiceImpl.bindUserAccountsToConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, null, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testBindUserAccountsToConsentWithoutNewAuthStatus() throws Exception { - - consentCoreServiceImpl.bindUserAccountsToConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, "authID", - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, null, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testBindUserAccountsToConsentWithEmptyAccountsAndPermissionsMap() throws Exception { - - Assert.assertTrue(consentCoreServiceImpl - .bindUserAccountsToConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, "authID", new HashMap<>(), - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testBindUserAccountsToConsentDataRetrieveError() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).searchConsents(Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.any(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyInt(), Mockito.anyInt()); - consentCoreServiceImpl.bindUserAccountsToConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, "authID", - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testBindUserAccountsToConsentDataUpdateError() throws Exception { - - Mockito.doThrow(OBConsentDataUpdationException.class) - .when(mockedConsentCoreDAO).updateAuthorizationUser(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - consentCoreServiceImpl.bindUserAccountsToConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, "authID", - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testBindUserAccountsToConsentDataInsertError() throws Exception { - - Mockito.doThrow(OBConsentDataInsertionException.class) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - consentCoreServiceImpl.bindUserAccountsToConsent(ConsentMgtServiceTestData.getSampleStoredTestConsentResource(), - ConsentMgtServiceTestData.SAMPLE_USER_ID, "authID", - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - } - - @Test - public void testReAuthorizeExistingAuthResources() throws Exception { - - ConsentResource consentResource = ConsentMgtServiceTestData.getSampleStoredTestConsentResource(); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentMappingResource(ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID)) - .when(mockedConsentCoreDAO).storeConsentMappingResource(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.anyObject(), Mockito.anyString()); - Assert.assertTrue(consentCoreServiceImpl - .reAuthorizeExistingAuthResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, consentResource.getCurrentStatus())); - } - - @Test - public void testReAuthorizeExistingAuthResourceAccountsAddScenario() throws Exception { - - ConsentMappingResource consentMappingResource = - ConsentMgtServiceTestData.getSampleTestConsentMappingResource(sampleID); - consentMappingResource.setAccountID("accountID1"); - ArrayList mappingResources = new ArrayList<>(); - mappingResources.add(consentMappingResource); - - DetailedConsentResource detailedConsentResource = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - detailedConsentResource.setConsentMappingResources(mappingResources); - - ConsentResource consentResource = ConsentMgtServiceTestData.getSampleStoredTestConsentResource(); - Mockito.doReturn(detailedConsentResource) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentMappingResource(ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID)) - .when(mockedConsentCoreDAO).storeConsentMappingResource(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.anyObject(), Mockito.anyString()); - Assert.assertTrue(consentCoreServiceImpl - .reAuthorizeExistingAuthResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, consentResource.getCurrentStatus())); - } - - @Test - public void testReAuthorizeExistingAuthResourceNoAccountsRemoveOrAddScenario() throws Exception { - - ConsentResource consentResource = ConsentMgtServiceTestData.getSampleStoredTestConsentResource(); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentMappingResource(ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID)) - .when(mockedConsentCoreDAO).storeConsentMappingResource(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.anyObject(), Mockito.anyString()); - Assert.assertTrue(consentCoreServiceImpl - .reAuthorizeExistingAuthResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP2, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, consentResource.getCurrentStatus())); - } - - @Test - public void testReAuthorizeExistingAuthResourceAccountsRemoveScenario() throws Exception { - - ConsentResource consentResource = ConsentMgtServiceTestData.getSampleStoredTestConsentResource(); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResourceWithMultipleAccountIDs()) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentMappingResource(ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID)) - .when(mockedConsentCoreDAO).storeConsentMappingResource(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.anyObject(), Mockito.anyString()); - Assert.assertTrue(consentCoreServiceImpl - .reAuthorizeExistingAuthResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP3, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, consentResource.getCurrentStatus())); - } - - @Test - public void testReAuthorizeExistingAuthResourcesWithoutMatchingStatuses() throws Exception { - - ConsentResource consentResource = ConsentMgtServiceTestData.getSampleStoredTestConsentResource(); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentMappingResource(ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID)) - .when(mockedConsentCoreDAO).storeConsentMappingResource(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.anyObject(), Mockito.anyString()); - Assert.assertTrue(consentCoreServiceImpl - .reAuthorizeExistingAuthResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, consentResource.getCurrentStatus())); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeExistingAuthResourcesWithoutConsentID() throws Exception { - - Assert.assertTrue(consentCoreServiceImpl.reAuthorizeExistingAuthResource(null, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeExistingAuthResourcesWithoutAuthID() throws Exception { - - Assert.assertTrue(consentCoreServiceImpl - .reAuthorizeExistingAuthResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - null, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeExistingAuthResourcesWithoutUserID() throws Exception { - - Assert.assertTrue(consentCoreServiceImpl - .reAuthorizeExistingAuthResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - sampleID, null, ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeExistingAuthResourcesWithoutCurrentConsentStatus() throws Exception { - - Assert.assertTrue(consentCoreServiceImpl - .reAuthorizeExistingAuthResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - sampleID, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - null, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeExistingAuthResourcesWithoutApplicableConsentStatus() throws Exception { - - Assert.assertTrue(consentCoreServiceImpl - .reAuthorizeExistingAuthResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeExistingAuthResourcesWithoutNewConsentStatus() throws Exception { - - Assert.assertTrue(consentCoreServiceImpl - .reAuthorizeExistingAuthResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, null)); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeExistingAuthResourcesWithoutAccountsAndPermissionsMap() throws Exception { - - Assert.assertTrue(consentCoreServiceImpl - .reAuthorizeExistingAuthResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, ConsentMgtServiceTestData.SAMPLE_USER_ID, - new HashMap<>(), ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeExistingAuthResourcesDataRetrieveError() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - consentCoreServiceImpl.reAuthorizeExistingAuthResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeExistingAuthResourcesDataInsertError() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleConsentMappingResourcesList(ConsentMgtServiceTestData.SAMPLE_CLIENT_IDS_LIST)) - .when(mockedConsentCoreDAO).getConsentMappingResources(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - Mockito.doThrow(OBConsentDataInsertionException.class) - .when(mockedConsentCoreDAO).storeConsentMappingResource(Mockito.any(), Mockito.anyObject()); - consentCoreServiceImpl.reAuthorizeExistingAuthResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeExistingAuthResourcesDataUpdateError() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleConsentMappingResourcesList(ConsentMgtServiceTestData.SAMPLE_CLIENT_IDS_LIST)) - .when(mockedConsentCoreDAO).getConsentMappingResources(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - Mockito.doThrow(OBConsentDataUpdationException.class) - .when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), Mockito.anyObject(), - Mockito.anyString()); - consentCoreServiceImpl.reAuthorizeExistingAuthResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - } - - @Test - public void storeConsentAttributes() throws Exception { - - consentCoreServiceImpl.storeConsentAttributes(ConsentMgtServiceTestData.CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void storeConsentAttributesWithoutParameters() throws Exception { - - consentCoreServiceImpl.storeConsentAttributes(null, null); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void storeConsentAttributesWithoutConsentId() throws Exception { - - consentCoreServiceImpl.storeConsentAttributes(null, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void storeConsentAttributesWithoutAttributeMap() throws Exception { - - consentCoreServiceImpl.storeConsentAttributes(ConsentMgtServiceTestData.CONSENT_ID, - null); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void storeConsentAttributesEmptyAttributeMap() throws Exception { - - consentCoreServiceImpl.storeConsentAttributes(ConsentMgtServiceTestData.CONSENT_ID, - new HashMap<>()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void storeConsentAttributesDataInsertError() throws Exception { - - Mockito.doThrow(OBConsentDataInsertionException.class) - .when(mockedConsentCoreDAO).storeConsentAttributes(Mockito.any(), Mockito.any()); - - consentCoreServiceImpl.storeConsentAttributes(ConsentMgtServiceTestData.CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP); - } - - @Test - public void testGetConsentAttributesWithAttributeKeys() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentAttributesObject(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID)) - .when(mockedConsentCoreDAO).getConsentAttributes(Mockito.any(), Mockito.anyString(), - Mockito.anyObject()); - ConsentAttributes consentAttributes = - consentCoreServiceImpl.getConsentAttributes(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_KEYS); - Assert.assertNotNull(consentAttributes); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentAttributesWithoutConsentID() throws Exception { - - consentCoreServiceImpl.getConsentAttributes(null, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_KEYS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentAttributesDataRetrieveError() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).getConsentAttributes(Mockito.any(), Mockito.anyString()); - consentCoreServiceImpl.getConsentAttributes(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, new ArrayList<>()); - } - - @Test - public void testGetConsentAttributes() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentAttributesObject(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID)) - .when(mockedConsentCoreDAO).getConsentAttributes(Mockito.any(), Mockito.anyString()); - ConsentAttributes consentAttributes = - consentCoreServiceImpl.getConsentAttributes(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID); - Assert.assertNotNull(consentAttributes); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentAttributesWithoutAttributeKeys() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).getConsentAttributes(Mockito.any(), Mockito.anyString()); - consentCoreServiceImpl.getConsentAttributes(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentAttributesWithoutAttributesWithoutConsentID() throws Exception { - - consentCoreServiceImpl.getConsentAttributes(null); - } - - @Test - public void testGetConsentAttributesByName() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP) - .when(mockedConsentCoreDAO).getConsentAttributesByName(Mockito.any(), Mockito.anyString()); - Map retrievedAttributesMap = - consentCoreServiceImpl.getConsentAttributesByName("x-request-id"); - Assert.assertTrue(retrievedAttributesMap.containsKey("x-request-id")); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentAttributesByNameWithoutAttributeName() throws Exception { - - consentCoreServiceImpl.getConsentAttributesByName(null); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentAttributesByNameDataRetrieveError() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).getConsentAttributesByName(Mockito.any(), Mockito.anyString()); - consentCoreServiceImpl.getConsentAttributesByName("x-request-id"); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentIdByConsentAttributeNameAndValueWithoutAttributeName() throws Exception { - - consentCoreServiceImpl.getConsentIdByConsentAttributeNameAndValue(null, - "domestic-payments"); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentIdByConsentAttributeNameAndValueWithoutAttributeValues() throws Exception { - - consentCoreServiceImpl.getConsentIdByConsentAttributeNameAndValue("payment-type", - null); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentIdByConsentAttributeNameAndValueDataRetrieveError() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).getConsentIdByConsentAttributeNameAndValue(Mockito.any(), - Mockito.anyString(), Mockito.anyString()); - consentCoreServiceImpl.getConsentIdByConsentAttributeNameAndValue("payment-type", - "domestic-payments"); - } - - @Test - public void testGetConsentIdByConsentAttributeNameAndValue() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.SAMPLE_CONSENT_IS_ARRAY) - .when(mockedConsentCoreDAO).getConsentIdByConsentAttributeNameAndValue(Mockito.any(), - Mockito.anyString(), Mockito.anyString()); - ArrayList consentIdList = consentCoreServiceImpl.getConsentIdByConsentAttributeNameAndValue( - "payment-type", "domestic-payments"); - Assert.assertFalse(consentIdList.isEmpty()); - } - - @Test - public void testGetConsentFile() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleConsentFileObject(ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT)) - .when(mockedConsentCoreDAO).getConsentFile(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - ConsentFile consentFile = consentCoreServiceImpl - .getConsentFile(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID); - Assert.assertNotNull(consentFile); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentFileWithoutConsentID() throws Exception { - - consentCoreServiceImpl.getConsentFile(null); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentFileDataRetrieveError() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).getConsentFile(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - consentCoreServiceImpl.getConsentFile(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID); - } - - @Test - public void testGetAuthorizationResource() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestAuthorizationResource()) - .when(mockedConsentCoreDAO).getAuthorizationResource(Mockito.any(), Mockito.anyString()); - AuthorizationResource authorizationResource = - consentCoreServiceImpl.getAuthorizationResource(ConsentMgtServiceTestData - .getSampleStoredTestAuthorizationResource().getAuthorizationID()); - Assert.assertNotNull(authorizationResource); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetAuthorizationResourceWithoutAuthID() throws Exception { - - consentCoreServiceImpl.getAuthorizationResource(null); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetAuthorizationResourceDataRetrieveError() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).getAuthorizationResource(Mockito.any(), Mockito.anyString()); - consentCoreServiceImpl.getAuthorizationResource(ConsentMgtServiceTestData - .getSampleStoredTestAuthorizationResource().getAuthorizationID()); - } - - @Test - public void testSearchConsentStatusAuditRecords() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleStoredTestConsentStatusAuditRecordsList(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)) - .when(mockedConsentCoreDAO).getConsentStatusAuditRecords(Mockito.any(), Mockito.anyString(), - Mockito.anyString(), Mockito.anyString(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyString(), - Mockito.anyBoolean()); - ArrayList consentStatusAuditRecords = - consentCoreServiceImpl.searchConsentStatusAuditRecords(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_ACTION_BY, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.SAMPLE_AUDIT_ID); - Assert.assertNotNull(consentStatusAuditRecords); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testSearchConsentStatusAuditRecordsDataRetrieveError() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class).when(mockedConsentCoreDAO) - .getConsentStatusAuditRecords(Mockito.any(), Mockito.anyString(), Mockito.anyString(), - Mockito.anyString(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyString(), - Mockito.anyBoolean()); - consentCoreServiceImpl.searchConsentStatusAuditRecords(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_ACTION_BY, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, ConsentMgtServiceTestData.SAMPLE_AUDIT_ID); - } - - @Test - public void testSearchAuthorizations() throws Exception { - - ArrayList consentIDs = new ArrayList<>(); - consentIDs.add(UUID.randomUUID().toString()); - - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleAuthorizationResourcesList(consentIDs)) - .when(mockedConsentCoreDAO).searchConsentAuthorizations(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - ArrayList retrievedAuthorizations = - consentCoreServiceImpl.searchAuthorizations(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_USER_ID); - Assert.assertNotNull(retrievedAuthorizations); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testSearchAuthorizationsDataRetrieveError() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).searchConsentAuthorizations(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - - consentCoreServiceImpl.searchAuthorizations(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_USER_ID); - } - - @Test - public void testDeleteConsentAttributes() throws Exception { - - Mockito.doReturn(true).when(mockedConsentCoreDAO).deleteConsentAttributes(Mockito.any(), - Mockito.anyString(), Mockito.anyObject()); - consentCoreServiceImpl.deleteConsentAttributes(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_KEYS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testDeleteConsentAttributesDeleteError() throws Exception { - - Mockito.doThrow(OBConsentDataDeletionException.class) - .when(mockedConsentCoreDAO).deleteConsentAttributes(Mockito.any(), Mockito.anyString(), - Mockito.anyObject()); - consentCoreServiceImpl.deleteConsentAttributes(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_KEYS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testDeleteConsentAttributesWithoutConsentID() throws Exception { - - Mockito.doReturn(true).when(mockedConsentCoreDAO).deleteConsentAttributes(Mockito.any(), - Mockito.anyString(), Mockito.anyObject()); - consentCoreServiceImpl.deleteConsentAttributes(null, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_KEYS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testDeleteConsentAttributesWithoutAttributeKeysList() throws Exception { - - Mockito.doReturn(true).when(mockedConsentCoreDAO).deleteConsentAttributes(Mockito.any(), - Mockito.anyString(), Mockito.anyObject()); - consentCoreServiceImpl.deleteConsentAttributes(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - null); - } - - @Test - public void testReAuthorizeConsentWithNewAuthResource() throws Exception { - - AuthorizationResource authorizationResource = ConsentMgtServiceTestData - .getSampleTestAuthorizationResource(sampleID); - ArrayList consentIDs = new ArrayList<>(); - consentIDs.add(sampleID); - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleAuthorizationResourcesList(consentIDs)) - .when(mockedConsentCoreDAO).searchConsentAuthorizations(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - Mockito.doReturn(authorizationResource) - .when(mockedConsentCoreDAO).updateAuthorizationStatus(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - Mockito.doReturn(authorizationResource).when(mockedConsentCoreDAO).storeAuthorizationResource(Mockito.any(), - Mockito.anyObject()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentMappingResource(sampleID)) - .when(mockedConsentCoreDAO).storeConsentMappingResource(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.anyObject(), Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleTestConsentResource()) - .when(mockedConsentCoreDAO).updateConsentStatus(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentStatusAuditRecord(sampleID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - - consentCoreServiceImpl.reAuthorizeConsentWithNewAuthResource(sampleID, sampleID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeConsentWithNewAuthResourceDataRetrieveError() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).searchConsentAuthorizations(Mockito.any(), Mockito.any(), Mockito.any()); - - consentCoreServiceImpl.reAuthorizeConsentWithNewAuthResource(sampleID, sampleID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeConsentWithNewAuthResourceDataUpdateError() throws Exception { - - ArrayList consentIDs = new ArrayList<>(); - consentIDs.add(sampleID); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleAuthorizationResourcesList(consentIDs)) - .when(mockedConsentCoreDAO).searchConsentAuthorizations(Mockito.any(), Mockito.any(), Mockito.any()); - Mockito.doThrow(OBConsentDataUpdationException.class) - .when(mockedConsentCoreDAO).updateAuthorizationStatus(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - - consentCoreServiceImpl.reAuthorizeConsentWithNewAuthResource(sampleID, sampleID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeConsentWithNewAuthResourceDataInsertError() throws Exception { - - AuthorizationResource authorizationResource = ConsentMgtServiceTestData - .getSampleTestAuthorizationResource(sampleID); - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResourcesList()) - .when(mockedConsentCoreDAO).searchConsents(Mockito.any(), Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.any(), Mockito.anyLong(), Mockito.anyLong(), Mockito.anyInt(), Mockito.anyInt()); - Mockito.doReturn(authorizationResource) - .when(mockedConsentCoreDAO).updateAuthorizationStatus(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - Mockito.doReturn(authorizationResource).when(mockedConsentCoreDAO).storeAuthorizationResource(Mockito.any(), - Mockito.anyObject()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - Mockito.doThrow(OBConsentDataInsertionException.class) - .when(mockedConsentCoreDAO).storeConsentMappingResource(Mockito.any(), Mockito.anyObject()); - - consentCoreServiceImpl.reAuthorizeConsentWithNewAuthResource(sampleID, sampleID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeConsentWithNewAuthResourceWithoutConsentID() throws Exception { - - consentCoreServiceImpl.reAuthorizeConsentWithNewAuthResource(null, sampleID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeConsentWithNewAuthResourceWithoutUserID() throws Exception { - - consentCoreServiceImpl.reAuthorizeConsentWithNewAuthResource(sampleID, null, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeConsentWithNewAuthResourceWithoutAccountsMap() throws Exception { - - consentCoreServiceImpl.reAuthorizeConsentWithNewAuthResource(sampleID, sampleID, - null, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeConsentWithNewAuthResourceWithoutCurrentConsentStatus() throws Exception { - - consentCoreServiceImpl.reAuthorizeConsentWithNewAuthResource(sampleID, sampleID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, null, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeConsentWithNewAuthResourceWithoutNewConsentStatus() throws Exception { - - consentCoreServiceImpl.reAuthorizeConsentWithNewAuthResource(sampleID, sampleID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - null, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeConsentWithNewAuthResourceWithoutNewExistingAuthStatus() throws Exception { - - consentCoreServiceImpl.reAuthorizeConsentWithNewAuthResource(sampleID, sampleID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, null, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeConsentWithNewAuthResourceWithoutNewAuthStatus() throws Exception { - - consentCoreServiceImpl.reAuthorizeConsentWithNewAuthResource(sampleID, sampleID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - null, ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testReAuthorizeConsentWithNewAuthResourceWithoutNewAuthType() throws Exception { - - consentCoreServiceImpl.reAuthorizeConsentWithNewAuthResource(sampleID, sampleID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS, null); - } - - private void mockStaticClasses() throws ConsentManagementException, IdentityOAuth2Exception { - - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()).thenReturn(Mockito.mock(Connection.class)); - PowerMockito.when(DatabaseUtil.getRetentionDBConnection()).thenReturn(Mockito.mock(Connection.class)); - - PowerMockito.mockStatic(ConsentStoreInitializer.class); - PowerMockito.when(ConsentStoreInitializer.getInitializedConsentCoreDAOImpl()).thenReturn(mockedConsentCoreDAO); - PowerMockito.when(ConsentStoreInitializer - .getInitializedConsentRetentionDAOImpl()).thenReturn(mockedConsentCoreDAO); - - PowerMockito.mockStatic(ConsentManagementDataHolder.class); - PowerMockito.when(ConsentManagementDataHolder.getInstance()).thenReturn(consentManagementDataHolderMock); - - PowerMockito.when(consentManagementDataHolderMock.getOBEventQueue()).thenReturn(obEventQueueMock); - - AccessTokenDAOImpl accessTokenDAOMock = Mockito.mock(AccessTokenDAOImpl.class); - Mockito.doNothing().when(accessTokenDAOMock).revokeAccessTokens(Mockito.any(String[].class)); - PowerMockito.when(consentManagementDataHolderMock.getAccessTokenDAO()).thenReturn(accessTokenDAOMock); - - OpenBankingConfigParser openBankingConfigParserMock = Mockito.mock(OpenBankingConfigParser.class); - Map configuration = new HashMap<>(); - configuration.put(OpenBankingConstants.CONSENT_ID_CLAIM_NAME, "OB_CONSENT_ID"); - Mockito.when(openBankingConfigParserMock.getConfiguration()).thenReturn(configuration); - Mockito.when(openBankingConfigParserMock.isConsentDataRetentionEnabled()).thenReturn(true); - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - } - - @Test - public void testAmendConsentData() throws Exception { - - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentReceipt(Mockito.any(), - Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentValidityTime(Mockito.any(), - Mockito.anyString(), Mockito.anyLong()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getConsentResource(Mockito.any(), Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentStatusAuditRecord(sampleID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - - ConsentResource consentResource = - consentCoreServiceImpl.amendConsentData(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.SAMPLE_USER_ID); - - Assert.assertNotNull(consentResource); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testAmendConsentDataWithoutConsentID() throws Exception { - - consentCoreServiceImpl.amendConsentData(null, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.SAMPLE_USER_ID); - - } - - @Test - public void testAmendConsentValidityPeriod() throws Exception { - - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentValidityTime(Mockito.any(), - Mockito.anyString(), Mockito.anyLong()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getConsentResource(Mockito.any(), Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentStatusAuditRecord(sampleID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - - ConsentResource consentResource = - consentCoreServiceImpl.amendConsentData(sampleID, null, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.SAMPLE_USER_ID); - - Assert.assertNotNull(consentResource); - } - - @Test - public void testAmendConsentReceipt() throws Exception { - - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentReceipt(Mockito.any(), - Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getConsentResource(Mockito.any(), Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentStatusAuditRecord(sampleID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - - ConsentResource consentResource = - consentCoreServiceImpl.amendConsentData(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - null, ConsentMgtServiceTestData.SAMPLE_USER_ID); - - Assert.assertNotNull(consentResource); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testAmendConsentDataWithoutReceiptAndValidityTime() throws Exception { - - consentCoreServiceImpl.amendConsentData(sampleID, null, null, - ConsentMgtServiceTestData.SAMPLE_USER_ID); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testAmendConsentDataUpdateError() throws Exception { - - Mockito.doThrow(OBConsentDataUpdationException.class) - .when(mockedConsentCoreDAO).updateConsentReceipt(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - - consentCoreServiceImpl.amendConsentData(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.SAMPLE_USER_ID); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testAmendConsentDataRetrieveError() throws Exception { - - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentReceipt(Mockito.any(), - Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentValidityTime(Mockito.any(), - Mockito.anyString(), Mockito.anyLong()); - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).getConsentResource(Mockito.any(), Mockito.anyString()); - - consentCoreServiceImpl.amendConsentData(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.SAMPLE_USER_ID); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testAmendConsentDataInsertError() throws Exception { - - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentReceipt(Mockito.any(), - Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentValidityTime(Mockito.any(), - Mockito.anyString(), Mockito.anyLong()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getConsentResource(Mockito.any(), Mockito.anyString()); - Mockito.doThrow(OBConsentDataInsertionException.class) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - - consentCoreServiceImpl.amendConsentData(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.SAMPLE_USER_ID); - } - - @Test - public void testUpdateConsentStatus() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - - consentCoreServiceImpl.updateConsentStatus(ConsentMgtServiceTestData.CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CONSUMED_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testUpdateConsentStatusDataRetrievalError() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).getConsentResource(Mockito.any(), Mockito.anyObject()); - - consentCoreServiceImpl.updateConsentStatus(ConsentMgtServiceTestData.CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CONSUMED_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testUpdateConsentStatusDataInsertError() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getConsentResource(Mockito.any(), Mockito.anyString()); - Mockito.doThrow(OBConsentDataInsertionException.class) - .when(mockedConsentCoreDAO).updateConsentStatus(Mockito.any(), Mockito.anyObject(), - Mockito.anyObject()); - - consentCoreServiceImpl.updateConsentStatus(ConsentMgtServiceTestData.CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CONSUMED_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testUpdateConsentStatusDataUpdateError() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getConsentResource(Mockito.any(), Mockito.anyString()); - Mockito.doThrow(OBConsentDataUpdationException.class) - .when(mockedConsentCoreDAO).updateConsentStatus(Mockito.any(), Mockito.anyObject(), - Mockito.anyObject()); - - consentCoreServiceImpl.updateConsentStatus(ConsentMgtServiceTestData.CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CONSUMED_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testUpdateConsentStatusWithoutConsentId() throws Exception { - - consentCoreServiceImpl.updateConsentStatus(null, - ConsentMgtServiceTestData.SAMPLE_CONSUMED_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testUpdateConsentStatusWithoutUserId() throws Exception { - - consentCoreServiceImpl.updateConsentStatus(ConsentMgtServiceTestData.CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CONSUMED_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testUpdateConsentStatusWithoutConsentStatus() throws Exception { - - consentCoreServiceImpl.updateConsentStatus(ConsentMgtServiceTestData.CONSENT_ID, - null); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentsEligibleForExpirationDataRetrievalError() throws Exception { - - Mockito.doThrow(new OBConsentDataRetrievalException("Error")) - .when(mockedConsentCoreDAO).getExpiringConsents(Mockito.any(), Mockito.anyString()); - ArrayList consentsEligibleForExpiration = - consentCoreServiceImpl.getConsentsEligibleForExpiration("authorised"); - - Assert.assertTrue(!consentsEligibleForExpiration.isEmpty()); - Assert.assertEquals(consentsEligibleForExpiration.get(0).getConsentID(), - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResourcesList().get(0).getConsentID()); - } - - @Test - public void testGetConsentsEligibleForExpiration() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResourcesList()) - .when(mockedConsentCoreDAO).getExpiringConsents(Mockito.any(), Mockito.anyString()); - ArrayList consentsEligibleForExpiration = - consentCoreServiceImpl.getConsentsEligibleForExpiration("authorised"); - - Assert.assertTrue(!consentsEligibleForExpiration.isEmpty()); - Assert.assertEquals(consentsEligibleForExpiration.get(0).getConsentID(), - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResourcesList().get(0).getConsentID()); - } - - @Test - public void testRevokeConsentWithoutReason() throws Exception { - - DetailedConsentResource retrievedDetailedConsentResource = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - - Mockito.doReturn(retrievedDetailedConsentResource).when(mockedConsentCoreDAO) - .getDetailedConsentResource(Mockito.any(), Mockito.any(), Mockito.anyBoolean()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentResource()).when(mockedConsentCoreDAO) - .updateConsentStatus(Mockito.any(), Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleTestConsentStatusAuditRecord( - retrievedDetailedConsentResource.getConsentID(), - retrievedDetailedConsentResource.getCurrentStatus())) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.any(), Mockito.anyString()); - - boolean isConsentRevoked = new MockConsentCoreServiceImpl() - .revokeConsent(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - - Assert.assertTrue(isConsentRevoked); - } - - @Test (priority = 1) - public void testRevokeConsentWithUserIDWithoutReason() throws Exception { - - DetailedConsentResource retrievedDetailedConsentResource = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - - Mockito.doReturn(retrievedDetailedConsentResource).when(mockedConsentCoreDAO) - .getDetailedConsentResource(Mockito.any(), Mockito.any(), Mockito.anyBoolean()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentResource()).when(mockedConsentCoreDAO) - .updateConsentStatus(Mockito.any(), Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleTestConsentStatusAuditRecord( - retrievedDetailedConsentResource.getConsentID(), - retrievedDetailedConsentResource.getCurrentStatus())) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.any(), Mockito.anyString()); - - boolean isConsentRevoked = new MockConsentCoreServiceImpl() - .revokeConsent(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, ConsentMgtServiceTestData.SAMPLE_USER_ID); - - Assert.assertTrue(isConsentRevoked); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testUpdateAuthorizationStatusWithoutAuthId() throws Exception { - - consentCoreServiceImpl.updateAuthorizationStatus(null, - ConsentMgtServiceTestData.SAMPLE_CONSUMED_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testUpdateAuthorizationStatusWithoutNewAuthStatus() throws Exception { - - consentCoreServiceImpl.updateAuthorizationStatus(ConsentMgtServiceTestData.CONSENT_ID, - null); - } - - @Test - public void testUpdateAuthorizationStatus() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestAuthorizationResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID)) - .when(mockedConsentCoreDAO).updateAuthorizationStatus(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - - consentCoreServiceImpl.updateAuthorizationStatus(ConsentMgtServiceTestData.CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_CONSUMED_STATUS); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testUpdateAuthorizationUserWithoutAuthorizationID() throws Exception { - - consentCoreServiceImpl.updateAuthorizationUser(null, - ConsentMgtServiceTestData.SAMPLE_USER_ID); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testUpdateAuthorizationUserWithoutUserID() throws Exception { - - consentCoreServiceImpl.updateAuthorizationUser(ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_ID_1, - null); - } - - @Test - public void testUpdateAuthorizationUser() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestAuthorizationResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID)) - .when(mockedConsentCoreDAO).updateAuthorizationUser(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - - consentCoreServiceImpl.updateAuthorizationUser(ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_ID_1, - ConsentMgtServiceTestData.SAMPLE_USER_ID); - } - - private void setInitialDataForAmendDetailedConsentSuccessFlow() throws Exception { - - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentReceipt(Mockito.any(), - Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentValidityTime(Mockito.any(), - Mockito.anyString(), Mockito.anyLong()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getConsentResource(Mockito.any(), Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentStatusAuditRecord(sampleID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentMappingResource(ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID)) - .when(mockedConsentCoreDAO).storeConsentMappingResource(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.anyObject(), Mockito.anyString()); - - Mockito.doReturn(true).when(mockedConsentCoreDAO).deleteConsentAttributes(Mockito.any(), - Mockito.anyString(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).storeConsentAttributes(Mockito.any(), - Mockito.anyObject()); - } - - @Test - public void testAmendDetailedConsentData() throws Exception { - - setInitialDataForAmendDetailedConsentSuccessFlow(); - DetailedConsentResource detailedConsentResource = - consentCoreServiceImpl.amendDetailedConsent(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP, - ConsentMgtServiceTestData.SAMPLE_USER_ID, - new HashMap<>()); - - Assert.assertNotNull(detailedConsentResource); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testAmendDetailedConsentDataWithoutConsentID() throws Exception { - - consentCoreServiceImpl.amendDetailedConsent(null, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP, - ConsentMgtServiceTestData.SAMPLE_USER_ID, - new HashMap<>()); - } - - @Test - public void testAmendDetailedConsentDataWithoutReceiptOnly() throws Exception { - - setInitialDataForAmendDetailedConsentSuccessFlow(); - DetailedConsentResource detailedConsentResource = - consentCoreServiceImpl.amendDetailedConsent(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - null, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP, - ConsentMgtServiceTestData.SAMPLE_USER_ID, - new HashMap<>()); - - Assert.assertNotNull(detailedConsentResource); - } - - @Test - public void testAmendDetailedConsentDataWithoutValidityTimeOnly() throws Exception { - - setInitialDataForAmendDetailedConsentSuccessFlow(); - DetailedConsentResource detailedConsentResource = - consentCoreServiceImpl.amendDetailedConsent(sampleID, null, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP, - ConsentMgtServiceTestData.SAMPLE_USER_ID, - new HashMap<>()); - - Assert.assertNotNull(detailedConsentResource); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testAmendDetailedConsentDataWithoutReceiptAndValidityTime() throws Exception { - - consentCoreServiceImpl.amendDetailedConsent(sampleID, null, null, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP, - ConsentMgtServiceTestData.SAMPLE_USER_ID, - new HashMap<>()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testAmendDetailedConsentDataWithoutUserId() throws Exception { - - consentCoreServiceImpl.amendDetailedConsent(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP, null, - new HashMap<>()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testAmendDetailedConsentDataWithoutAuthId() throws Exception { - - consentCoreServiceImpl.amendDetailedConsent(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, null, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP, ConsentMgtServiceTestData.SAMPLE_USER_ID, - new HashMap<>()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testAmendDetailedConsentDataWithoutNewConsentStatus() throws Exception { - - consentCoreServiceImpl.amendDetailedConsent(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, null, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP, ConsentMgtServiceTestData.SAMPLE_USER_ID, - new HashMap<>()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testAmendDetailedConsentDataWithoutNewConsentAttributes() throws Exception { - - consentCoreServiceImpl.amendDetailedConsent(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, null, - ConsentMgtServiceTestData.SAMPLE_USER_ID, - new HashMap<>()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testAmendDetailedConsentDataWithoutAccountIdMapWithPermissions() throws Exception { - - consentCoreServiceImpl.amendDetailedConsent(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, - null, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP, ConsentMgtServiceTestData.SAMPLE_USER_ID, - new HashMap<>()); - } - - @Test - public void testAmendDetailedConsentDataWithAdditionalAmendmentData() throws Exception { - - setInitialDataForAmendDetailedConsentSuccessFlow(); - Mockito.doReturn(new AuthorizationResource()).when(mockedConsentCoreDAO) - .storeAuthorizationResource(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(new ConsentMappingResource()).when(mockedConsentCoreDAO) - .storeConsentMappingResource(Mockito.any(), Mockito.anyObject()); - - DetailedConsentResource detailedConsentResource = - consentCoreServiceImpl.amendDetailedConsent(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP, - ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.getSampleAdditionalConsentAmendmentDataMap()); - - Assert.assertNotNull(detailedConsentResource); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testAmendDetailedConsentDataWithAdditionalAmendmentDataWithoutConsentIdInAuthResources() - throws Exception { - - setInitialDataForAmendDetailedConsentSuccessFlow(); - consentCoreServiceImpl.amendDetailedConsent(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP, - ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.getSampleAdditionalConsentAmendmentDataMapWithoutConsentId()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testAmendDetailedConsentDataWithAdditionalAmendmentDataWithoutAccountIdInMappingResources() - throws Exception { - - setInitialDataForAmendDetailedConsentSuccessFlow(); - consentCoreServiceImpl.amendDetailedConsent(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP, - ConsentMgtServiceTestData.SAMPLE_USER_ID, - ConsentMgtServiceTestData.getSampleAdditionalConsentAmendmentDataMapWithoutAccountId()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testAmendDetailedConsentDataUpdateError() throws Exception { - - Mockito.doThrow(OBConsentDataUpdationException.class) - .when(mockedConsentCoreDAO).updateConsentReceipt(Mockito.any(), Mockito.anyString(), - Mockito.anyString()); - - consentCoreServiceImpl.amendDetailedConsent(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP, - ConsentMgtServiceTestData.SAMPLE_USER_ID, - new HashMap<>()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testAmendDetailedConsentDataRetrieveError() throws Exception { - - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentReceipt(Mockito.any(), - Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentValidityTime(Mockito.any(), - Mockito.anyString(), Mockito.anyLong()); - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - - consentCoreServiceImpl.amendDetailedConsent(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP, - ConsentMgtServiceTestData.SAMPLE_USER_ID, - new HashMap<>()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testAmendDetailedConsentDataInsertError() throws Exception { - - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentReceipt(Mockito.any(), - Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentValidityTime(Mockito.any(), - Mockito.anyString(), Mockito.anyLong()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getConsentResource(Mockito.any(), Mockito.anyString()); - Mockito.doThrow(OBConsentDataInsertionException.class) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - - consentCoreServiceImpl.amendDetailedConsent(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP, - ConsentMgtServiceTestData.SAMPLE_USER_ID, - new HashMap<>()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testAmendDetailedConsentDataDeletionError() throws Exception { - - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentReceipt(Mockito.any(), - Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentValidityTime(Mockito.any(), - Mockito.anyString(), Mockito.anyLong()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getConsentResource(Mockito.any(), Mockito.anyString()); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleStoredTestConsentStatusAuditRecord(sampleID, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS)) - .when(mockedConsentCoreDAO).storeConsentStatusAuditRecord(Mockito.any(), Mockito.anyObject()); - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - Mockito.doReturn(ConsentMgtServiceTestData - .getSampleTestConsentMappingResource(ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID)) - .when(mockedConsentCoreDAO).storeConsentMappingResource(Mockito.any(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).updateConsentMappingStatus(Mockito.any(), - Mockito.anyObject(), Mockito.anyString()); - - Mockito.doThrow(OBConsentDataDeletionException.class).when(mockedConsentCoreDAO) - .deleteConsentAttributes(Mockito.any(), Mockito.anyString(), Mockito.anyObject()); - Mockito.doReturn(true).when(mockedConsentCoreDAO).storeConsentAttributes(Mockito.any(), - Mockito.anyObject()); - consentCoreServiceImpl.amendDetailedConsent(sampleID, ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT, - ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD, - ConsentMgtServiceTestData.UNMATCHED_AUTHORIZATION_ID, - ConsentMgtServiceTestData.SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP, - ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS, - ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP, - ConsentMgtServiceTestData.SAMPLE_USER_ID, - new HashMap<>()); - } - - @Test - public void testStoreConsentAmendmentHistory() throws Exception { - - boolean result = consentCoreServiceImpl.storeConsentAmendmentHistory(sampleID, - ConsentMgtServiceTestData.getSampleTestConsentHistoryResource(), - ConsentMgtServiceTestData.getSampleDetailedStoredTestCurrentConsentResource()); - - Assert.assertNotNull(result); - } - - @Test - public void testStoreConsentAmendmentHistoryWithoutPassingCurrentConsent() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestCurrentConsentResource()) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - - ConsentHistoryResource consentHistoryResource = - new ConsentHistoryResource(); - consentHistoryResource.setTimestamp(ConsentMgtServiceTestData.SAMPLE_CONSENT_AMENDMENT_TIMESTAMP); - consentHistoryResource.setReason(ConsentMgtServiceTestData.SAMPLE_AMENDMENT_REASON); - consentHistoryResource.setDetailedConsentResource(ConsentMgtServiceTestData - .getSampleDetailedStoredTestCurrentConsentResource()); - - boolean result = consentCoreServiceImpl.storeConsentAmendmentHistory( - ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - consentHistoryResource, null); - - Assert.assertNotNull(result); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testStoreConsentAmendmentHistoryWithoutConsentID() throws Exception { - - consentCoreServiceImpl.storeConsentAmendmentHistory(null, - ConsentMgtServiceTestData.getSampleTestConsentHistoryResource(), - ConsentMgtServiceTestData.getSampleDetailedStoredTestCurrentConsentResource()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testStoreConsentAmendmentHistoryWithoutConsentHistoryResource() throws Exception { - - consentCoreServiceImpl.storeConsentAmendmentHistory(sampleID, - null, - ConsentMgtServiceTestData.getSampleDetailedStoredTestCurrentConsentResource()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testStoreConsentAmendmentHistoryWithZeroAsConsentAmendedTimestamp() throws Exception { - - ConsentHistoryResource consentHistoryResource = ConsentMgtServiceTestData.getSampleTestConsentHistoryResource(); - consentHistoryResource.setTimestamp(0); - consentCoreServiceImpl.storeConsentAmendmentHistory(sampleID, consentHistoryResource, - ConsentMgtServiceTestData.getSampleDetailedStoredTestCurrentConsentResource()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testStoreConsentAmendmentHistoryWithoutConsentAmendedReason() throws Exception { - - ConsentHistoryResource consentHistoryResource = ConsentMgtServiceTestData.getSampleTestConsentHistoryResource(); - consentHistoryResource.setReason(null); - consentCoreServiceImpl.storeConsentAmendmentHistory(sampleID, - consentHistoryResource, - ConsentMgtServiceTestData.getSampleDetailedStoredTestCurrentConsentResource()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testStoreConsentAmendmentHistoryDataInsertError() throws Exception { - - Mockito.doThrow(OBConsentDataInsertionException.class) - .when(mockedConsentCoreDAO).storeConsentAmendmentHistory(Mockito.any(), Mockito.anyString(), - Mockito.anyLong(), Mockito.anyString(), Mockito.anyString(), Mockito.anyObject(), Mockito.anyString()); - - consentCoreServiceImpl.storeConsentAmendmentHistory(sampleID, - ConsentMgtServiceTestData.getSampleTestConsentHistoryResource(), - ConsentMgtServiceTestData.getSampleDetailedStoredTestCurrentConsentResource()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testStoreConsentAmendmentHistoryDataRetrievalError() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - - consentCoreServiceImpl.storeConsentAmendmentHistory(sampleID, - ConsentMgtServiceTestData.getSampleTestConsentHistoryResource(), null); - } - - @Test () - public void testGetConsentAmendmentHistoryData() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentHistoryDataMap()) - .when(mockedConsentCoreDAO).retrieveConsentAmendmentHistory(Mockito.any(), any(ArrayList.class)); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - - Map consentAmendmentHistory = - consentCoreServiceImpl.getConsentAmendmentHistoryData(sampleID); - - Assert.assertTrue(consentAmendmentHistory.containsKey(ConsentMgtServiceTestData.SAMPLE_HISTORY_ID)); - Assert.assertNotNull(consentAmendmentHistory.get(ConsentMgtServiceTestData.SAMPLE_HISTORY_ID)); - } - - @Test - public void testGetConsentAmendmentHistoryDataWithOnlyBasicConsentData() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleConsentHistoryBasicConsentDataMap()) - .when(mockedConsentCoreDAO).retrieveConsentAmendmentHistory(Mockito.any(), any(ArrayList.class)); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - - Map consentAmendmentHistory = - consentCoreServiceImpl.getConsentAmendmentHistoryData(sampleID); - - Assert.assertTrue(consentAmendmentHistory.containsKey(ConsentMgtServiceTestData.SAMPLE_HISTORY_ID)); - Assert.assertNotNull(consentAmendmentHistory.get(ConsentMgtServiceTestData.SAMPLE_HISTORY_ID)); - } - - @Test - public void testGetConsentAmendmentHistoryDataWithOnlyConsentAttributesData() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleConsentHistoryConsentAttributesDataMap()) - .when(mockedConsentCoreDAO).retrieveConsentAmendmentHistory(Mockito.any(), any(ArrayList.class)); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - - Map consentAmendmentHistory = - consentCoreServiceImpl.getConsentAmendmentHistoryData(sampleID); - - Assert.assertTrue(consentAmendmentHistory.containsKey(ConsentMgtServiceTestData.SAMPLE_HISTORY_ID)); - Assert.assertNotNull(consentAmendmentHistory.get(ConsentMgtServiceTestData.SAMPLE_HISTORY_ID)); - } - - @Test - public void testGetConsentAmendmentHistoryDataWithOnlyConsentMappingsData() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleConsentHistoryConsentMappingsDataMap()) - .when(mockedConsentCoreDAO).retrieveConsentAmendmentHistory(Mockito.any(), any(ArrayList.class)); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - - Map consentAmendmentHistory = - consentCoreServiceImpl.getConsentAmendmentHistoryData(sampleID); - - Assert.assertTrue(consentAmendmentHistory.containsKey(ConsentMgtServiceTestData.SAMPLE_HISTORY_ID)); - Assert.assertNotNull(consentAmendmentHistory.get(ConsentMgtServiceTestData.SAMPLE_HISTORY_ID)); - } - - @Test - public void testGetConsentAmendmentHistoryDataWithNoConsentHistoryEntries() throws Exception { - - Mockito.doReturn(new HashMap<>()) - .when(mockedConsentCoreDAO).retrieveConsentAmendmentHistory(Mockito.any(), any(ArrayList.class)); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - - Map consentAmendmentHistory = - consentCoreServiceImpl.getConsentAmendmentHistoryData(sampleID); - - Assert.assertEquals(0, consentAmendmentHistory.size()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentAmendmentHistoryDataWithoutConsentID() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentHistoryDataMap()) - .when(mockedConsentCoreDAO).retrieveConsentAmendmentHistory(Mockito.any(), any(ArrayList.class)); - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource()) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - - consentCoreServiceImpl.getConsentAmendmentHistoryData(null); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentAmendmentHistoryDataRetrieveError() throws Exception { - - Mockito.doReturn(ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentHistoryDataMap()) - .when(mockedConsentCoreDAO).retrieveConsentAmendmentHistory(Mockito.any(), any(ArrayList.class)); - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).getDetailedConsentResource(Mockito.any(), Mockito.anyString(), - Mockito.anyBoolean()); - - consentCoreServiceImpl.getConsentAmendmentHistoryData(sampleID); - } - - @Test - public void testRevokeTokensByClientId() throws Exception { - MockConsentCoreServiceImpl mockConsentCoreService = new MockConsentCoreServiceImpl(); - DetailedConsentResource retrievedDetailedConsentResource = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - - mockConsentCoreService.revokeTokens( - retrievedDetailedConsentResource, ConsentMgtServiceTestData.SAMPLE_USER_ID); - } - - @Test - public void testRevokeTokensByClientIdError() { - MockConsentCoreServiceImplTokenError mockedConsentCoreServiceImplTokenError = - new MockConsentCoreServiceImplTokenError(); - DetailedConsentResource retrievedDetailedConsentResource = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - - try { - mockedConsentCoreServiceImplTokenError.revokeTokens( - retrievedDetailedConsentResource, ConsentMgtServiceTestData.SAMPLE_USER_ID); - } catch (Exception e) { - Assert.assertTrue(e instanceof IdentityOAuth2Exception); - } - } - - @Test - public void testSyncRetentionDatabaseWithPurgedConsent() throws Exception { - - DetailedConsentResource detailedConsent = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - - ArrayList consentIds = new ArrayList<>(); - ArrayList consentStatusAuditRecords = new ArrayList<>(); - consentStatusAuditRecords.add(new ConsentStatusAuditRecord()); - consentIds.add(detailedConsent.getConsentID()); - - Mockito.doReturn(consentIds).when(mockedConsentCoreDAO) - .getListOfConsentIds(Mockito.any(), Mockito.anyBoolean()); - Mockito.doReturn(detailedConsent).when(mockedConsentCoreDAO) - .getDetailedConsentResource(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - Mockito.doReturn(null).when(mockedConsentCoreDAO) - .getConsentFile(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - Mockito.doReturn(consentStatusAuditRecords).when(mockedConsentCoreDAO) - .getConsentStatusAuditRecordsByConsentId(Mockito.any(), any(ArrayList.class), Mockito.anyInt(), - Mockito.anyInt(), Mockito.anyBoolean()); - - Mockito.doReturn(new AuthorizationResource()).when(mockedConsentCoreDAO) - .storeAuthorizationResource(Mockito.any(), any(AuthorizationResource.class)); - Mockito.doReturn(true).when(mockedConsentCoreDAO) - .storeConsentAttributes(Mockito.any(), any(ConsentAttributes.class)); - Mockito.doReturn(true).when(mockedConsentCoreDAO) - .storeConsentFile(Mockito.any(), any(ConsentFile.class)); - Mockito.doReturn(new ConsentMappingResource()).when(mockedConsentCoreDAO) - .storeConsentMappingResource(Mockito.any(), any(ConsentMappingResource.class)); - Mockito.doReturn(new ConsentStatusAuditRecord()).when(mockedConsentCoreDAO) - .storeConsentStatusAuditRecord(Mockito.any(), any(ConsentStatusAuditRecord.class)); - Mockito.doReturn(new ConsentResource()).when(mockedConsentCoreDAO) - .storeConsentResource(Mockito.any(), any(ConsentResource.class)); - Mockito.doReturn(true).when(mockedConsentCoreDAO) - .deleteConsentData(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - - Assert.assertTrue(consentCoreServiceImpl.syncRetentionDatabaseWithPurgedConsent()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testSyncRetentionDatabaseWithPurgedConsentRetentionDisabled() throws Exception { - - OpenBankingConfigParser openBankingConfigParserMock = Mockito.mock(OpenBankingConfigParser.class); - Mockito.when(openBankingConfigParserMock.isConsentDataRetentionEnabled()).thenReturn(false); - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - consentCoreServiceImpl.syncRetentionDatabaseWithPurgedConsent(); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testSyncRetentionDatabaseWithPurgedConsentConsentListError() throws Exception { - - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).getListOfConsentIds(Mockito.any(), Mockito.anyBoolean()); - consentCoreServiceImpl.syncRetentionDatabaseWithPurgedConsent(); - } - - @Test - public void testSyncRetentionDatabaseWithPurgedConsentFileGetError() throws Exception { - - DetailedConsentResource detailedConsent = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - - ArrayList consentIds = new ArrayList<>(); - consentIds.add(detailedConsent.getConsentID()); - - Mockito.doReturn(consentIds).when(mockedConsentCoreDAO) - .getListOfConsentIds(Mockito.any(), Mockito.anyBoolean()); - Mockito.doReturn(detailedConsent).when(mockedConsentCoreDAO) - .getDetailedConsentResource(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - Mockito.doThrow(OBConsentDataRetrievalException.class) - .when(mockedConsentCoreDAO).getConsentFile(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - Mockito.doThrow(OBConsentDataRetrievalException.class).when(mockedConsentCoreDAO) - .getConsentStatusAuditRecordsByConsentId(Mockito.any(), any(ArrayList.class), Mockito.anyInt(), - Mockito.anyInt(), Mockito.anyBoolean()); - - Mockito.doReturn(new AuthorizationResource()).when(mockedConsentCoreDAO) - .storeAuthorizationResource(Mockito.any(), any(AuthorizationResource.class)); - Mockito.doReturn(true).when(mockedConsentCoreDAO) - .storeConsentAttributes(Mockito.any(), any(ConsentAttributes.class)); - Mockito.doReturn(true).when(mockedConsentCoreDAO) - .storeConsentFile(Mockito.any(), any(ConsentFile.class)); - Mockito.doReturn(new ConsentMappingResource()).when(mockedConsentCoreDAO) - .storeConsentMappingResource(Mockito.any(), any(ConsentMappingResource.class)); - Mockito.doReturn(new ConsentStatusAuditRecord()).when(mockedConsentCoreDAO) - .storeConsentStatusAuditRecord(Mockito.any(), any(ConsentStatusAuditRecord.class)); - Mockito.doReturn(new ConsentResource()).when(mockedConsentCoreDAO) - .storeConsentResource(Mockito.any(), any(ConsentResource.class)); - Mockito.doReturn(true).when(mockedConsentCoreDAO) - .deleteConsentData(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - Assert.assertTrue(consentCoreServiceImpl.syncRetentionDatabaseWithPurgedConsent()); - } - - @Test - public void testSyncRetentionDatabaseWithPurgedConsentFileStoreError() throws Exception { - - DetailedConsentResource detailedConsent = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - - ArrayList consentIds = new ArrayList<>(); - ArrayList consentStatusAuditRecords = new ArrayList<>(); - consentStatusAuditRecords.add(new ConsentStatusAuditRecord()); - consentIds.add(detailedConsent.getConsentID()); - - Mockito.doReturn(consentIds).when(mockedConsentCoreDAO) - .getListOfConsentIds(Mockito.any(), Mockito.anyBoolean()); - Mockito.doReturn(detailedConsent).when(mockedConsentCoreDAO) - .getDetailedConsentResource(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - Mockito.doReturn(new ConsentFile()).when(mockedConsentCoreDAO) - .getConsentFile(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - Mockito.doReturn(consentStatusAuditRecords).when(mockedConsentCoreDAO) - .getConsentStatusAuditRecordsByConsentId(Mockito.any(), any(ArrayList.class), Mockito.anyInt(), - Mockito.anyInt(), Mockito.anyBoolean()); - - Mockito.doReturn(new ConsentResource()).when(mockedConsentCoreDAO) - .storeConsentResource(Mockito.any(), any(ConsentResource.class)); - Mockito.doReturn(new AuthorizationResource()).when(mockedConsentCoreDAO) - .storeAuthorizationResource(Mockito.any(), any(AuthorizationResource.class)); - Mockito.doReturn(new ConsentMappingResource()).when(mockedConsentCoreDAO) - .storeConsentMappingResource(Mockito.any(), any(ConsentMappingResource.class)); - Mockito.doReturn(true).when(mockedConsentCoreDAO) - .storeConsentAttributes(Mockito.any(), any(ConsentAttributes.class)); - Mockito.doReturn(false).when(mockedConsentCoreDAO) - .storeConsentFile(Mockito.any(), any(ConsentFile.class)); - - Mockito.doReturn(true).when(mockedConsentCoreDAO) - .deleteConsentData(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - Assert.assertTrue(consentCoreServiceImpl.syncRetentionDatabaseWithPurgedConsent()); - } - - @Test - public void testSyncRetentionDatabaseWithPurgedConsentAuditStoreError() throws Exception { - - DetailedConsentResource detailedConsent = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - - ArrayList consentIds = new ArrayList<>(); - ArrayList consentStatusAuditRecords = new ArrayList<>(); - consentStatusAuditRecords.add(new ConsentStatusAuditRecord()); - consentIds.add(detailedConsent.getConsentID()); - - Mockito.doReturn(consentIds).when(mockedConsentCoreDAO) - .getListOfConsentIds(Mockito.any(), Mockito.anyBoolean()); - Mockito.doReturn(detailedConsent).when(mockedConsentCoreDAO) - .getDetailedConsentResource(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - Mockito.doReturn(new ConsentFile()).when(mockedConsentCoreDAO) - .getConsentFile(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - Mockito.doReturn(consentStatusAuditRecords).when(mockedConsentCoreDAO) - .getConsentStatusAuditRecordsByConsentId(Mockito.any(), any(ArrayList.class), Mockito.anyInt(), - Mockito.anyInt(), Mockito.anyBoolean()); - - Mockito.doReturn(new ConsentResource()).when(mockedConsentCoreDAO) - .storeConsentResource(Mockito.any(), any(ConsentResource.class)); - Mockito.doReturn(new AuthorizationResource()).when(mockedConsentCoreDAO) - .storeAuthorizationResource(Mockito.any(), any(AuthorizationResource.class)); - Mockito.doReturn(new ConsentMappingResource()).when(mockedConsentCoreDAO) - .storeConsentMappingResource(Mockito.any(), any(ConsentMappingResource.class)); - Mockito.doReturn(true).when(mockedConsentCoreDAO) - .storeConsentAttributes(Mockito.any(), any(ConsentAttributes.class)); - Mockito.doReturn(true).when(mockedConsentCoreDAO) - .storeConsentFile(Mockito.any(), any(ConsentFile.class)); - Mockito.doReturn(null).when(mockedConsentCoreDAO) - .storeConsentStatusAuditRecord(Mockito.any(), any(ConsentStatusAuditRecord.class)); - - Mockito.doReturn(true).when(mockedConsentCoreDAO) - .deleteConsentData(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - Assert.assertTrue(consentCoreServiceImpl.syncRetentionDatabaseWithPurgedConsent()); - } - - @Test - public void testSyncRetentionDatabaseWithPurgedConsentAttributeStoreError() throws Exception { - - DetailedConsentResource detailedConsent = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - - ArrayList consentIds = new ArrayList<>(); - ArrayList consentStatusAuditRecords = new ArrayList<>(); - consentStatusAuditRecords.add(new ConsentStatusAuditRecord()); - consentIds.add(detailedConsent.getConsentID()); - - Mockito.doReturn(consentIds).when(mockedConsentCoreDAO) - .getListOfConsentIds(Mockito.any(), Mockito.anyBoolean()); - Mockito.doReturn(detailedConsent).when(mockedConsentCoreDAO) - .getDetailedConsentResource(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - Mockito.doReturn(new ConsentFile()).when(mockedConsentCoreDAO) - .getConsentFile(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - Mockito.doReturn(consentStatusAuditRecords).when(mockedConsentCoreDAO) - .getConsentStatusAuditRecordsByConsentId(Mockito.any(), any(ArrayList.class), Mockito.anyInt(), - Mockito.anyInt(), Mockito.anyBoolean()); - - Mockito.doReturn(new ConsentResource()).when(mockedConsentCoreDAO) - .storeConsentResource(Mockito.any(), any(ConsentResource.class)); - Mockito.doReturn(new AuthorizationResource()).when(mockedConsentCoreDAO) - .storeAuthorizationResource(Mockito.any(), any(AuthorizationResource.class)); - Mockito.doReturn(new ConsentMappingResource()).when(mockedConsentCoreDAO) - .storeConsentMappingResource(Mockito.any(), any(ConsentMappingResource.class)); - Mockito.doReturn(false).when(mockedConsentCoreDAO) - .storeConsentAttributes(Mockito.any(), any(ConsentAttributes.class)); - - Assert.assertTrue(consentCoreServiceImpl.syncRetentionDatabaseWithPurgedConsent()); - } - - @Test - public void testSyncRetentionDatabaseWithPurgedConsentMappingStoreError() throws Exception { - - DetailedConsentResource detailedConsent = - ConsentMgtServiceTestData.getSampleDetailedStoredTestConsentResource(); - - ArrayList consentIds = new ArrayList<>(); - ArrayList consentStatusAuditRecords = new ArrayList<>(); - consentStatusAuditRecords.add(new ConsentStatusAuditRecord()); - consentIds.add(detailedConsent.getConsentID()); - - Mockito.doReturn(consentIds).when(mockedConsentCoreDAO) - .getListOfConsentIds(Mockito.any(), Mockito.anyBoolean()); - Mockito.doReturn(detailedConsent).when(mockedConsentCoreDAO) - .getDetailedConsentResource(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - Mockito.doReturn(new ConsentFile()).when(mockedConsentCoreDAO) - .getConsentFile(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - Mockito.doReturn(consentStatusAuditRecords).when(mockedConsentCoreDAO) - .getConsentStatusAuditRecordsByConsentId(Mockito.any(), any(ArrayList.class), Mockito.anyInt(), - Mockito.anyInt(), Mockito.anyBoolean()); - - Mockito.doReturn(new ConsentResource()).when(mockedConsentCoreDAO) - .storeConsentResource(Mockito.any(), any(ConsentResource.class)); - Mockito.doReturn(new AuthorizationResource()).when(mockedConsentCoreDAO) - .storeAuthorizationResource(Mockito.any(), any(AuthorizationResource.class)); - Mockito.doReturn(null).when(mockedConsentCoreDAO) - .storeConsentMappingResource(Mockito.any(), any(ConsentMappingResource.class)); - - Mockito.doReturn(true).when(mockedConsentCoreDAO) - .deleteConsentData(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - Assert.assertTrue(consentCoreServiceImpl.syncRetentionDatabaseWithPurgedConsent()); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentFileError() throws Exception { - OpenBankingConfigParser openBankingConfigParserMock = Mockito.mock(OpenBankingConfigParser.class); - Mockito.when(openBankingConfigParserMock.isConsentDataRetentionEnabled()).thenReturn(false); - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - consentCoreServiceImpl.getConsentFile("3d22259e-942c-46b8-8f75-a608c677a6e6", true); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentFileErrorRetentionData() throws Exception { - consentCoreServiceImpl.getConsentFile("", true); - } - - @Test - public void testGetConsentFileRetentionDataSuccess() throws Exception { - - Mockito.doReturn(new ConsentFile()).when(mockedConsentCoreDAO) - .getConsentFile(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - ConsentFile consentFile = consentCoreServiceImpl.getConsentFile("3d22259e-942c-46b8-8f75-a608c677a6e6", - true); - Assert.assertNotNull(consentFile); - } - - @Test - public void testGetConsentFileConsentData() throws Exception { - - Mockito.doReturn(new ConsentFile()).when(mockedConsentCoreDAO) - .getConsentFile(Mockito.any(), Mockito.anyString(), Mockito.anyBoolean()); - ConsentFile consentFile = consentCoreServiceImpl.getConsentFile("3d22259e-942c-46b8-8f75-a608c677a6e6", - false); - Assert.assertNotNull(consentFile); - } - - @Test - public void testGetConsentStatusAuditRecords() throws Exception { - - ArrayList consentStatusAuditRecords = new ArrayList<>(); - ArrayList consentIds = new ArrayList<>(); - - Mockito.doReturn(consentStatusAuditRecords).when(mockedConsentCoreDAO) - .getConsentStatusAuditRecordsByConsentId(Mockito.any(), any(ArrayList.class), Mockito.anyInt(), - Mockito.anyInt(), Mockito.anyBoolean()); - ArrayList statusAuditRecords = - consentCoreServiceImpl.getConsentStatusAuditRecords(consentIds, null, null, false); - Assert.assertNotNull(statusAuditRecords); - } - - @Test - public void testGetConsentStatusAuditRecordsInRetention() throws Exception { - - ArrayList consentStatusAuditRecords = new ArrayList<>(); - ArrayList consentIds = new ArrayList<>(); - - Mockito.doReturn(consentStatusAuditRecords).when(mockedConsentCoreDAO) - .getConsentStatusAuditRecordsByConsentId(Mockito.any(), any(ArrayList.class), Mockito.anyInt(), - Mockito.anyInt(), Mockito.anyBoolean()); - ArrayList statusAuditRecords = - consentCoreServiceImpl.getConsentStatusAuditRecords(consentIds, null, null, true); - Assert.assertNotNull(statusAuditRecords); - } - - @Test (expectedExceptions = ConsentManagementException.class) - public void testGetConsentStatusAuditRecordsInRetentionError() throws Exception { - ArrayList consentIds = new ArrayList<>(); - OpenBankingConfigParser openBankingConfigParserMock = Mockito.mock(OpenBankingConfigParser.class); - Mockito.when(openBankingConfigParserMock.isConsentDataRetentionEnabled()).thenReturn(false); - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - consentCoreServiceImpl.getConsentStatusAuditRecords(consentIds, null, null, true); - } -} - -class MockConsentCoreServiceImpl extends ConsentCoreServiceImpl { - - @Override - OAuth2Service getOAuth2Service() { - - return Mockito.mock(OAuth2Service.class); - } - - @Override - AuthenticatedUser getAuthenticatedUser(String userID) { - - return Mockito.mock(AuthenticatedUser.class); - } - - @Override - Set getAccessTokenDOSet(DetailedConsentResource detailedConsentResource, - AuthenticatedUser authenticatedUser) { - - String[] scopes = {"OB_CONSENT_ID" + detailedConsentResource.getConsentID()}; - AccessTokenDO sampleAccessTokenDO = new AccessTokenDO(); - sampleAccessTokenDO.setScope(scopes); - sampleAccessTokenDO.setAccessToken("sample_token"); - - Set accessTokenDOS = new HashSet<>(); - accessTokenDOS.add(sampleAccessTokenDO); - - return accessTokenDOS; - } - - @Override - OAuthRevocationResponseDTO revokeTokenByClient(OAuth2Service oAuth2Service, - OAuthRevocationRequestDTO revocationRequestDTO) { - - return Mockito.mock(OAuthRevocationResponseDTO.class); - } -} - -class MockConsentCoreServiceImplTokenError extends ConsentCoreServiceImpl { - - @Override - OAuth2Service getOAuth2Service() { - - return Mockito.mock(OAuth2Service.class); - } - - @Override - AuthenticatedUser getAuthenticatedUser(String userID) { - - return Mockito.mock(AuthenticatedUser.class); - } - - @Override - Set getAccessTokenDOSet(DetailedConsentResource detailedConsentResource, - AuthenticatedUser authenticatedUser) { - - String[] scopes = {"OB_CONSENT_ID" + detailedConsentResource.getConsentID()}; - AccessTokenDO sampleAccessTokenDO = new AccessTokenDO(); - sampleAccessTokenDO.setScope(scopes); - sampleAccessTokenDO.setAccessToken("sample_token"); - - Set accessTokenDOS = new HashSet<>(); - accessTokenDOS.add(sampleAccessTokenDO); - - return accessTokenDOS; - } - - @Override - OAuthRevocationResponseDTO revokeTokenByClient(OAuth2Service oAuth2Service, - OAuthRevocationRequestDTO revocationRequestDTO) { - - OAuthRevocationResponseDTO errorResponse = new OAuthRevocationResponseDTO(); - errorResponse.setError(true); - errorResponse.setErrorMsg("Error occurred while revoking authorization grant for applications"); - return errorResponse; - } -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/service/util/ConsentMgtServiceTestData.java b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/service/util/ConsentMgtServiceTestData.java deleted file mode 100644 index 08a3eab8..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/test/java/com/wso2/openbanking/accelerator/consent/mgt/service/util/ConsentMgtServiceTestData.java +++ /dev/null @@ -1,823 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.mgt.service.util; - -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentAttributes; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentFile; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentHistoryResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentMappingResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentStatusAuditRecord; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.constants.ConsentCoreServiceConstants; -import net.minidev.json.JSONObject; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.LinkedHashMap; -import java.util.Map; -import java.util.UUID; - -/** - * Test data fore consent management service. - */ -public class ConsentMgtServiceTestData { - - public static final String SAMPLE_CONSENT_RECEIPT = "{\"validUntil\": \"2020-10-20\", \"frequencyPerDay\": 1," + - " \"recurringIndicator\": false, \"combinedServiceIndicator\": true}"; - - public static final String SAMPLE_CONSENT_TYPE = "accounts"; - - public static final String SAMPLE_CLIENT_ID = "sampleClientID"; - - public static final int SAMPLE_CONSENT_FREQUENCY = 1; - - public static final Long SAMPLE_CONSENT_VALIDITY_PERIOD = 1638337852L; - - public static final long SAMPLE_CONSENT_AMENDMENT_TIMESTAMP = 1638337852; - - public static final String UNMATCHED_CONSENT_ID = "2222"; - - public static final String UNMATCHED_AUTHORIZATION_ID = "3333"; - - public static final String SAMPLE_MAPPING_ID = "sampleMappingId"; - - public static final String SAMPLE_MAPPING_ID_2 = "sampleMappingId2"; - - public static final String SAMPLE_AUTHORIZATION_ID_1 = "88888"; - - public static final String SAMPLE_AUTHORIZATION_ID_2 = "99999"; - - public static final boolean SAMPLE_RECURRING_INDICATOR = true; - - public static final String SAMPLE_CURRENT_STATUS = "Authorized"; - - public static final String AWAITING_UPLOAD_STATUS = "awaitingUpload"; - - public static final String SAMPLE_PREVIOUS_STATUS = "Received"; - - public static final String SAMPLE_AUTHORIZATION_TYPE = "authorizationType"; - - public static final String CONSENT_ID = "464ef174-9877-4c71-940c-93d6e069eaf9"; - - public static final String SAMPLE_USER_ID = "admin@wso2.com"; - - public static final String SAMPLE_AUDIT_ID = "4321234"; - - public static final String SAMPLE_NEW_USER_ID = "ann@gold.com"; - - public static final String SAMPLE_AUTHORIZATION_STATUS = "awaitingAuthorization"; - - public static final String SAMPLE_ACCOUNT_ID = "123456789"; - - public static final String SAMPLE_MAPPING_STATUS = "active"; - - public static final String SAMPLE_NEW_MAPPING_STATUS = "inactive"; - - public static final String SAMPLE_PERMISSION = "samplePermission"; - - public static final String SAMPLE_REASON = "sample reason"; - - public static final String SAMPLE_ACTION_BY = "admin@wso2.com"; - - public static final String SAMPLE_CONSUMED_STATUS = "Consumed"; - - public static final String SAMPLE_AMENDMENT_REASON = "sampleReason"; - - public static final String SAMPLE_CONSENT_HISTORY_RECEIPT = "{\"validUntil\": \"2020-10-20\", " + - "\"frequencyPerDay\": 5, \"recurringIndicator\": true, \"combinedServiceIndicator\": true}"; - - public static final Long SAMPLE_CONSENT_HISTORY_VALIDITY_PERIOD = 1538337852L; - - public static final long SAMPLE_CONSENT_HISTORY_AMENDMENT_TIMESTAMP = 1538337852; - - public static final String SAMPLE_HISTORY_ID = "sampleHistoryID"; - - public static final Map SAMPLE_CONSENT_ATTRIBUTES_MAP = new HashMap() { - { - put("x-request-id", UUID.randomUUID().toString()); - put("idenpotency-key", UUID.randomUUID().toString()); - put("sampleAttributeKey", "sampleAttributeValue"); - - } - }; - - public static final Map SAMPLE_CONSENT_HISTORY_ATTRIBUTES_MAP = new HashMap() { - { - put("sampleAttributeKey", "sampleAttributeValue"); - put("sampleAttributeKey2", "sampleAttributeValue2"); - put("idenpotency-key", UUID.randomUUID().toString()); - } - }; - - public static final Map> SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP = new HashMap>() { - { - put("accountID1", new ArrayList() { - { - add("permission1"); - add("permission2"); - } - }); - put("accountID2", new ArrayList() { - { - add("permission3"); - add("permission4"); - } - }); - } - }; - - public static final Map> SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP2 = new HashMap>() { - { - put(SAMPLE_ACCOUNT_ID, new ArrayList() { - { - add("permission5"); - add("permission6"); - } - }); - } - }; - - public static final Map> SAMPLE_ACCOUNT_IDS_AND_PERMISSIONS_MAP3 = new HashMap>() { - { - put("mismatching account ID", new ArrayList() { - { - add("permission5"); - add("permission6"); - } - }); - } - }; - - public static final ArrayList SAMPLE_CONSENT_ATTRIBUTES_KEYS = new ArrayList() { - { - add("x-request-id"); - add("idenpotency-key"); - } - }; - - public static final ArrayList UNMATCHED_MAPPING_IDS = new ArrayList() { - { - add("4444"); - add("5555"); - } - }; - - public static final HashMap SAMPLE_MAPPING_ID_PERMISSION_MAP = new HashMap() {{ - put("mapping_id_1", "permission_1"); - put("mapping_id_2", "permission_2"); - }}; - - private static final ArrayList SAMPLE_CONSENT_RECEIPTS_LIST = new ArrayList() { - { - add("{\"element1\": \"value1\"}"); - add("{\"element2\": \"value2\"}"); - add("{\"element3\": \"value3\"}"); - } - }; - - public static final ArrayList SAMPLE_CONSENT_TYPES_LIST = new ArrayList() { - { - add("accounts"); - add("payments"); - add("cof"); - } - }; - - public static final ArrayList SAMPLE_CONSENT_STATUSES_LIST = new ArrayList() { - { - add("created"); - add("authorized"); - add("awaitingAuthorization"); - - } - }; - - public static final ArrayList SAMPLE_CLIENT_IDS_LIST = new ArrayList() { - { - add("clientID1"); - add("clientID2"); - add("clientID3"); - - } - }; - - public static final ArrayList SAMPLE_USER_IDS_LIST = new ArrayList() { - { - add("userID1"); - add("userID2"); - add("userID3"); - } - }; - - private static final ArrayList SAMPLE_VALIDITY_PERIOD_LIST = new ArrayList() { - { - add(1613454661L); - add(1623654661L); - add(1633654671L); - } - }; - - public static final ArrayList SAMPLE_ACCOUNT_ID_LIST = new ArrayList() { - { - add(SAMPLE_ACCOUNT_ID); - } - }; - - public static final ArrayList SAMPLE_CONSENT_IS_ARRAY = new ArrayList() { - { - add(CONSENT_ID); - } - }; - - public static final String SAMPLE_CONSENT_FILE = "sample file content"; - - public static ConsentResource getSampleTestConsentResource() { - - ConsentResource consentResource = new ConsentResource(); - consentResource.setReceipt(ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT); - consentResource.setClientID(UUID.randomUUID().toString()); - consentResource.setConsentType(ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE); - consentResource.setCurrentStatus(ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - consentResource.setConsentFrequency(ConsentMgtServiceTestData.SAMPLE_CONSENT_FREQUENCY); - consentResource.setValidityPeriod(ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD); - consentResource.setRecurringIndicator(ConsentMgtServiceTestData.SAMPLE_RECURRING_INDICATOR); - - return consentResource; - } - - public static ConsentResource getSampleStoredTestConsentResource() { - - ConsentResource consentResource = new ConsentResource(); - consentResource.setConsentID(UUID.randomUUID().toString()); - consentResource.setReceipt(ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT); - consentResource.setClientID(UUID.randomUUID().toString()); - consentResource.setConsentType(ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE); - consentResource.setCurrentStatus(ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - consentResource.setConsentFrequency(ConsentMgtServiceTestData.SAMPLE_CONSENT_FREQUENCY); - consentResource.setValidityPeriod(ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD); - consentResource.setRecurringIndicator(ConsentMgtServiceTestData.SAMPLE_RECURRING_INDICATOR); - - return consentResource; - } - - public static ArrayList getSampleAuthorizationResourcesList(ArrayList consentIDs) { - - ArrayList authorizationResources = new ArrayList<>(); - - for (int i = 0; i < consentIDs.size(); i++) { - for (int j = 0; j < 2; j++) { - AuthorizationResource authorizationResource = new AuthorizationResource(); - authorizationResource.setConsentID(consentIDs.get(i)); - authorizationResource.setAuthorizationType(SAMPLE_AUTHORIZATION_TYPE); - authorizationResource.setUserID(SAMPLE_USER_IDS_LIST.get(i)); - authorizationResource.setAuthorizationStatus(SAMPLE_AUTHORIZATION_STATUS); - authorizationResources.add(authorizationResource); - } - } - return authorizationResources; - } - - public static ArrayList getSampleConsentMappingResourcesList(ArrayList authIDs) { - - ArrayList consentMappingResources = new ArrayList<>(); - - for (int i = 0; i < authIDs.size(); i++) { - for (int j = 0; j < 2; j++) { - ConsentMappingResource consentMappingResource = new ConsentMappingResource(); - consentMappingResource.setAuthorizationID(authIDs.get(i)); - consentMappingResource.setAccountID(SAMPLE_ACCOUNT_ID); - consentMappingResource.setPermission(SAMPLE_PERMISSION); - consentMappingResource.setMappingStatus(SAMPLE_MAPPING_STATUS); - consentMappingResources.add(consentMappingResource); - } - } - return consentMappingResources; - } - - public static DetailedConsentResource getSampleDetailedStoredTestConsentResource() { - - DetailedConsentResource detailedConsentResource = new DetailedConsentResource(); - detailedConsentResource.setConsentID(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID); - detailedConsentResource.setReceipt(ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT); - detailedConsentResource.setClientID(UUID.randomUUID().toString()); - detailedConsentResource.setConsentType(ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE); - detailedConsentResource.setCurrentStatus(ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - detailedConsentResource.setConsentFrequency(ConsentMgtServiceTestData.SAMPLE_CONSENT_FREQUENCY); - detailedConsentResource.setValidityPeriod(ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD); - detailedConsentResource.setRecurringIndicator(ConsentMgtServiceTestData.SAMPLE_RECURRING_INDICATOR); - detailedConsentResource.setCreatedTime(System.currentTimeMillis() / 1000); - detailedConsentResource.setConsentAttributes(ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP); - - ArrayList authorizationResources = new ArrayList<>(); - authorizationResources.add(ConsentMgtServiceTestData - .getSampleTestAuthorizationResource(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_ID_1)); - - ArrayList consentMappingResources = new ArrayList<>(); - consentMappingResources.add(ConsentMgtServiceTestData - .getSampleTestConsentMappingResource(ConsentMgtServiceTestData - .getSampleStoredTestAuthorizationResource().getAuthorizationID())); - consentMappingResources.add(ConsentMgtServiceTestData - .getSampleTestInactiveConsentMappingResource(ConsentMgtServiceTestData - .getSampleStoredTestAuthorizationResource().getAuthorizationID())); - - detailedConsentResource.setAuthorizationResources(authorizationResources); - detailedConsentResource.setConsentMappingResources(consentMappingResources); - - return detailedConsentResource; - } - - public static DetailedConsentResource getSampleDetailedStoredTestConsentResourceWithMultipleAccountIDs() { - - DetailedConsentResource detailedConsentResource = new DetailedConsentResource(); - detailedConsentResource.setConsentID(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID); - detailedConsentResource.setReceipt(ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT); - detailedConsentResource.setClientID(UUID.randomUUID().toString()); - detailedConsentResource.setConsentType(ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE); - detailedConsentResource.setCurrentStatus(ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - detailedConsentResource.setConsentFrequency(ConsentMgtServiceTestData.SAMPLE_CONSENT_FREQUENCY); - detailedConsentResource.setValidityPeriod(ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD); - detailedConsentResource.setRecurringIndicator(ConsentMgtServiceTestData.SAMPLE_RECURRING_INDICATOR); - detailedConsentResource.setCreatedTime(System.currentTimeMillis() / 1000); - detailedConsentResource.setConsentAttributes(ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP); - - ArrayList authorizationResources = new ArrayList<>(); - authorizationResources.add(ConsentMgtServiceTestData - .getSampleStoredTestAuthorizationResource()); - - ConsentMappingResource mappingResource1 = new ConsentMappingResource(); - mappingResource1.setAccountID(UUID.randomUUID().toString()); - - ConsentMappingResource mappingResource2 = new ConsentMappingResource(); - mappingResource2.setAccountID(UUID.randomUUID().toString()); - - ArrayList consentMappingResources = new ArrayList<>(); - consentMappingResources.add(mappingResource1); - consentMappingResources.add(mappingResource2); - - detailedConsentResource.setAuthorizationResources(authorizationResources); - detailedConsentResource.setConsentMappingResources(consentMappingResources); - - return detailedConsentResource; - } - - public static ArrayList getSampleDetailedStoredTestConsentResourcesList() { - - ArrayList detailedConsentResourcesList = new ArrayList<>(); - - for (int i = 0; i < 2; i++) { - DetailedConsentResource detailedConsentResource = new DetailedConsentResource(); - detailedConsentResource.setConsentID(ConsentMgtServiceTestData.UNMATCHED_CONSENT_ID); - detailedConsentResource.setReceipt(ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT); - detailedConsentResource.setClientID(UUID.randomUUID().toString()); - detailedConsentResource.setConsentType(ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE); - detailedConsentResource.setCurrentStatus(ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - detailedConsentResource.setConsentFrequency(ConsentMgtServiceTestData.SAMPLE_CONSENT_FREQUENCY); - detailedConsentResource.setValidityPeriod(ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD); - detailedConsentResource.setRecurringIndicator(ConsentMgtServiceTestData.SAMPLE_RECURRING_INDICATOR); - detailedConsentResource.setCreatedTime(System.currentTimeMillis() / 1000); - detailedConsentResource.setConsentAttributes(ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP); - - ArrayList authorizationResources = new ArrayList<>(); - authorizationResources.add(ConsentMgtServiceTestData - .getSampleStoredTestAuthorizationResource()); - - ArrayList consentMappingResources = new ArrayList<>(); - consentMappingResources.add(ConsentMgtServiceTestData - .getSampleTestConsentMappingResource(ConsentMgtServiceTestData - .getSampleStoredTestAuthorizationResource().getAuthorizationID())); - - detailedConsentResource.setAuthorizationResources(authorizationResources); - detailedConsentResource.setConsentMappingResources(consentMappingResources); - detailedConsentResourcesList.add(detailedConsentResource); - } - return detailedConsentResourcesList; - } - - public static ConsentResource getSampleStoredTestConsentResourceWithAttributes() { - - ConsentResource consentResource = new ConsentResource(); - consentResource.setConsentID(UUID.randomUUID().toString()); - consentResource.setReceipt(ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT); - consentResource.setClientID(UUID.randomUUID().toString()); - consentResource.setConsentType(ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE); - consentResource.setCurrentStatus(ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - consentResource.setConsentFrequency(ConsentMgtServiceTestData.SAMPLE_CONSENT_FREQUENCY); - consentResource.setValidityPeriod(ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD); - consentResource.setRecurringIndicator(ConsentMgtServiceTestData.SAMPLE_RECURRING_INDICATOR); - consentResource.setConsentAttributes(ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP); - - return consentResource; - } - - public static AuthorizationResource getSampleTestAuthorizationResource(String consentID) { - - AuthorizationResource authorizationResource = new AuthorizationResource(); - authorizationResource.setConsentID(consentID); - authorizationResource.setAuthorizationType(ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_TYPE); - authorizationResource.setUserID(ConsentMgtServiceTestData.SAMPLE_USER_ID); - authorizationResource.setAuthorizationStatus(ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS); - - return authorizationResource; - } - - public static AuthorizationResource getSampleTestAuthorizationResource(String consentID, String authorizationId) { - - AuthorizationResource authorizationResource = new AuthorizationResource(); - authorizationResource.setConsentID(consentID); - authorizationResource.setAuthorizationID(authorizationId); - authorizationResource.setAuthorizationType(ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_TYPE); - authorizationResource.setUserID(ConsentMgtServiceTestData.SAMPLE_USER_ID); - authorizationResource.setAuthorizationStatus(ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS); - - return authorizationResource; - } - - public static AuthorizationResource getSampleStoredTestAuthorizationResource() { - - AuthorizationResource authorizationResource = new AuthorizationResource(); - authorizationResource.setConsentID(UUID.randomUUID().toString()); - authorizationResource.setAuthorizationID(UUID.randomUUID().toString()); - authorizationResource.setAuthorizationType(ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_TYPE); - authorizationResource.setUserID(ConsentMgtServiceTestData.SAMPLE_USER_ID); - authorizationResource.setAuthorizationStatus(ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_STATUS); - authorizationResource.setUpdatedTime(System.currentTimeMillis() / 1000); - - return authorizationResource; - } - - public static ConsentMappingResource getSampleTestConsentMappingResource(String authorizationID) { - - ConsentMappingResource consentMappingResource = new ConsentMappingResource(); - consentMappingResource.setMappingID(ConsentMgtServiceTestData.SAMPLE_MAPPING_ID); - consentMappingResource.setAuthorizationID(authorizationID); - consentMappingResource.setAccountID(ConsentMgtServiceTestData.SAMPLE_ACCOUNT_ID); - consentMappingResource.setPermission(ConsentMgtServiceTestData.SAMPLE_PERMISSION); - consentMappingResource.setMappingStatus(ConsentMgtServiceTestData.SAMPLE_MAPPING_STATUS); - - return consentMappingResource; - } - - public static ConsentMappingResource getSampleTestInactiveConsentMappingResource(String authorizationID) { - - ConsentMappingResource consentMappingResource = new ConsentMappingResource(); - consentMappingResource.setMappingID(ConsentMgtServiceTestData.SAMPLE_MAPPING_ID_2); - consentMappingResource.setAuthorizationID(authorizationID); - consentMappingResource.setAccountID(ConsentMgtServiceTestData.SAMPLE_ACCOUNT_ID); - consentMappingResource.setPermission(ConsentMgtServiceTestData.SAMPLE_PERMISSION); - consentMappingResource.setMappingStatus(ConsentMgtServiceTestData.SAMPLE_NEW_MAPPING_STATUS); - - return consentMappingResource; - } - - public static ConsentMappingResource getSampleTestConsentHistoryMappingResource(String authorizationID, - String mappingId) { - - ConsentMappingResource consentMappingResource = new ConsentMappingResource(); - consentMappingResource.setMappingID(mappingId); - consentMappingResource.setAuthorizationID(authorizationID); - consentMappingResource.setAccountID(ConsentMgtServiceTestData.SAMPLE_ACCOUNT_ID); - consentMappingResource.setPermission(ConsentMgtServiceTestData.SAMPLE_PERMISSION); - consentMappingResource.setMappingStatus(ConsentMgtServiceTestData.SAMPLE_NEW_MAPPING_STATUS); - - return consentMappingResource; - } - - public static ConsentMappingResource getSampleStoredTestConsentMappingResource(String authorizationID) { - - ConsentMappingResource consentMappingResource = new ConsentMappingResource(); - consentMappingResource.setMappingID(UUID.randomUUID().toString()); - consentMappingResource.setAuthorizationID(authorizationID); - consentMappingResource.setAccountID(ConsentMgtServiceTestData.SAMPLE_ACCOUNT_ID); - consentMappingResource.setPermission(ConsentMgtServiceTestData.SAMPLE_PERMISSION); - consentMappingResource.setMappingStatus(ConsentMgtServiceTestData.SAMPLE_MAPPING_STATUS); - - return consentMappingResource; - } - - public static ConsentStatusAuditRecord getSampleTestConsentStatusAuditRecord(String consentID, - String currentStatus) { - - ConsentStatusAuditRecord consentStatusAuditRecord = new ConsentStatusAuditRecord(); - consentStatusAuditRecord.setConsentID(consentID); - consentStatusAuditRecord.setCurrentStatus(currentStatus); - consentStatusAuditRecord.setReason(ConsentMgtServiceTestData.SAMPLE_REASON); - consentStatusAuditRecord.setActionBy(ConsentMgtServiceTestData.SAMPLE_ACTION_BY); - consentStatusAuditRecord.setPreviousStatus(ConsentMgtServiceTestData.SAMPLE_PREVIOUS_STATUS); - - return consentStatusAuditRecord; - } - - public static ConsentStatusAuditRecord getSampleStoredTestConsentStatusAuditRecord(String sampleID, - String currentStatus) { - - ConsentStatusAuditRecord consentStatusAuditRecord = new ConsentStatusAuditRecord(); - consentStatusAuditRecord.setConsentID(sampleID); - consentStatusAuditRecord.setStatusAuditID(sampleID); - consentStatusAuditRecord.setCurrentStatus(currentStatus); - consentStatusAuditRecord.setReason(ConsentMgtServiceTestData.SAMPLE_REASON); - consentStatusAuditRecord.setActionBy(ConsentMgtServiceTestData.SAMPLE_ACTION_BY); - consentStatusAuditRecord.setPreviousStatus(ConsentMgtServiceTestData.SAMPLE_PREVIOUS_STATUS); - consentStatusAuditRecord.setActionTime(System.currentTimeMillis() / 1000); - - return consentStatusAuditRecord; - } - - public static ArrayList getSampleStoredTestConsentStatusAuditRecordsList(String sampleID, - String currentStatus) { - - ArrayList consentStatusAuditRecords = new ArrayList<>(); - for (int i = 0; i < 2; i++) { - ConsentStatusAuditRecord consentStatusAuditRecord = new ConsentStatusAuditRecord(); - consentStatusAuditRecord.setConsentID(sampleID); - consentStatusAuditRecord.setStatusAuditID(sampleID); - consentStatusAuditRecord.setCurrentStatus(currentStatus); - consentStatusAuditRecord.setReason(ConsentMgtServiceTestData.SAMPLE_REASON); - consentStatusAuditRecord.setActionBy(ConsentMgtServiceTestData.SAMPLE_ACTION_BY); - consentStatusAuditRecord.setPreviousStatus(ConsentMgtServiceTestData.SAMPLE_PREVIOUS_STATUS); - consentStatusAuditRecord.setActionTime(System.currentTimeMillis() / 1000); - } - return consentStatusAuditRecords; - } - - public static ConsentAttributes getSampleTestConsentAttributesObject(String consentID) { - - ConsentAttributes consentAttributes = new ConsentAttributes(); - consentAttributes.setConsentID(consentID); - consentAttributes.setConsentAttributes(ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP); - - return consentAttributes; - } - - public static ConsentFile getSampleConsentFileObject(String fileContent) { - - ConsentFile consentFile = new ConsentFile(); - consentFile.setConsentID(UUID.randomUUID().toString()); - consentFile.setConsentFile(fileContent); - - return consentFile; - } - - public static ConsentHistoryResource getSampleTestConsentHistoryResource() { - - ConsentHistoryResource consentHistoryResource = new ConsentHistoryResource(); - consentHistoryResource.setTimestamp(ConsentMgtServiceTestData.SAMPLE_CONSENT_AMENDMENT_TIMESTAMP); - consentHistoryResource.setReason(ConsentMgtServiceTestData.SAMPLE_AMENDMENT_REASON); - consentHistoryResource.setDetailedConsentResource(getSampleDetailedConsentHistoryResource()); - - return consentHistoryResource; - } - - public static DetailedConsentResource getSampleDetailedConsentHistoryResource() { - - DetailedConsentResource detailedConsentResource = new DetailedConsentResource(); - detailedConsentResource.setConsentID(ConsentMgtServiceTestData.CONSENT_ID); - detailedConsentResource.setReceipt(ConsentMgtServiceTestData.SAMPLE_CONSENT_HISTORY_RECEIPT); - detailedConsentResource.setClientID(UUID.randomUUID().toString()); - detailedConsentResource.setConsentType(ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE); - detailedConsentResource.setCurrentStatus(ConsentMgtServiceTestData.SAMPLE_PREVIOUS_STATUS); - detailedConsentResource.setConsentFrequency(ConsentMgtServiceTestData.SAMPLE_CONSENT_FREQUENCY); - detailedConsentResource.setValidityPeriod(ConsentMgtServiceTestData.SAMPLE_CONSENT_HISTORY_VALIDITY_PERIOD); - detailedConsentResource.setUpdatedTime(ConsentMgtServiceTestData.SAMPLE_CONSENT_HISTORY_AMENDMENT_TIMESTAMP); - detailedConsentResource.setRecurringIndicator(ConsentMgtServiceTestData.SAMPLE_RECURRING_INDICATOR); - detailedConsentResource.setCreatedTime(System.currentTimeMillis() / 1000); - detailedConsentResource.setConsentAttributes(ConsentMgtServiceTestData.SAMPLE_CONSENT_HISTORY_ATTRIBUTES_MAP); - - ArrayList authorizationResources = new ArrayList<>(); - authorizationResources.add(ConsentMgtServiceTestData.getSampleTestAuthorizationResource( - ConsentMgtServiceTestData.CONSENT_ID, ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_ID_1)); - - ArrayList consentMappingResources = new ArrayList<>(); - consentMappingResources.add(ConsentMgtServiceTestData - .getSampleTestConsentHistoryMappingResource(ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_ID_1, - ConsentMgtServiceTestData.SAMPLE_MAPPING_ID)); - - detailedConsentResource.setAuthorizationResources(authorizationResources); - detailedConsentResource.setConsentMappingResources(consentMappingResources); - - return detailedConsentResource; - } - - public static DetailedConsentResource getSampleDetailedStoredTestCurrentConsentResource() { - - DetailedConsentResource detailedConsentResource = new DetailedConsentResource(); - detailedConsentResource.setConsentID(ConsentMgtServiceTestData.CONSENT_ID); - detailedConsentResource.setReceipt(ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT); - detailedConsentResource.setClientID(UUID.randomUUID().toString()); - detailedConsentResource.setConsentType(ConsentMgtServiceTestData.SAMPLE_CONSENT_TYPE); - detailedConsentResource.setCurrentStatus(ConsentMgtServiceTestData.SAMPLE_CURRENT_STATUS); - detailedConsentResource.setConsentFrequency(ConsentMgtServiceTestData.SAMPLE_CONSENT_FREQUENCY); - detailedConsentResource.setValidityPeriod(ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD); - detailedConsentResource.setUpdatedTime(ConsentMgtServiceTestData.SAMPLE_CONSENT_AMENDMENT_TIMESTAMP); - detailedConsentResource.setRecurringIndicator(ConsentMgtServiceTestData.SAMPLE_RECURRING_INDICATOR); - detailedConsentResource.setCreatedTime(System.currentTimeMillis() / 1000); - detailedConsentResource.setConsentAttributes(ConsentMgtServiceTestData.SAMPLE_CONSENT_ATTRIBUTES_MAP); - - ArrayList authorizationResources = new ArrayList<>(); - authorizationResources.add(ConsentMgtServiceTestData - .getSampleTestAuthorizationResource(ConsentMgtServiceTestData.CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_ID_1)); - authorizationResources.add(ConsentMgtServiceTestData - .getSampleTestAuthorizationResource(ConsentMgtServiceTestData.CONSENT_ID, - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_ID_2)); - - ArrayList consentMappingResources = new ArrayList<>(); - consentMappingResources.add(ConsentMgtServiceTestData - .getSampleTestConsentHistoryMappingResource(ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_ID_1, - ConsentMgtServiceTestData.SAMPLE_MAPPING_ID)); - - // new mapping that is not included in the previous state of the consent - consentMappingResources.add(ConsentMgtServiceTestData.getSampleTestConsentHistoryMappingResource( - ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_ID_1, ConsentMgtServiceTestData.SAMPLE_MAPPING_ID_2)); - - detailedConsentResource.setAuthorizationResources(authorizationResources); - detailedConsentResource.setConsentMappingResources(consentMappingResources); - - return detailedConsentResource; - } - - public static Map getSampleDetailedStoredTestConsentHistoryDataMap() { - - Map consentAmendmentHistoryDataMap = new LinkedHashMap<>(); - - Map changedAttributesJson = new HashMap<>(); - changedAttributesJson.put("ConsentData", getBasicConsentDataChangedAttributesJson()); - changedAttributesJson.put("ConsentAttributesData", getConsentAttributesDataChangedAttributesJson()); - - Map consentAuthResources = new HashMap<>(); - consentAuthResources.put(ConsentMgtServiceTestData.SAMPLE_AUTHORIZATION_ID_1, "null"); - changedAttributesJson.put("ConsentAuthResourceData", consentAuthResources); - - Map consentMappingResources = new HashMap<>(); - JSONObject consentMappingDataJson1 = new JSONObject(); - consentMappingDataJson1.put("MAPPING_STATUS", ConsentMgtServiceTestData.SAMPLE_MAPPING_STATUS); - consentMappingResources.put(ConsentMgtServiceTestData.SAMPLE_MAPPING_ID, consentMappingDataJson1); - - JSONObject consentMappingDataJson2 = new JSONObject(); - consentMappingDataJson2.put("MAPPING_STATUS", ConsentMgtServiceTestData.SAMPLE_NEW_MAPPING_STATUS); - consentMappingResources.put(ConsentMgtServiceTestData.SAMPLE_MAPPING_ID_2, consentMappingDataJson2); - - consentMappingResources.put(UUID.randomUUID().toString(), "null"); - changedAttributesJson.put("ConsentMappingData", consentMappingResources); - - ConsentHistoryResource consentHistoryResource = new ConsentHistoryResource(); - consentHistoryResource.setChangedAttributesJsonDataMap(changedAttributesJson); - consentHistoryResource.setReason("SampleReason"); - consentAmendmentHistoryDataMap.put(ConsentMgtServiceTestData.SAMPLE_HISTORY_ID, consentHistoryResource); - return consentAmendmentHistoryDataMap; - } - - private static String getBasicConsentDataChangedAttributesJson() { - - JSONObject consentBasicDataJson = new JSONObject(); - consentBasicDataJson.put("RECEIPT", ConsentMgtServiceTestData.SAMPLE_CONSENT_RECEIPT); - consentBasicDataJson.put("UPDATED_TIME", - String.valueOf(ConsentMgtServiceTestData.SAMPLE_CONSENT_AMENDMENT_TIMESTAMP)); - consentBasicDataJson.put("VALIDITY_TIME", - String.valueOf(ConsentMgtServiceTestData.SAMPLE_CONSENT_VALIDITY_PERIOD)); - consentBasicDataJson.put("CURRENT_STATUS", ConsentMgtServiceTestData.SAMPLE_PREVIOUS_STATUS); - return consentBasicDataJson.toString(); - } - - private static String getConsentAttributesDataChangedAttributesJson() { - - JSONObject consentAttributesDataJson = new JSONObject(); - consentAttributesDataJson.put("sample_consent_attribute_name", "sample_consent_attribute_value"); - consentAttributesDataJson.put("sampleAttributeKey", null); - return consentAttributesDataJson.toString(); - } - - public static Map getSampleConsentHistoryBasicConsentDataMap() { - - Map consentAmendmentHistoryDataMap = new LinkedHashMap<>(); - Map changedAttributesJson = new HashMap<>(); - changedAttributesJson.put("ConsentData", getBasicConsentDataChangedAttributesJson()); - ConsentHistoryResource consentHistoryResource = new ConsentHistoryResource(); - consentHistoryResource.setChangedAttributesJsonDataMap(changedAttributesJson); - consentHistoryResource.setReason("SampleReason"); - consentAmendmentHistoryDataMap.put(ConsentMgtServiceTestData.SAMPLE_HISTORY_ID, consentHistoryResource); - return consentAmendmentHistoryDataMap; - } - - public static Map getSampleConsentHistoryConsentAttributesDataMap() { - - Map consentAmendmentHistoryDataMap = new LinkedHashMap<>(); - Map changedAttributesJson = new HashMap<>(); - changedAttributesJson.put("ConsentAttributesData", getConsentAttributesDataChangedAttributesJson()); - ConsentHistoryResource consentHistoryResource = new ConsentHistoryResource(); - consentHistoryResource.setChangedAttributesJsonDataMap(changedAttributesJson); - consentHistoryResource.setReason("SampleReason"); - consentAmendmentHistoryDataMap.put(ConsentMgtServiceTestData.SAMPLE_HISTORY_ID, consentHistoryResource); - return consentAmendmentHistoryDataMap; - } - - public static Map getSampleConsentHistoryConsentMappingsDataMap() { - - Map consentAmendmentHistoryDataMap = new LinkedHashMap<>(); - - Map changedAttributesJson = new HashMap<>(); - - Map consentMappingResources = new HashMap<>(); - JSONObject consentMappingDataJson1 = new JSONObject(); - consentMappingDataJson1.put("MAPPING_STATUS", ConsentMgtServiceTestData.SAMPLE_MAPPING_STATUS); - consentMappingResources.put(ConsentMgtServiceTestData.SAMPLE_MAPPING_ID, consentMappingDataJson1); - - JSONObject consentMappingDataJson2 = new JSONObject(); - consentMappingDataJson2.put("MAPPING_STATUS", "null"); - consentMappingResources.put(ConsentMgtServiceTestData.SAMPLE_MAPPING_ID_2, consentMappingDataJson2); - changedAttributesJson.put("ConsentMappingData", consentMappingResources); - - ConsentHistoryResource consentHistoryResource = new ConsentHistoryResource(); - consentHistoryResource.setChangedAttributesJsonDataMap(changedAttributesJson); - consentHistoryResource.setReason("SampleReason"); - consentAmendmentHistoryDataMap.put(ConsentMgtServiceTestData.SAMPLE_HISTORY_ID, consentHistoryResource); - return consentAmendmentHistoryDataMap; - } - - public static Map getSampleAdditionalConsentAmendmentDataMap() { - - Map additionalAmendmentData = new HashMap<>(); - Map newUserAuthResources = new HashMap<>(); - Map> newUserAccountMappings = new HashMap<>(); - - AuthorizationResource newAuthResource = getSampleStoredTestAuthorizationResource(); - newUserAuthResources.put(SAMPLE_NEW_USER_ID, newAuthResource); - - ConsentMappingResource consentMappingResource = getSampleStoredTestConsentMappingResource(null); - ArrayList consentMappingResourceList = new ArrayList(); - consentMappingResourceList.add(consentMappingResource); - newUserAccountMappings.put(SAMPLE_NEW_USER_ID, consentMappingResourceList); - - additionalAmendmentData - .put(ConsentCoreServiceConstants.ADDITIONAL_AUTHORIZATION_RESOURCES, newUserAuthResources); - additionalAmendmentData - .put(ConsentCoreServiceConstants.ADDITIONAL_MAPPING_RESOURCES, newUserAccountMappings); - return additionalAmendmentData; - } - - public static Map getSampleAdditionalConsentAmendmentDataMapWithoutConsentId() { - - Map additionalAmendmentData = new HashMap<>(); - Map newUserAuthResources = new HashMap<>(); - Map> newUserAccountMappings = new HashMap<>(); - - AuthorizationResource newAuthResource = getSampleStoredTestAuthorizationResource(); - newAuthResource.setConsentID(null); - newUserAuthResources.put(SAMPLE_NEW_USER_ID, newAuthResource); - - ConsentMappingResource consentMappingResource = getSampleStoredTestConsentMappingResource(null); - ArrayList consentMappingResourceList = new ArrayList(); - consentMappingResourceList.add(consentMappingResource); - newUserAccountMappings.put(SAMPLE_NEW_USER_ID, consentMappingResourceList); - - additionalAmendmentData - .put(ConsentCoreServiceConstants.ADDITIONAL_AUTHORIZATION_RESOURCES, newUserAuthResources); - additionalAmendmentData - .put(ConsentCoreServiceConstants.ADDITIONAL_MAPPING_RESOURCES, newUserAccountMappings); - return additionalAmendmentData; - } - - public static Map getSampleAdditionalConsentAmendmentDataMapWithoutAccountId() { - - Map additionalAmendmentData = new HashMap<>(); - Map newUserAuthResources = new HashMap<>(); - Map> newUserAccountMappings = new HashMap<>(); - - AuthorizationResource newAuthResource = getSampleStoredTestAuthorizationResource(); - newUserAuthResources.put(SAMPLE_NEW_USER_ID, newAuthResource); - - ConsentMappingResource consentMappingResource = getSampleStoredTestConsentMappingResource(null); - consentMappingResource.setAccountID(null); - ArrayList consentMappingResourceList = new ArrayList(); - consentMappingResourceList.add(consentMappingResource); - newUserAccountMappings.put(SAMPLE_NEW_USER_ID, consentMappingResourceList); - - additionalAmendmentData - .put(ConsentCoreServiceConstants.ADDITIONAL_AUTHORIZATION_RESOURCES, newUserAuthResources); - additionalAmendmentData - .put(ConsentCoreServiceConstants.ADDITIONAL_MAPPING_RESOURCES, newUserAccountMappings); - return additionalAmendmentData; - } - -} diff --git a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/test/resources/testng.xml b/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/test/resources/testng.xml deleted file mode 100644 index 78e08e53..00000000 --- a/open-banking-accelerator/components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service/src/test/resources/testng.xml +++ /dev/null @@ -1,27 +0,0 @@ - - - - - - - - - - - diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/pom.xml b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/pom.xml deleted file mode 100644 index 3d7a58f5..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/pom.xml +++ /dev/null @@ -1,237 +0,0 @@ - - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../../pom.xml - - 4.0.0 - - com.wso2.openbanking.accelerator.event.notifications.service - bundle - WSO2 Open Banking - Event Notification Service Module - - - - org.springframework - spring-web - provided - - - org.apache.cxf - cxf-bundle-jaxrs - provided - - - org.apache.commons - commons-lang3 - provided - - - org.testng - testng - test - - - org.mockito - mockito-all - - - org.powermock - powermock-module-testng - - - org.powermock - powermock-api-mockito - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.identity - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.consent.service - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - provided - - - org.junit.jupiter - junit-jupiter - RELEASE - test - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - - - org.apache.synapse - synapse-core - - - - - - - org.apache.maven.plugins - maven-surefire-plugin - - - src/test/resources/testng.xml - - - - - - org.jacoco - jacoco-maven-plugin - ${jacoco.version} - - - - **/*Constants.class - **/*Component.class - **/*DataHolder.class - src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/**/*SqlStatements.* - **/exceptions/* - - - - - default-prepare-agent - - prepare-agent - - - - default-prepare-agent-integration - - prepare-agent-integration - - - - default-report - - report - - - - default-report-integration - - report-integration - - - - default-check - - check - - - - - BUNDLE - - - INSTRUCTION - COVEREDRATIO - 0.8 - - - - - - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - false - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-exclude.xml - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - org.apache.maven.plugins - maven-compiler-plugin - - 1.8 - 1.8 - UTF-8 - - - - org.apache.felix - maven-bundle-plugin - true - - - - ${project.artifactId} - - - com.wso2.openbanking.accelerator.event.notifications.service.internal - - - org.osgi.framework;version="${osgi.framework.imp.pkg.version.range}", - org.osgi.service.component;version="${osgi.service.component.imp.pkg.version.range}", - com.wso2.openbanking.accelerator.common.*;version="${project.version}", - org.apache.commons.lang3;version="${commons-lang.version}" - com.wso2.openbanking.accelerator.consent.mgt.service.*;version="${project.version}", - - - !com.wso2.openbanking.accelerator.event.notifications.service.internal, - com.wso2.openbanking.accelerator.event.notifications.service.*;version="${project.version}", - - - javax.ws.rs-api;scope=compile;inline=false, - - * - <_dsannotations>* - - - - - - diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/constants/EventNotificationConstants.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/constants/EventNotificationConstants.java deleted file mode 100644 index 835e2248..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/constants/EventNotificationConstants.java +++ /dev/null @@ -1,118 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.constants; - -/** - * Event Notification Constants. - */ -public class EventNotificationConstants { - - //Service level constants - public static final String X_WSO2_CLIENT_ID = "x-wso2-client_id"; - - //Event Notification Status - public static final String ACK = "ACK"; - public static final String ERROR = "ERR"; - public static final String OPEN = "OPEN"; - - //Response Status - public static final String NOT_FOUND = "NOTFOUND"; - public static final String OK = "OK"; - public static final String CREATED = "CREATED"; - public static final String BAD_REQUEST = "BADREQUEST"; - public static final String NO_CONTENT = "NO_CONTENT"; - public static final String INTERNAL_SERVER_ERROR = "INTERNAL_SERVER_ERROR"; - //Database columns - public static final String NOTIFICATION_ID = "NOTIFICATION_ID"; - public static final String CLIENT_ID = "CLIENT_ID"; - public static final String RESOURCE_ID = "RESOURCE_ID"; - public static final String STATUS = "STATUS"; - public static final String UPDATED_TIMESTAMP = "UPDATED_TIMESTAMP"; - public static final String EVENT_INFO = "EVENT_INFO"; - public static final String EVENT_TYPE = "EVENT_TYPE"; - public static final String SUBSCRIPTION_ID = "SUBSCRIPTION_ID"; - public static final String CALLBACK_URL = "CALLBACK_URL"; - public static final String TIME_STAMP = "TIMESTAMP"; - public static final String SPEC_VERSION = "SPEC_VERSION"; - public static final String REQUEST = "REQUEST"; - - //Error Constants - public static final String INVALID_REQUEST = "invalid_request"; - public static final String EVENT_NOTIFICATION_CREATION_ERROR = "Error occurred while saving event " + - "notifications in the database"; - public static final String MISSING_REQ_PAYLOAD = "No request payload found"; - public static final String MISSING_HEADER_PARAM_CLIENT_ID = "Missing header x-wso2-client_id"; - public static final String MISSING_HEADER_PARAM_RESOURCE_ID = "Missing header x-wso2-resource_id"; - public static final String ERROR_IN_EVENT_POLLING_REQUEST = "Error in event polling request"; - public static final String INVALID_CHARS_IN_HEADER_ERROR = "Invalid characters found in the request headers"; - - //Polling request params - public static final String SET_ERRORS = "setErrs"; - public static final String MAX_EVENTS = "maxEvents"; - public static final String DESCRIPTION = "description"; - public static final String RETURN_IMMEDIATELY = "returnImmediately"; - - //Polling response params - public static final String SETS = "sets"; - public static final String MORE_AVAILABLE = "moreAvailable"; - public static final String NOTIFICATIONS_ID = "notificationsID"; - - // Event Subscription Request Params - public static final String SUBSCRIPTION_ID_PARAM = "subscriptionId"; - public static final String CALLBACK_URL_PARAM = "callbackUrl"; - public static final String VERSION_PARAM = "version"; - public static final String EVENT_TYPE_PARAM = "eventTypes"; - public static final String DATA_PARAM = "data"; - - public static final String DB_ERROR_UPDATING = "Database error while updating notification with ID : " + - "'%s' in the database. "; - public static final String DB_ERROR_NOTIFICATION_RETRIEVE = "Error occurred while retrieving" + - " notifications for client ID : '%s'."; - public static final String DB_FAILED_ERROR_NOTIFICATION_STORING = "Failed to store error notification with ID : "; - public static final String DB_ERROR_STORING_ERROR_NOTIFICATION = "Error occurred while closing the " + - "event-notification database connection"; - public static final String DB_ERROR_EVENTS_RETRIEVE = "Error occurred while retrieving events for" + - " notifications ID : '%s'."; - public static final String PARSE_ERROR_NOTIFICATION_ID = "Error occurred while parsing events for" + - " notifications ID : '%s'."; - public static final String DB_CONN_ESTABLISHED = "Database connection is established to get notification " + - "for client ID : '%s' in the database. "; - public static final String RETRIEVED_NOTIFICATION_CLIENT = "Retrieved notification for client ID: '%s'. "; - - public static final String RETRIEVED_EVENTS_NOTIFICATION = "Retrieved events for notification ID: '%s'. "; - public static final String NO_NOTIFICATIONS_FOUND_CLIENT = "No notifications found for client ID - '%s'"; - public static final String NO_EVENTS_NOTIFICATION_ID = "No events found for notification ID - '%s'"; - public static final String INVALID_CLIENT_ID = "Invalid mandatory parameter x-wso2-client-id."; - public static final String DATABASE_CONNECTION_CLOSE_LOG_MSG = "Closing database connection"; - - public static final String ERROR_STORING_EVENT_SUBSCRIPTION = "Error occurred while storing event " + - "subscription in the database. "; - public static final String ERROR_UPDATING_EVENT_SUBSCRIPTION = "Error occurred while updating event " + - "subscription in the database. "; - public static final String ERROR_RETRIEVING_EVENT_SUBSCRIPTION = "Error occurred while retrieving event " + - "subscription in the database. "; - public static final String ERROR_RETRIEVING_EVENT_SUBSCRIPTIONS = "Error occurred while retrieving event " + - "subscriptions in the database."; - public static final String ERROR_DELETING_EVENT_SUBSCRIPTION = "Error occurred while deleting event " + - "subscription in the database. "; - public static final String EVENT_SUBSCRIPTION_NOT_FOUND = "Event subscription not found."; - public static final String EVENT_SUBSCRIPTIONS_NOT_FOUND = "Event subscriptions not found for the given client id."; - public static final String ERROR_HANDLING_EVENT_SUBSCRIPTION = "Error occurred while handling the event " + - "subscription request"; -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/AggregatedPollingDAO.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/AggregatedPollingDAO.java deleted file mode 100644 index 26325d13..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/AggregatedPollingDAO.java +++ /dev/null @@ -1,105 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dao; - -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationError; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationEvent; - -import java.util.List; -import java.util.Map; - -/** - * Aggregated Polling DAO impl. - */ -public interface AggregatedPollingDAO { - - /** - * This method is to update the notification status by ID, allowed values are. - * OPEN,ACK and ERR - * - * @param notificationId Notification ID to update - * @param notificationStatus Notification status to update - * @return Update is success or not - * @throws OBEventNotificationException Exception when updating notification status by ID - */ - Boolean updateNotificationStatusById(String notificationId, String notificationStatus) - throws OBEventNotificationException; - - /** - * This method is to store event notifications error details in the OB_NOTIFICATION table. - * - * @param notificationError Notification error details - * @return Stored event notifications error details - * @throws OBEventNotificationException Exception when storing event notifications error details - */ - Map storeErrorNotification(NotificationError notificationError) - throws OBEventNotificationException; - - /** - * This method is to retrieve given number of notifications in the OB_NOTIFICATION table by client and status. - * - * @param clientId Client ID to retrieve notifications - * @param status Notification status to retrieve - * @param max Maximum number of notifications to retrieve - * @return List of notifications by client and status - * @throws OBEventNotificationException Exception when retrieving notifications by client ID and status - */ - List getNotificationsByClientIdAndStatus(String clientId, String - status, int max) throws OBEventNotificationException; - - /** - * This method is to retrieve notifications by NotificationID. - * - * @param notificationId Notification ID to retrieve - * @return List of notifications by notification ID - * @throws OBEventNotificationException Exception when retrieving notifications by notification ID - */ - List getEventsByNotificationID(String notificationId) throws OBEventNotificationException; - - /** - * This method is to retrieve notifications in the OB_NOTIFICATION table by status. - * - * @param status Notification status to retrieve - * @return List of notifications by status - * @throws OBEventNotificationException Exception when retrieving notifications by status - */ - List getNotificationsByStatus(String status) throws OBEventNotificationException; - - /** - * This method is to retrieve notificationsCount by ClientId and Status. - * - * @param clientId Client ID to retrieve notifications - * @param eventStatus Notification status to retrieve - * @return List of notifications by status and client id - * @throws OBEventNotificationException Exception when retrieving notification count by client ID and status - */ - int getNotificationCountByClientIdAndStatus(String clientId, String eventStatus) - throws OBEventNotificationException; - - /** - * This method is to retrieve the notification status. - * - * @param notificationId Notification ID to retrieve - * @return Notification status by notification ID - * @throws OBEventNotificationException Exception when retrieving notification status - */ - boolean getNotificationStatus(String notificationId) throws OBEventNotificationException; -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/AggregatedPollingDAOImpl.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/AggregatedPollingDAOImpl.java deleted file mode 100644 index a14b74b7..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/AggregatedPollingDAOImpl.java +++ /dev/null @@ -1,448 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dao; - -import com.wso2.openbanking.accelerator.common.util.DatabaseUtil; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationError; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationEvent; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import net.minidev.json.parser.ParseException; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.sql.Savepoint; -import java.sql.Timestamp; -import java.util.ArrayList; -import java.util.Date; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -/** - * Default PollingDAO Impl. - */ -public class AggregatedPollingDAOImpl implements AggregatedPollingDAO { - - private static Log log = LogFactory.getLog(AggregatedPollingDAOImpl.class); - protected NotificationPollingSqlStatements sqlStatements; - - - public AggregatedPollingDAOImpl(NotificationPollingSqlStatements notificationPollingSqlStatements) { - this.sqlStatements = notificationPollingSqlStatements; - } - - @Override - public Boolean updateNotificationStatusById(String notificationId, String notificationStatus) - throws OBEventNotificationException { - - Connection connection = DatabaseUtil.getDBConnection(); - if (log.isDebugEnabled()) { - log.debug(String.format("Database connection is established for updating notification with " + - "ID : '%s' in the database. ", notificationId.replaceAll("[\r\n]", ""))); - } - try { - connection.setAutoCommit(false); - Savepoint savepoint = connection.setSavepoint(); - final String sql = sqlStatements.updateNotificationStatusQueryById(); - try (PreparedStatement updateNotificationStatusById = connection.prepareStatement(sql)) { - Timestamp currentTimeStamp = new Timestamp(new Date().getTime()); - updateNotificationStatusById.setString(1, notificationStatus); - updateNotificationStatusById.setTimestamp(2, currentTimeStamp); - updateNotificationStatusById.setString(3, notificationId); - - int affectedRows = updateNotificationStatusById.executeUpdate(); - - if (affectedRows != 0) { - connection.commit(); - if (log.isDebugEnabled()) { - log.debug(String.format("Updated notification with Notification ID '%s'", - notificationId.replaceAll("[\r\n]", ""))); - } - - return true; - } else { - if (log.isDebugEnabled()) { - log.debug(String.format("Failed updating notification with ID : '%s'", - notificationId.replaceAll("[\r\n]", ""))); - } - return false; - } - } catch (SQLException e) { - connection.rollback(savepoint); - log.error(String.format(EventNotificationConstants.DB_ERROR_UPDATING, - notificationId.replaceAll("[\r\n]", "")), e); - throw new OBEventNotificationException(String.format(EventNotificationConstants.DB_ERROR_UPDATING, - notificationId)); - } - } catch (SQLException e) { - log.debug("SQL exception when updating notification status", e); - throw new OBEventNotificationException("Database error while closing the connection to the" + - " the database."); - } finally { - log.debug(EventNotificationConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public Map storeErrorNotification(NotificationError notificationError) - throws OBEventNotificationException { - - Map response = new HashMap<>(); - Connection connection = DatabaseUtil.getDBConnection(); - - try { - connection.setAutoCommit(false); - - if (log.isDebugEnabled()) { - log.debug(String.format("Database connection is established for storing error notification with ID" + - " : '%s' in the database. ", - notificationError.getNotificationId().replaceAll("[\r\n]", ""))); - } - - final String storeErrorNotificationQuery = sqlStatements.storeErrorNotificationQuery(); - try (PreparedStatement storeErrorNotificationPreparedStatement = - connection.prepareStatement(storeErrorNotificationQuery)) { - - storeErrorNotificationPreparedStatement.setString(1, notificationError. - getNotificationId()); - storeErrorNotificationPreparedStatement.setString(2, notificationError. - getErrorCode()); - storeErrorNotificationPreparedStatement.setString(3, notificationError. - getErrorDescription()); - - int affectedRows = storeErrorNotificationPreparedStatement.executeUpdate(); - if (affectedRows == 1) { - connection.commit(); - if (log.isDebugEnabled()) { - log.debug(String.format("Successfully stored error notification with ID:'%s'.", - notificationError.getNotificationId().replaceAll("[\r\n]", ""))); - } - response.put(notificationError.getNotificationId(), notificationError); - } else { - if (log.isDebugEnabled()) { - log.debug(String.format("Failed store error notification with ID:'%s'.", - notificationError.getNotificationId().replaceAll("[\r\n]", ""))); - } - throw new OBEventNotificationException(EventNotificationConstants. - DB_FAILED_ERROR_NOTIFICATION_STORING + notificationError.getNotificationId()); - } - - } catch (SQLException e) { - connection.rollback(); - throw new OBEventNotificationException(EventNotificationConstants. - DB_ERROR_STORING_ERROR_NOTIFICATION, e); - } - } catch (SQLException e) { - throw new OBEventNotificationException(EventNotificationConstants.DB_ERROR_STORING_ERROR_NOTIFICATION, e); - } finally { - log.debug(EventNotificationConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - - return response; - } - - @Override - public List getNotificationsByClientIdAndStatus(String clientId, String status, int max) - throws OBEventNotificationException { - - List notificationList; - Connection connection = DatabaseUtil.getDBConnection(); - try { - notificationList = new ArrayList<>(); - - if (log.isDebugEnabled()) { - log.debug(String.format(EventNotificationConstants.DB_CONN_ESTABLISHED, - clientId.replaceAll("[\r\n]", ""))); - } - - final String sql = sqlStatements.getMaxNotificationsQuery(); - try (PreparedStatement getNotificationsPreparedStatement = connection.prepareStatement(sql)) { - getNotificationsPreparedStatement.setString(1, clientId); - getNotificationsPreparedStatement.setString(2, status); - getNotificationsPreparedStatement.setInt(3, max); - - try (ResultSet notificationResultSet = getNotificationsPreparedStatement.executeQuery()) { - if (notificationResultSet.next()) { - - //bring pointer back to the top of the result set if not on the top - if (!notificationResultSet.isBeforeFirst()) { - notificationResultSet.beforeFirst(); - } - - //read event notifications from the result set - while (notificationResultSet.next()) { - NotificationDTO notification = new NotificationDTO(); - - notification.setNotificationId(notificationResultSet.getString - (EventNotificationConstants.NOTIFICATION_ID)); - notification.setClientId(notificationResultSet.getString - (EventNotificationConstants.CLIENT_ID)); - notification.setResourceId(notificationResultSet.getString - (EventNotificationConstants.RESOURCE_ID)); - notification.setStatus(notificationResultSet.getString - (EventNotificationConstants.STATUS)); - notification.setUpdatedTimeStamp((notificationResultSet.getTimestamp( - (EventNotificationConstants.UPDATED_TIMESTAMP)).getTime())); - - notificationList.add(notification); - } - notificationResultSet.close(); - getNotificationsPreparedStatement.close(); - - if (log.isDebugEnabled()) { - log.debug(String.format(EventNotificationConstants.RETRIEVED_NOTIFICATION_CLIENT, - clientId.replaceAll("[\r\n]", ""))); - } - } else { - if (log.isDebugEnabled()) { - log.debug(String.format(EventNotificationConstants.NO_NOTIFICATIONS_FOUND_CLIENT, - clientId.replaceAll("[\r\n]", ""))); - } - } - } - } catch (SQLException e) { - throw new OBEventNotificationException(String.format - (EventNotificationConstants.DB_ERROR_NOTIFICATION_RETRIEVE, clientId), e); - } - } finally { - log.debug(EventNotificationConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - - return notificationList; - } - - @Override - public List getEventsByNotificationID(String notificationId) - throws OBEventNotificationException { - - List eventList = new ArrayList<>(); - - Connection connection = DatabaseUtil.getDBConnection(); - try { - - final String sql = sqlStatements.getEventsByNotificationIdQuery(); - - try (PreparedStatement getEventsPreparedStatement = connection.prepareStatement(sql)) { - - getEventsPreparedStatement.setString(1, notificationId); - - try (ResultSet eventsResultSet = getEventsPreparedStatement.executeQuery()) { - if (eventsResultSet.next()) { - - //bring pointer back to the top of the result set if not on the top - if (!eventsResultSet.isBeforeFirst()) { - eventsResultSet.beforeFirst(); - } - - //read event notifications from the result set - while (eventsResultSet.next()) { - NotificationEvent event = new NotificationEvent(); - event.setNotificationId(eventsResultSet.getString - (EventNotificationConstants.NOTIFICATION_ID)); - event.setEventType(eventsResultSet.getString - (EventNotificationConstants.EVENT_TYPE)); - event.setEventInformation(EventNotificationServiceUtil. - getEventJSONFromString(eventsResultSet.getString - (EventNotificationConstants.EVENT_INFO))); - eventList.add(event); - } - eventsResultSet.close(); - getEventsPreparedStatement.close(); - - if (log.isDebugEnabled()) { - log.debug(String.format(EventNotificationConstants.RETRIEVED_EVENTS_NOTIFICATION, - notificationId.replaceAll("[\r\n]", ""))); - } - } else { - if (log.isDebugEnabled()) { - log.debug(String.format(EventNotificationConstants.NO_EVENTS_NOTIFICATION_ID, - notificationId.replaceAll("[\r\n]", ""))); - } - } - } catch (ParseException e) { - log.error(String.format(EventNotificationConstants.PARSE_ERROR_NOTIFICATION_ID, - notificationId.replaceAll("[\r\n]", "")), e); - throw new OBEventNotificationException(String.format ( - EventNotificationConstants.PARSE_ERROR_NOTIFICATION_ID, notificationId), e); - } - } catch (SQLException e) { - log.error(String.format(EventNotificationConstants.DB_ERROR_EVENTS_RETRIEVE, - notificationId.replaceAll("[\r\n]", "")), e); - throw new OBEventNotificationException(String.format - (EventNotificationConstants.DB_ERROR_EVENTS_RETRIEVE, notificationId), e); - } - - } finally { - log.debug(EventNotificationConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - - return eventList; - } - - @Override - public List getNotificationsByStatus(String status) throws OBEventNotificationException { - List notificationList; - Connection connection = DatabaseUtil.getDBConnection(); - try { - notificationList = new ArrayList<>(); - final String sql = sqlStatements.getNotificationsByState(); - try (PreparedStatement getNotificationsPreparedStatement = connection.prepareStatement(sql)) { - getNotificationsPreparedStatement.setString(1, status); - - try (ResultSet notificationResultSet = getNotificationsPreparedStatement.executeQuery()) { - if (notificationResultSet.next()) { - //bring pointer back to the top of the result set if not on the top - if (!notificationResultSet.isBeforeFirst()) { - notificationResultSet.beforeFirst(); - } - //read event notifications from the result set - while (notificationResultSet.next()) { - NotificationDTO notification = new NotificationDTO(); - notification.setNotificationId(notificationResultSet.getString - (EventNotificationConstants.NOTIFICATION_ID)); - notification.setClientId(notificationResultSet.getString - (EventNotificationConstants.CLIENT_ID)); - notification.setResourceId(notificationResultSet.getString - (EventNotificationConstants.RESOURCE_ID)); - notification.setStatus(notificationResultSet.getString - (EventNotificationConstants.STATUS)); - notification.setUpdatedTimeStamp((notificationResultSet.getTimestamp( - (EventNotificationConstants.UPDATED_TIMESTAMP)).getTime())); - notificationList.add(notification); - } - notificationResultSet.close(); - getNotificationsPreparedStatement.close(); - if (log.isDebugEnabled()) { - log.debug( - EventNotificationConstants.RETRIEVED_NOTIFICATION_CLIENT); - } - } else { - if (log.isDebugEnabled()) { - log.debug(EventNotificationConstants.NO_NOTIFICATIONS_FOUND_CLIENT); - } - } - } - } catch (SQLException e) { - throw new OBEventNotificationException(EventNotificationConstants.DB_ERROR_NOTIFICATION_RETRIEVE, e); - } - } finally { - log.debug(EventNotificationConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - - return notificationList; - } - - @Override - public int getNotificationCountByClientIdAndStatus(String clientId, String eventStatus) - throws OBEventNotificationException { - - Connection connection = DatabaseUtil.getDBConnection(); - try { - - final String sql = sqlStatements.getNotificationsCountQuery(); - try (PreparedStatement getNotificationCount = connection.prepareStatement(sql)) { - - getNotificationCount.setString(1, clientId); - getNotificationCount.setString(2, eventStatus); - - try (ResultSet notificationCountResultSet = getNotificationCount.executeQuery()) { - if (notificationCountResultSet.next()) { - - int count = notificationCountResultSet.getInt("NOTIFICATION_COUNT"); - notificationCountResultSet.close(); - getNotificationCount.close(); - - if (log.isDebugEnabled()) { - log.debug(String.format("Retrieved notification count for client ID: '%s'. ", - clientId.replaceAll("[\r\n]", ""))); - } - - return count; - } else { - if (log.isDebugEnabled()) { - log.debug(String.format( - EventNotificationConstants.NO_NOTIFICATIONS_FOUND_CLIENT, - clientId.replaceAll("[\r\n]", ""))); - } - - return 0; - } - } - } catch (SQLException e) { - throw new OBEventNotificationException(String.format - (EventNotificationConstants.DB_ERROR_NOTIFICATION_RETRIEVE, clientId), e); - } - } finally { - log.debug(EventNotificationConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - @Override - public boolean getNotificationStatus(String notificationId) throws OBEventNotificationException { - - boolean isOpenStatus = false; - Connection connection = DatabaseUtil.getDBConnection(); - try { - - final String sql = sqlStatements.getNotificationByNotificationId(); - try (PreparedStatement getNotificationStatus = connection.prepareStatement(sql)) { - getNotificationStatus.setString(1, notificationId); - - try (ResultSet notificationResultSet = getNotificationStatus.executeQuery()) { - if (notificationResultSet.next()) { - - if (EventNotificationConstants.OPEN.equals(notificationResultSet. - getString("STATUS"))) { - isOpenStatus = true; - } - - return isOpenStatus; - } else { - if (log.isDebugEnabled()) { - log.debug(String.format("No notifications found for notification ID - '%s'", - notificationId.replaceAll("[\r\n]", ""))); - } - } - } - } catch (SQLException e) { - throw new OBEventNotificationException(String.format - ("Error occurred while retrieving status for the notifications ID : '%s'.", - notificationId), e); - } - } finally { - log.debug(EventNotificationConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - - return false; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventPublisherDAO.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventPublisherDAO.java deleted file mode 100644 index 0ff62185..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventPublisherDAO.java +++ /dev/null @@ -1,44 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dao; - -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationEvent; - -import java.sql.Connection; -import java.util.ArrayList; - -/** - * Event Publisher DAO interface. - */ -public interface EventPublisherDAO { - - /** - * This method is used to persist event notifications in the database. - * @param connection Database connection - * @param notificationDTO Notification details DTO - * @param eventsList List of notification events - * @return NotificationID of the saved notification. - * @throws OBEventNotificationException Exception when persisting event notification data - */ - String persistEventNotification(Connection connection, NotificationDTO notificationDTO, - ArrayList eventsList) throws OBEventNotificationException; - -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventPublisherDAOImpl.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventPublisherDAOImpl.java deleted file mode 100644 index de7e06f8..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventPublisherDAOImpl.java +++ /dev/null @@ -1,89 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dao; - -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationEvent; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.SQLException; -import java.util.ArrayList; - -/** - * Persisting event notifications to database. - */ -public class EventPublisherDAOImpl implements EventPublisherDAO { - - private static Log log = LogFactory.getLog(EventPublisherDAOImpl.class); - protected NotificationPublisherSqlStatements sqlStatements; - - public EventPublisherDAOImpl(NotificationPublisherSqlStatements notificationPublisherSqlStatements) { - this.sqlStatements = notificationPublisherSqlStatements; - } - - @Override - public String persistEventNotification(Connection connection, NotificationDTO notificationDTO, - ArrayList eventsList) throws OBEventNotificationException { - - int result; - int[] noOfRows; - String persistEventNotification = sqlStatements.getStoreNotification(); - String persistEvents = sqlStatements.getStoreNotificationEvents(); - - try (PreparedStatement persistEventNotificationStmnt = - connection.prepareStatement(persistEventNotification); - PreparedStatement persistEventsStmt = connection.prepareStatement(persistEvents)) { - - log.debug("Setting parameters to prepared statement to add event notification "); - - persistEventNotificationStmnt.setString(1, notificationDTO.getNotificationId()); - persistEventNotificationStmnt.setString(2, notificationDTO.getClientId()); - persistEventNotificationStmnt.setString(3, notificationDTO.getResourceId()); - persistEventNotificationStmnt.setString(4, notificationDTO.getStatus()); - - // with result, we can determine whether the insertion was successful or not - result = persistEventNotificationStmnt.executeUpdate(); - - // to insert notification events - for (NotificationEvent event : eventsList) { - persistEventsStmt.setString(1, notificationDTO.getNotificationId()); - persistEventsStmt.setString(2, event.getEventType()); - persistEventsStmt.setString(3, event.getEventInformation().toString()); - persistEventsStmt.addBatch(); - } - noOfRows = persistEventsStmt.executeBatch(); - } catch (SQLException e) { - log.error(EventNotificationConstants.EVENT_NOTIFICATION_CREATION_ERROR, e); - throw new OBEventNotificationException(EventNotificationConstants. - EVENT_NOTIFICATION_CREATION_ERROR, e); - } - // Confirm that the data are updated successfully - if (result > 0 && noOfRows.length != 0) { - log.info("Created the event notification successfully"); - return notificationDTO.getNotificationId(); - } else { - throw new OBEventNotificationException("Failed to create the event notification."); - } - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventSubscriptionDAO.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventSubscriptionDAO.java deleted file mode 100644 index dc41501e..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventSubscriptionDAO.java +++ /dev/null @@ -1,120 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dao; - -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.EventSubscription; - -import java.sql.Connection; -import java.util.List; - -/** - * Event Notification Subscription DAO interface. - */ -public interface EventSubscriptionDAO { - - /** - * This method is used to store event notification subscription in the database. - * - * @param connection Database connection. - * @param eventSubscription EventSubscription object. - * @return EventSubscription object. - * @throws OBEventNotificationException Exception when storing event subscription - */ - EventSubscription storeEventSubscription(Connection connection, EventSubscription eventSubscription) - throws OBEventNotificationException; - - /** - * This method is used to store subscribed event types in the database. - * - * @param connection Database connection. - * @param subscriptionId Subscription ID. - * @param eventTypes Event types to be stored. - * @return List of strings with subscribed event types. - * @throws OBEventNotificationException Exception when storing subscribed event types - */ - List storeSubscribedEventTypes(Connection connection, String subscriptionId, List eventTypes) - throws OBEventNotificationException; - - /** - * This method is used to retrieve an event subscription by a subscription ID. - * - * @param connection Database connection. - * @param subscriptionId Subscription ID. - * @return EventSubscription model. - * @throws OBEventNotificationException Exception when retrieving event subscription by subscription ID - */ - EventSubscription getEventSubscriptionBySubscriptionId(Connection connection, String subscriptionId) - throws OBEventNotificationException; - - /** - * This method is used to retrieve all event subscriptions a client. - * - * @param connection Database connection. - * @param clientId Client ID. - * @return List of EventSubscription models. - * @throws OBEventNotificationException Exception when retrieving event subscriptions by client ID - */ - List getEventSubscriptionsByClientId(Connection connection, String clientId) - throws OBEventNotificationException; - - /** - * This method is used to retrieve all event subscriptions by event type. - * - * @param connection Database connection. - * @param eventType Event type that need to be subscribed by the retrieving subscriptions. - * @return List of EventSubscription models. - * @throws OBEventNotificationException Exception when retrieving event subscriptions by event type - */ - List getEventSubscriptionsByEventType(Connection connection, String eventType) - throws OBEventNotificationException; - - /** - * This method is used to update an event subscription. - * - * @param connection Database connection. - * @param eventSubscription eventSubscription object. - * @return true if update was successful. - * @throws OBEventNotificationException Exception when updating event subscription - */ - Boolean updateEventSubscription(Connection connection, EventSubscription eventSubscription) - throws OBEventNotificationException; - - /** - * This method is used to delete an event subscription. - * - * @param connection Database connection. - * @param subscriptionId Subscription ID. - * @return true if deletion was successful. - * @throws OBEventNotificationException Exception when deleting event subscription - */ - Boolean deleteEventSubscription(Connection connection, String subscriptionId) throws OBEventNotificationException; - - /** - * This method is used to delete subscribed event types of a subscription. - * - * @param connection Database connection. - * @param subscriptionId subscription ID. - * @return true if deletion was successful. - * @throws OBEventNotificationException Exception when deleting subscribed event types - */ - Boolean deleteSubscribedEventTypes(Connection connection, String subscriptionId) - throws OBEventNotificationException; - -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventSubscriptionDAOImpl.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventSubscriptionDAOImpl.java deleted file mode 100644 index 7f86e8fa..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventSubscriptionDAOImpl.java +++ /dev/null @@ -1,309 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dao; - -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.EventSubscription; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.time.Instant; -import java.util.ArrayList; -import java.util.List; -import java.util.UUID; - -import static java.sql.Statement.EXECUTE_FAILED; - -/** - * Default EventSubscriptionDAO Impl. - */ -public class EventSubscriptionDAOImpl implements EventSubscriptionDAO { - private static Log log = LogFactory.getLog(EventSubscriptionDAOImpl.class); - - protected EventSubscriptionSqlStatements sqlStatements; - - public EventSubscriptionDAOImpl(EventSubscriptionSqlStatements sqlStatements) { - this.sqlStatements = sqlStatements; - } - - public EventSubscription storeEventSubscription(Connection connection, EventSubscription eventSubscription) - throws OBEventNotificationException { - - int storeSubscriptionAffectedRows; - - UUID subscriptionId = UUID.randomUUID(); - long unixTime = Instant.now().getEpochSecond(); - eventSubscription.setSubscriptionId(subscriptionId.toString()); - eventSubscription.setTimeStamp(unixTime); - eventSubscription.setStatus(EventNotificationConstants.CREATED); - - final String sql = sqlStatements.storeEventSubscriptionQuery(); - try (PreparedStatement storeEventSubscriptionStatement = connection.prepareStatement(sql)) { - storeEventSubscriptionStatement.setString(1, eventSubscription.getSubscriptionId()); - storeEventSubscriptionStatement.setString(2, eventSubscription.getClientId()); - storeEventSubscriptionStatement.setString(3, eventSubscription.getCallbackUrl()); - storeEventSubscriptionStatement.setLong(4, eventSubscription.getTimeStamp()); - storeEventSubscriptionStatement.setString(5, eventSubscription.getSpecVersion()); - storeEventSubscriptionStatement.setString(6, eventSubscription.getStatus()); - storeEventSubscriptionStatement.setString(7, eventSubscription.getRequestData()); - storeSubscriptionAffectedRows = storeEventSubscriptionStatement.executeUpdate(); - if (storeSubscriptionAffectedRows == 0) { - log.error("Failed to store the event notification subscription."); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_STORING_EVENT_SUBSCRIPTION); - } - } catch (SQLException e) { - log.error("SQL exception when storing the event types of the subscription", e); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_STORING_EVENT_SUBSCRIPTION); - } - return eventSubscription; - } - - @Override - public List storeSubscribedEventTypes(Connection connection, String subscriptionId, List eventTypes) - throws OBEventNotificationException { - - final String sql = sqlStatements.storeSubscribedEventTypesQuery(); - try (PreparedStatement storeSubscribedEventTypesStatement = connection.prepareStatement(sql)) { - for (String eventType : eventTypes) { - storeSubscribedEventTypesStatement.setString(1, subscriptionId); - storeSubscribedEventTypesStatement.setString(2, eventType); - storeSubscribedEventTypesStatement.addBatch(); - } - int[] storeSubscribedEventTypesAffectedRows = storeSubscribedEventTypesStatement.executeBatch(); - for (int affectedRows : storeSubscribedEventTypesAffectedRows) { - if (affectedRows == 0 || affectedRows == EXECUTE_FAILED) { - log.error("Failed to store the subscribed event types."); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_STORING_EVENT_SUBSCRIPTION); - } - } - } catch (SQLException e) { - log.error("SQL exception when storing the subscribed event types.", e); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_STORING_EVENT_SUBSCRIPTION); - } - log.debug("Stored the subscribed event types successfully."); - return eventTypes; - - } - - @Override - public EventSubscription getEventSubscriptionBySubscriptionId(Connection connection, String subscriptionId) - throws OBEventNotificationException { - EventSubscription retrievedSubscription = new EventSubscription(); - List eventTypes = new ArrayList<>(); - - final String sql = sqlStatements.getEventSubscriptionBySubscriptionIdQuery(); - try (PreparedStatement getEventSubscriptionBySubscriptionIdStatement = connection.prepareStatement(sql)) { - getEventSubscriptionBySubscriptionIdStatement.setString(1, subscriptionId); - try (ResultSet resultSet = getEventSubscriptionBySubscriptionIdStatement.executeQuery()) { - if (resultSet.next()) { - mapResultSetToEventSubscription(retrievedSubscription, resultSet); - resultSet.beforeFirst(); // Reset the cursor position to the beginning of the result set. - while (resultSet.next()) { - String eventType = resultSet.getString(EventNotificationConstants.EVENT_TYPE); - if (eventType != null) { - eventTypes.add(eventType); - } - } - if (!eventTypes.isEmpty()) { - retrievedSubscription.setEventTypes(eventTypes); - } - } else { - log.error("No event notification subscription found for the given subscription id."); - throw new OBEventNotificationException( - EventNotificationConstants.EVENT_SUBSCRIPTION_NOT_FOUND); - } - } catch (SQLException e) { - log.error("SQL exception when retrieving the event notification subscription.", e); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_RETRIEVING_EVENT_SUBSCRIPTION); - } - } catch (SQLException e) { - log.error("SQL exception when retrieving the event notification subscription.", e); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_RETRIEVING_EVENT_SUBSCRIPTION); - } - return retrievedSubscription; - } - - @Override - public List getEventSubscriptionsByClientId(Connection connection, String clientId) - throws OBEventNotificationException { - List retrievedSubscriptions = new ArrayList<>(); - - final String sql = sqlStatements.getEventSubscriptionsByClientIdQuery(); - try (PreparedStatement getEventSubscriptionsByClientIdStatement = connection.prepareStatement(sql)) { - getEventSubscriptionsByClientIdStatement.setString(1, clientId); - try (ResultSet resultSet = getEventSubscriptionsByClientIdStatement.executeQuery()) { - if (resultSet.isBeforeFirst()) { - while (resultSet.next()) { - EventSubscription eventSubscription = new EventSubscription(); - List eventTypes = new ArrayList<>(); - mapResultSetToEventSubscription(eventSubscription, resultSet); - resultSet.previous(); - while (resultSet.next()) { - if (eventSubscription.getSubscriptionId().equals(resultSet. - getString(EventNotificationConstants.SUBSCRIPTION_ID))) { - if (resultSet.getString(EventNotificationConstants.EVENT_TYPE) != null) { - eventTypes.add(resultSet.getString(EventNotificationConstants.EVENT_TYPE)); - } - } else { - resultSet.previous(); - break; - } - } - if (!eventTypes.isEmpty()) { - eventSubscription.setEventTypes(eventTypes); - } - retrievedSubscriptions.add(eventSubscription); - } - log.debug("Retrieved the event notification subscriptions successfully."); - } - return retrievedSubscriptions; - } catch (SQLException e) { - log.error("SQL exception when retrieving the event notification subscriptions.", e); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_RETRIEVING_EVENT_SUBSCRIPTION); - } - } catch (SQLException e) { - log.error("SQL exception when retrieving the event notification subscriptions.", e); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_RETRIEVING_EVENT_SUBSCRIPTIONS); - } - } - - @Override - public List getEventSubscriptionsByEventType(Connection connection, String eventType) - throws OBEventNotificationException { - List retrievedSubscriptions = new ArrayList<>(); - - final String sql = sqlStatements.getEventSubscriptionsByEventTypeQuery(); - try (PreparedStatement getEventSubscriptionsByClientIdAndEventTypeStatement = - connection.prepareStatement(sql)) { - getEventSubscriptionsByClientIdAndEventTypeStatement.setString(1, eventType); - try (ResultSet resultSet = getEventSubscriptionsByClientIdAndEventTypeStatement.executeQuery()) { - if (resultSet.isBeforeFirst()) { - while (resultSet.next()) { - EventSubscription eventSubscription = new EventSubscription(); - List eventTypes = new ArrayList<>(); - mapResultSetToEventSubscription(eventSubscription, resultSet); - resultSet.previous(); - while (resultSet.next()) { - if (eventSubscription.getSubscriptionId().equals(resultSet. - getString(EventNotificationConstants.SUBSCRIPTION_ID))) { - if (resultSet.getString(EventNotificationConstants.EVENT_TYPE) != null) { - eventTypes.add(resultSet.getString(EventNotificationConstants.EVENT_TYPE)); - } - } else { - resultSet.previous(); - break; - } - } - if (!eventTypes.isEmpty()) { - eventSubscription.setEventTypes(eventTypes); - } - retrievedSubscriptions.add(eventSubscription); - } - log.debug("Retrieved the event notification subscriptions successfully."); - } - return retrievedSubscriptions; - } catch (SQLException e) { - log.error("SQL exception when retrieving the event notification subscriptions.", e); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_RETRIEVING_EVENT_SUBSCRIPTION); - } - } catch (SQLException e) { - log.error("SQL exception when retrieving the event notification subscriptions.", e); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_RETRIEVING_EVENT_SUBSCRIPTIONS); - } - } - - @Override - public Boolean updateEventSubscription(Connection connection, EventSubscription eventSubscription) - throws OBEventNotificationException { - boolean isUpdated = false; - final String sql = sqlStatements.updateEventSubscriptionQuery(); - try (PreparedStatement updateEventSubscriptionStatement = connection.prepareStatement(sql)) { - updateEventSubscriptionStatement.setString(1, eventSubscription.getCallbackUrl()); - updateEventSubscriptionStatement.setLong(2, Instant.now().getEpochSecond()); - updateEventSubscriptionStatement.setString(3, eventSubscription.getRequestData()); - updateEventSubscriptionStatement.setString(4, eventSubscription.getSubscriptionId()); - int affectedRows = updateEventSubscriptionStatement.executeUpdate(); - if (affectedRows > 0) { - log.debug("Event notification subscription is successfully updated."); - isUpdated = true; - } - } catch (SQLException e) { - log.error("SQL exception when updating event notification subscription", e); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_UPDATING_EVENT_SUBSCRIPTION); - } - return isUpdated; - } - - @Override - public Boolean deleteEventSubscription(Connection connection, String subscriptionId) - throws OBEventNotificationException { - - final String sql = sqlStatements.updateEventSubscriptionStatusQuery(); - try (PreparedStatement deleteEventSubscriptionStatement = connection.prepareStatement(sql)) { - deleteEventSubscriptionStatement.setString(1, "DELETED"); - deleteEventSubscriptionStatement.setString(2, subscriptionId); - int affectedRows = deleteEventSubscriptionStatement.executeUpdate(); - if (affectedRows == 0) { - log.debug("Failed deleting event notification subscription."); - return false; - } - log.debug("Event notification subscription is successfully deleted from the database."); - return true; - } catch (SQLException e) { - log.error("SQL exception when deleting event notification subscription data.", e); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_DELETING_EVENT_SUBSCRIPTION); - } - } - - @Override - public Boolean deleteSubscribedEventTypes(Connection connection, String subscriptionId) - throws OBEventNotificationException { - boolean isDeleted = false; - int affectedRowsCount; - final String deleteEventTypesQuery = sqlStatements.deleteSubscribedEventTypesQuery(); - try (PreparedStatement deleteEventTypesStatement = connection.prepareStatement(deleteEventTypesQuery)) { - deleteEventTypesStatement.setString(1, subscriptionId); - affectedRowsCount = deleteEventTypesStatement.executeUpdate(); - if (affectedRowsCount > 0) { - log.debug("Successfully deleted the subscribed event types"); - isDeleted = true; - } - } catch (SQLException e) { - log.error("SQL exception when deleting subscribed event types. ", e); - throw new OBEventNotificationException( - "Error occurred while deleting the event notification subscription."); - } - return isDeleted; - } - - private void mapResultSetToEventSubscription(EventSubscription response, ResultSet resultSet) throws SQLException { - response.setSubscriptionId(resultSet.getString(EventNotificationConstants.SUBSCRIPTION_ID)); - response.setClientId(resultSet.getString(EventNotificationConstants.CLIENT_ID)); - response.setCallbackUrl(resultSet.getString(EventNotificationConstants.CALLBACK_URL)); - response.setTimeStamp(resultSet.getLong(EventNotificationConstants.TIME_STAMP)); - response.setSpecVersion(resultSet.getString(EventNotificationConstants.SPEC_VERSION)); - response.setStatus(resultSet.getString(EventNotificationConstants.STATUS)); - response.setRequestData(resultSet.getString(EventNotificationConstants.REQUEST)); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventSubscriptionSqlStatements.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventSubscriptionSqlStatements.java deleted file mode 100644 index ee06d675..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventSubscriptionSqlStatements.java +++ /dev/null @@ -1,71 +0,0 @@ -/** - * Copyright (c) 2023-2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dao; - -/** - * SQL queries to store, retrieve, update and delete event notification subscriptions. - */ -public class EventSubscriptionSqlStatements { - - public String storeEventSubscriptionQuery() { - return "INSERT INTO OB_NOTIFICATION_SUBSCRIPTION (SUBSCRIPTION_ID, CLIENT_ID, CALLBACK_URL, TIMESTAMP, " + - "SPEC_VERSION, STATUS, REQUEST) VALUES (?,?,?,?,?,?,?)"; - } - - public String storeSubscribedEventTypesQuery() { - return "INSERT INTO OB_NOTIFICATION_SUBSCRIBED_EVENTS (SUBSCRIPTION_ID, EVENT_TYPE) VALUES (?,?)"; - } - - public String getEventSubscriptionBySubscriptionIdQuery() { - return "SELECT ns.SUBSCRIPTION_ID, ns.CLIENT_ID, ns.REQUEST, ns.CALLBACK_URL, ns.TIMESTAMP, ns.SPEC_VERSION, " + - "ns.STATUS, nse.EVENT_TYPE FROM OB_NOTIFICATION_SUBSCRIPTION ns LEFT JOIN " + - "OB_NOTIFICATION_SUBSCRIBED_EVENTS nse ON ns.SUBSCRIPTION_ID = nse.SUBSCRIPTION_ID WHERE " + - "ns.SUBSCRIPTION_ID = ? AND ns.STATUS = 'CREATED'"; - } - - public String getEventSubscriptionsByClientIdQuery() { - return "SELECT ns.SUBSCRIPTION_ID, ns.CLIENT_ID, ns.REQUEST, ns.CALLBACK_URL, ns.TIMESTAMP, ns.SPEC_VERSION, " + - "ns.STATUS, nse.EVENT_TYPE FROM OB_NOTIFICATION_SUBSCRIPTION ns LEFT JOIN " + - "OB_NOTIFICATION_SUBSCRIBED_EVENTS nse ON ns.SUBSCRIPTION_ID = nse.SUBSCRIPTION_ID WHERE " + - "ns.CLIENT_ID = ? AND ns.STATUS = 'CREATED'"; - } - - public String getEventSubscriptionsByEventTypeQuery() { - return "SELECT ns.SUBSCRIPTION_ID, ns.CLIENT_ID, ns.REQUEST, ns.CALLBACK_URL, ns.TIMESTAMP, ns.SPEC_VERSION, " + - "ns.STATUS, nse.EVENT_TYPE FROM OB_NOTIFICATION_SUBSCRIPTION ns LEFT JOIN " + - "OB_NOTIFICATION_SUBSCRIBED_EVENTS nse ON ns.SUBSCRIPTION_ID = nse.SUBSCRIPTION_ID WHERE " + - "ns.SUBSCRIPTION_ID IN (SELECT ns.SUBSCRIPTION_ID FROM OB_NOTIFICATION_SUBSCRIPTION ns LEFT " + - "JOIN OB_NOTIFICATION_SUBSCRIBED_EVENTS nse ON ns.SUBSCRIPTION_ID = nse.SUBSCRIPTION_ID WHERE " + - "nse.EVENT_TYPE = ? AND ns.STATUS = 'CREATED')"; - } - - public String updateEventSubscriptionQuery() { - return "UPDATE OB_NOTIFICATION_SUBSCRIPTION SET CALLBACK_URL = ?, TIMESTAMP = ?, REQUEST = ?" + - "WHERE SUBSCRIPTION_ID = ?"; - } - - public String updateEventSubscriptionStatusQuery() { - return "UPDATE OB_NOTIFICATION_SUBSCRIPTION SET STATUS = ? WHERE SUBSCRIPTION_ID = ? AND STATUS = 'CREATED'"; - } - - public String deleteSubscribedEventTypesQuery() { - return "DELETE FROM OB_NOTIFICATION_SUBSCRIBED_EVENTS WHERE SUBSCRIPTION_ID = ?"; - } - -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/MSSQLNotificationPollingSqlStatements.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/MSSQLNotificationPollingSqlStatements.java deleted file mode 100644 index e1c879d2..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/MSSQLNotificationPollingSqlStatements.java +++ /dev/null @@ -1,35 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dao; - -import com.wso2.openbanking.accelerator.common.util.Generated; - -/** - * MSSQL Queries for Event Polling. - */ -@Generated(message = "Returns sql statements to the dao method") -public class MSSQLNotificationPollingSqlStatements extends NotificationPollingSqlStatements { - - @Override - public String getMaxNotificationsQuery() { - - return "SELECT * FROM OB_NOTIFICATION WHERE CLIENT_ID = ? AND STATUS = ? ORDER BY NOTIFICATION_ID " + - "OFFSET 0 ROWS FETCH NEXT ? ROWS ONLY"; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/NotificationPollingSqlStatements.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/NotificationPollingSqlStatements.java deleted file mode 100644 index 9b1e553c..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/NotificationPollingSqlStatements.java +++ /dev/null @@ -1,60 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dao; - -/** - * SQL queries to store and retrieve event notifications. - */ -public class NotificationPollingSqlStatements { - - public String getEventsByNotificationIdQuery() { - - return "SELECT * FROM OB_NOTIFICATION_EVENT WHERE NOTIFICATION_ID = ?"; - } - - public String getMaxNotificationsQuery() { - - return "SELECT * FROM OB_NOTIFICATION WHERE CLIENT_ID = ? AND STATUS = ? LIMIT ?"; - } - - public String getNotificationsCountQuery() { - - return "SELECT COUNT(*) AS NOTIFICATION_COUNT FROM OB_NOTIFICATION WHERE CLIENT_ID = ? AND STATUS = ?"; - } - - public String storeErrorNotificationQuery() { - - return "INSERT INTO OB_NOTIFICATION_ERROR (NOTIFICATION_ID, ERROR_CODE, DESCRIPTION) VALUES (?,?,?)"; - } - - public String updateNotificationStatusQueryById() { - - return "UPDATE OB_NOTIFICATION SET STATUS = ?, UPDATED_TIMESTAMP= ? WHERE NOTIFICATION_ID = ?"; - } - - public String getNotificationByNotificationId() { - - return "SELECT NOTIFICATION_ID, STATUS FROM OB_NOTIFICATION WHERE NOTIFICATION_ID = ?"; - } - - public String getNotificationsByState() { - - return "SELECT * FROM OB_NOTIFICATION WHERE STATUS = ?"; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/NotificationPublisherSqlStatements.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/NotificationPublisherSqlStatements.java deleted file mode 100644 index 292dde5c..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/NotificationPublisherSqlStatements.java +++ /dev/null @@ -1,40 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dao; - -/** - * SQL statements to store event notifications. - */ -public class NotificationPublisherSqlStatements { - - public String getStoreNotification() { - - final String storeNotifications = "INSERT INTO OB_NOTIFICATION (NOTIFICATION_ID, CLIENT_ID, " + - "RESOURCE_ID, STATUS) VALUES (?,?,?,?)"; - return storeNotifications; - } - - public String getStoreNotificationEvents() { - - final String storeNotificationEvents = - "INSERT INTO OB_NOTIFICATION_EVENT (NOTIFICATION_ID, EVENT_TYPE, EVENT_INFO) VALUES (?,?,?)"; - return storeNotificationEvents; - } - -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/PostgreSqlEventSubscriptionDAOImpl.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/PostgreSqlEventSubscriptionDAOImpl.java deleted file mode 100644 index e3d56cff..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/PostgreSqlEventSubscriptionDAOImpl.java +++ /dev/null @@ -1,197 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dao; - -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.EventSubscription; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.time.Instant; -import java.util.ArrayList; -import java.util.List; -import java.util.UUID; - -/** - * Postgres SQL EventSubscriptionDAO Impl. - */ -public class PostgreSqlEventSubscriptionDAOImpl extends EventSubscriptionDAOImpl { - - private static final Log log = LogFactory.getLog(PostgreSqlEventSubscriptionDAOImpl.class); - - public PostgreSqlEventSubscriptionDAOImpl(EventSubscriptionSqlStatements sqlStatements) { - super(sqlStatements); - } - - @Override - public EventSubscription storeEventSubscription(Connection connection, EventSubscription eventSubscription) - throws OBEventNotificationException { - - int storeSubscriptionAffectedRows; - - UUID subscriptionId = UUID.randomUUID(); - long unixTime = Instant.now().getEpochSecond(); - eventSubscription.setSubscriptionId(subscriptionId.toString()); - eventSubscription.setTimeStamp(unixTime); - eventSubscription.setStatus(EventNotificationConstants.CREATED); - - final String sql = sqlStatements.storeEventSubscriptionQuery(); - try (PreparedStatement storeEventSubscriptionStatement = connection.prepareStatement(sql)) { - storeEventSubscriptionStatement.setString(1, eventSubscription.getSubscriptionId()); - storeEventSubscriptionStatement.setString(2, eventSubscription.getClientId()); - storeEventSubscriptionStatement.setString(3, eventSubscription.getCallbackUrl()); - storeEventSubscriptionStatement.setLong(4, eventSubscription.getTimeStamp()); - storeEventSubscriptionStatement.setString(5, eventSubscription.getSpecVersion()); - storeEventSubscriptionStatement.setString(6, eventSubscription.getStatus()); - storeEventSubscriptionStatement.setObject(7, eventSubscription.getRequestData(), - java.sql.Types.OTHER); - storeSubscriptionAffectedRows = storeEventSubscriptionStatement.executeUpdate(); - if (storeSubscriptionAffectedRows == 0) { - log.error("Failed to store the event notification subscription."); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_STORING_EVENT_SUBSCRIPTION); - } - } catch (SQLException e) { - log.error("SQL exception when storing the event types of the subscription", e); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_STORING_EVENT_SUBSCRIPTION); - } - return eventSubscription; - } - - @Override - public List getEventSubscriptionsByClientId(Connection connection, String clientId) - throws OBEventNotificationException { - List retrievedSubscriptions = new ArrayList<>(); - - final String sql = sqlStatements.getEventSubscriptionsByClientIdQuery(); - try (PreparedStatement getEventSubscriptionsByClientIdStatement = connection.prepareStatement(sql, - ResultSet.TYPE_SCROLL_SENSITIVE, ResultSet.CONCUR_UPDATABLE)) { - getEventSubscriptionsByClientIdStatement.setString(1, clientId); - try (ResultSet resultSet = getEventSubscriptionsByClientIdStatement.executeQuery()) { - if (resultSet.isBeforeFirst()) { - while (resultSet.next()) { - EventSubscription eventSubscription = new EventSubscription(); - List eventTypes = new ArrayList<>(); - mapResultSetToEventSubscription(eventSubscription, resultSet); - resultSet.previous(); - while (resultSet.next()) { - if (eventSubscription.getSubscriptionId().equals(resultSet. - getString(EventNotificationConstants.SUBSCRIPTION_ID))) { - if (resultSet.getString(EventNotificationConstants.EVENT_TYPE) != null) { - eventTypes.add(resultSet.getString(EventNotificationConstants.EVENT_TYPE)); - } - } else { - resultSet.previous(); - break; - } - } - if (!eventTypes.isEmpty()) { - eventSubscription.setEventTypes(eventTypes); - } - retrievedSubscriptions.add(eventSubscription); - } - log.debug("Retrieved the event notification subscriptions successfully."); - } - return retrievedSubscriptions; - } catch (SQLException e) { - log.error("SQL exception when retrieving the event notification subscriptions.", e); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_RETRIEVING_EVENT_SUBSCRIPTION); - } - } catch (SQLException e) { - log.error("SQL exception when retrieving the event notification subscriptions.", e); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_RETRIEVING_EVENT_SUBSCRIPTIONS); - } - } - - @Override - public EventSubscription getEventSubscriptionBySubscriptionId(Connection connection, String subscriptionId) - throws OBEventNotificationException { - EventSubscription retrievedSubscription = new EventSubscription(); - List eventTypes = new ArrayList<>(); - - final String sql = sqlStatements.getEventSubscriptionBySubscriptionIdQuery(); - try (PreparedStatement getEventSubscriptionBySubscriptionIdStatement = connection.prepareStatement(sql, - ResultSet.TYPE_SCROLL_SENSITIVE, ResultSet.CONCUR_UPDATABLE)) { - getEventSubscriptionBySubscriptionIdStatement.setString(1, subscriptionId); - try (ResultSet resultSet = getEventSubscriptionBySubscriptionIdStatement.executeQuery()) { - if (resultSet.next()) { - mapResultSetToEventSubscription(retrievedSubscription, resultSet); - resultSet.beforeFirst(); // Reset the cursor position to the beginning of the result set. - while (resultSet.next()) { - String eventType = resultSet.getString(EventNotificationConstants.EVENT_TYPE); - if (eventType != null) { - eventTypes.add(eventType); - } - } - if (!eventTypes.isEmpty()) { - retrievedSubscription.setEventTypes(eventTypes); - } - } else { - log.error("No event notification subscription found for the given subscription id."); - throw new OBEventNotificationException( - EventNotificationConstants.EVENT_SUBSCRIPTION_NOT_FOUND); - } - } catch (SQLException e) { - log.error("SQL exception when retrieving the event notification subscription.", e); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_RETRIEVING_EVENT_SUBSCRIPTION); - } - } catch (SQLException e) { - log.error("SQL exception when retrieving the event notification subscription.", e); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_RETRIEVING_EVENT_SUBSCRIPTION); - } - return retrievedSubscription; - } - - @Override - public Boolean updateEventSubscription(Connection connection, EventSubscription eventSubscription) - throws OBEventNotificationException { - boolean isUpdated = false; - final String sql = sqlStatements.updateEventSubscriptionQuery(); - try (PreparedStatement updateEventSubscriptionStatement = connection.prepareStatement(sql)) { - updateEventSubscriptionStatement.setString(1, eventSubscription.getCallbackUrl()); - updateEventSubscriptionStatement.setLong(2, Instant.now().getEpochSecond()); - updateEventSubscriptionStatement.setObject(3, eventSubscription.getRequestData(), - java.sql.Types.OTHER); - updateEventSubscriptionStatement.setString(4, eventSubscription.getSubscriptionId()); - int affectedRows = updateEventSubscriptionStatement.executeUpdate(); - if (affectedRows > 0) { - log.debug("Event notification subscription is successfully updated."); - isUpdated = true; - } - } catch (SQLException e) { - log.error("SQL exception when updating event notification subscription", e); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_UPDATING_EVENT_SUBSCRIPTION); - } - return isUpdated; - } - - private void mapResultSetToEventSubscription(EventSubscription response, ResultSet resultSet) throws SQLException { - response.setSubscriptionId(resultSet.getString(EventNotificationConstants.SUBSCRIPTION_ID)); - response.setClientId(resultSet.getString(EventNotificationConstants.CLIENT_ID)); - response.setCallbackUrl(resultSet.getString(EventNotificationConstants.CALLBACK_URL)); - response.setTimeStamp(resultSet.getLong(EventNotificationConstants.TIME_STAMP)); - response.setSpecVersion(resultSet.getString(EventNotificationConstants.SPEC_VERSION)); - response.setStatus(resultSet.getString(EventNotificationConstants.STATUS)); - response.setRequestData(resultSet.getString(EventNotificationConstants.REQUEST)); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/PostgreSqlPollingDAOImpl.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/PostgreSqlPollingDAOImpl.java deleted file mode 100644 index 0421e4ec..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/PostgreSqlPollingDAOImpl.java +++ /dev/null @@ -1,250 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dao; - -import com.wso2.openbanking.accelerator.common.util.DatabaseUtil; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationEvent; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import net.minidev.json.parser.ParseException; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.util.ArrayList; -import java.util.List; - -/** - * PostgreSql event polling dao class. - */ -@Generated(message = "Postgres Implementation") -public class PostgreSqlPollingDAOImpl extends AggregatedPollingDAOImpl { - - private static Log log = LogFactory.getLog(PostgreSqlPollingDAOImpl.class); - - public PostgreSqlPollingDAOImpl(NotificationPollingSqlStatements notificationPollingSqlStatements) { - super(notificationPollingSqlStatements); - } - - @Override - public List getNotificationsByClientIdAndStatus(String clientId, String status, int max) - throws OBEventNotificationException { - - List notificationList; - Connection connection = DatabaseUtil.getDBConnection(); - try { - - notificationList = new ArrayList<>(); - - if (log.isDebugEnabled()) { - log.debug(String.format(EventNotificationConstants.DB_CONN_ESTABLISHED, - clientId.replaceAll("[\r\n]", ""))); - } - - final String sql = sqlStatements.getMaxNotificationsQuery(); - try (PreparedStatement getNotificationsPreparedStatement = connection.prepareStatement(sql, - ResultSet.TYPE_SCROLL_SENSITIVE, ResultSet.CONCUR_UPDATABLE)) { - getNotificationsPreparedStatement.setString(1, clientId); - getNotificationsPreparedStatement.setString(2, status); - getNotificationsPreparedStatement.setInt(3, max); - - try (ResultSet notificationResultSet = getNotificationsPreparedStatement.executeQuery()) { - if (notificationResultSet.next()) { - - //bring pointer back to the top of the result set if not on the top - if (!notificationResultSet.isBeforeFirst()) { - notificationResultSet.beforeFirst(); - } - - //read event notifications from the result set - while (notificationResultSet.next()) { - NotificationDTO notification = new NotificationDTO(); - - notification.setNotificationId(notificationResultSet.getString - (EventNotificationConstants.NOTIFICATION_ID)); - notification.setClientId(notificationResultSet.getString - (EventNotificationConstants.CLIENT_ID)); - notification.setResourceId(notificationResultSet.getString - (EventNotificationConstants.RESOURCE_ID)); - notification.setStatus(notificationResultSet.getString - (EventNotificationConstants.STATUS)); - notification.setUpdatedTimeStamp((notificationResultSet.getTimestamp( - (EventNotificationConstants.UPDATED_TIMESTAMP)).getTime())); - - notificationList.add(notification); - } - notificationResultSet.close(); - getNotificationsPreparedStatement.close(); - - if (log.isDebugEnabled()) { - log.debug(String.format(EventNotificationConstants.RETRIEVED_NOTIFICATION_CLIENT, - clientId.replaceAll("[\r\n]", ""))); - } - - } else { - if (log.isDebugEnabled()) { - log.debug(String.format(EventNotificationConstants.NO_NOTIFICATIONS_FOUND_CLIENT, - clientId.replaceAll("[\r\n]", ""))); - } - } - } - } catch (SQLException e) { - throw new OBEventNotificationException(String.format - (EventNotificationConstants.DB_ERROR_NOTIFICATION_RETRIEVE, - clientId), e); - } - } finally { - log.debug(EventNotificationConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - return notificationList; - } - - @Override - public List getEventsByNotificationID(String notificationId) - throws OBEventNotificationException { - - List eventList = new ArrayList<>(); - - Connection connection = DatabaseUtil.getDBConnection(); - try { - - final String sql = sqlStatements.getEventsByNotificationIdQuery(); - - try (PreparedStatement getEventsPreparedStatement = connection.prepareStatement(sql, - ResultSet.TYPE_SCROLL_SENSITIVE, ResultSet.CONCUR_UPDATABLE)) { - - getEventsPreparedStatement.setString(1, notificationId); - - try (ResultSet eventsResultSet = getEventsPreparedStatement.executeQuery()) { - if (eventsResultSet.next()) { - - //bring pointer back to the top of the result set if not on the top - if (!eventsResultSet.isBeforeFirst()) { - eventsResultSet.beforeFirst(); - } - - //read event notifications from the result set - while (eventsResultSet.next()) { - NotificationEvent event = new NotificationEvent(); - event.setNotificationId(eventsResultSet.getString - (EventNotificationConstants.NOTIFICATION_ID)); - event.setEventType(eventsResultSet.getString - (EventNotificationConstants.EVENT_TYPE)); - event.setEventInformation(EventNotificationServiceUtil. - getEventJSONFromString(eventsResultSet.getString - (EventNotificationConstants.EVENT_INFO))); - eventList.add(event); - } - eventsResultSet.close(); - getEventsPreparedStatement.close(); - - if (log.isDebugEnabled()) { - log.debug(String.format(EventNotificationConstants.RETRIEVED_EVENTS_NOTIFICATION, - notificationId.replaceAll("[\r\n]", ""))); - } - } else { - if (log.isDebugEnabled()) { - log.debug(String.format(EventNotificationConstants.NO_EVENTS_NOTIFICATION_ID, - notificationId.replaceAll("[\r\n]", ""))); - } - } - } catch (ParseException e) { - log.error(String.format(EventNotificationConstants.PARSE_ERROR_NOTIFICATION_ID, - notificationId.replaceAll("[\r\n]", "")), e); - throw new OBEventNotificationException(String.format ( - EventNotificationConstants.PARSE_ERROR_NOTIFICATION_ID, notificationId), e); - } - } catch (SQLException e) { - log.error(String.format(EventNotificationConstants.DB_ERROR_EVENTS_RETRIEVE, - notificationId.replaceAll("[\r\n]", "")), e); - throw new OBEventNotificationException(String.format - (EventNotificationConstants.DB_ERROR_EVENTS_RETRIEVE, notificationId), e); - } - - } finally { - log.debug(EventNotificationConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - - return eventList; - } - - @Override - public List getNotificationsByStatus(String status) - throws OBEventNotificationException { - List notificationList; - Connection connection = DatabaseUtil.getDBConnection(); - try { - notificationList = new ArrayList<>(); - final String sql = sqlStatements.getMaxNotificationsQuery(); - try (PreparedStatement getNotificationsPreparedStatement = connection.prepareStatement(sql, - ResultSet.TYPE_SCROLL_SENSITIVE, ResultSet.CONCUR_UPDATABLE)) { - getNotificationsPreparedStatement.setString(1, status); - try (ResultSet notificationResultSet = getNotificationsPreparedStatement.executeQuery()) { - if (notificationResultSet.next()) { - //bring pointer back to the top of the result set if not on the top - if (!notificationResultSet.isBeforeFirst()) { - notificationResultSet.beforeFirst(); - } - //read event notifications from the result set - while (notificationResultSet.next()) { - NotificationDTO notification = new NotificationDTO(); - notification.setNotificationId(notificationResultSet.getString - (EventNotificationConstants.NOTIFICATION_ID)); - notification.setClientId(notificationResultSet.getString - (EventNotificationConstants.CLIENT_ID)); - notification.setResourceId(notificationResultSet.getString - (EventNotificationConstants.RESOURCE_ID)); - notification.setStatus(notificationResultSet.getString - (EventNotificationConstants.STATUS)); - notification.setUpdatedTimeStamp((notificationResultSet.getTimestamp( - (EventNotificationConstants.UPDATED_TIMESTAMP)).getTime())); - notificationList.add(notification); - } - notificationResultSet.close(); - getNotificationsPreparedStatement.close(); - if (log.isDebugEnabled()) { - log.debug( - EventNotificationConstants.RETRIEVED_NOTIFICATION_CLIENT); - } - } else { - if (log.isDebugEnabled()) { - log.debug( - EventNotificationConstants.NO_NOTIFICATIONS_FOUND_CLIENT); - } - } - } - } catch (SQLException e) { - throw new OBEventNotificationException( - EventNotificationConstants.DB_ERROR_NOTIFICATION_RETRIEVE, e); - } - } finally { - log.debug(EventNotificationConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - return notificationList; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/EventNotificationErrorDTO.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/EventNotificationErrorDTO.java deleted file mode 100644 index 1121b211..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/EventNotificationErrorDTO.java +++ /dev/null @@ -1,53 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dto; - -import com.fasterxml.jackson.annotation.JsonProperty; - -/** - * Error model for Event Notifications. - */ -public class EventNotificationErrorDTO { - - private String errorDescription; - private String error; - - @JsonProperty("error_description") - public String getErrorDescription() { - - return errorDescription; - } - - public void setErrorDescription(String errorDescription) { - - this.errorDescription = errorDescription; - } - - @JsonProperty("error") - public String getError() { - - return error; - } - - public void setError(String error) { - - this.error = error; - } - -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/EventPollingDTO.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/EventPollingDTO.java deleted file mode 100644 index 8fb476f9..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/EventPollingDTO.java +++ /dev/null @@ -1,76 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dto; - -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationError; - - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -/** - * Event Polling DTO. - */ -public class EventPollingDTO { - - //Set to true by default as WSO2 Open Banking don't support long polling - private final Boolean returnImmediately = true; - private String clientId = null; - private int maxEvents = 0; - private List ack = new ArrayList<>(); - private Map errors = new HashMap<>(); - - public String getClientId() { - return clientId; - } - - public void setClientId(String clientId) { - this.clientId = clientId; - } - - public Boolean getReturnImmediately() { - return returnImmediately; - } - - public int getMaxEvents() { - return maxEvents; - } - - public void setMaxEvents(int maxEvents) { - this.maxEvents = maxEvents; - } - - public List getAck() { - return ack; - } - - public void setAck(String ack) { - this.ack.add(ack); - } - - public Map getErrors() { - return errors; - } - - public void setErrors(String notificationId, NotificationError errorNotification) { - this.errors.put(notificationId, errorNotification); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/EventSubscriptionDTO.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/EventSubscriptionDTO.java deleted file mode 100644 index 3f19bd42..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/EventSubscriptionDTO.java +++ /dev/null @@ -1,54 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dto; - -import net.minidev.json.JSONObject; - -/** - * Event Subscription DTO. - */ -public class EventSubscriptionDTO { - private String clientId = null; - private String subscriptionId = null; - private JSONObject requestData = null; - - public String getClientId() { - return clientId; - } - - public void setClientId(String clientId) { - this.clientId = clientId; - } - - public String getSubscriptionId() { - return subscriptionId; - } - - public void setSubscriptionId(String subscriptionId) { - this.subscriptionId = subscriptionId; - } - - public JSONObject getRequestData() { - return requestData; - } - - public void setRequestData(JSONObject requestData) { - this.requestData = requestData; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/NotificationCreationDTO.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/NotificationCreationDTO.java deleted file mode 100644 index 2985c359..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/NotificationCreationDTO.java +++ /dev/null @@ -1,59 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dto; - -import net.minidev.json.JSONObject; - -import java.util.HashMap; -import java.util.Map; - -/** - * Event Creation DTO. - */ -public class NotificationCreationDTO { - - private Map events = new HashMap(); - private String clientId = null; - private String resourceId = null; - - public String getClientId() { - return clientId; - } - - public void setClientId(String clientId) { - this.clientId = clientId; - } - - public String getResourceId() { - return resourceId; - } - - public void setResourceId(String resourceId) { - this.resourceId = resourceId; - } - - public Map getEventPayload() { - return this.events; - } - - public void setEventPayload(String notificationType, JSONObject notificationInfo) { - this.events.put(notificationType, notificationInfo); - } -} - diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/NotificationDTO.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/NotificationDTO.java deleted file mode 100644 index b8ae5e20..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/dto/NotificationDTO.java +++ /dev/null @@ -1,70 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dto; - -/** - * DAO class to map notification data to db. - */ -public class NotificationDTO { - String notificationId = null; - String clientId = null; - String resourceId = null; - String status = null; - Long updatedTimeStamp = null; - - public String getNotificationId() { - return notificationId; - } - - public void setNotificationId(String notificationId) { - this.notificationId = notificationId; - } - - public String getClientId() { - return clientId; - } - - public void setClientId(String clientId) { - this.clientId = clientId; - } - - public String getResourceId() { - return resourceId; - } - - public void setResourceId(String resourceId) { - this.resourceId = resourceId; - } - - public String getStatus() { - return status; - } - - public void setStatus(String status) { - this.status = status; - } - - public Long getUpdatedTimeStamp() { - return updatedTimeStamp; - } - - public void setUpdatedTimeStamp(Long updatedTimeStamp) { - this.updatedTimeStamp = updatedTimeStamp; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/exceptions/OBEventNotificationException.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/exceptions/OBEventNotificationException.java deleted file mode 100644 index acd42226..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/exceptions/OBEventNotificationException.java +++ /dev/null @@ -1,35 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.exceptions; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; - -/** - * Event Notification Exceptions. - */ -public class OBEventNotificationException extends OpenBankingException { - - public OBEventNotificationException(String message) { - super(message); - } - - public OBEventNotificationException(String message, Throwable e) { - super(message, e); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventCreationServiceHandler.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventCreationServiceHandler.java deleted file mode 100644 index df5bd14a..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventCreationServiceHandler.java +++ /dev/null @@ -1,105 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.handler; - -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationCreationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.response.EventCreationResponse; -import com.wso2.openbanking.accelerator.event.notifications.service.service.EventCreationService; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import net.minidev.json.JSONObject; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -/** - * This is to handle OB Event Creation. - */ -public class DefaultEventCreationServiceHandler implements EventCreationServiceHandler { - - private static final Log log = LogFactory.getLog(DefaultEventCreationServiceHandler.class); - private EventCreationService eventCreationService = new EventCreationService(); - - public void setEventCreationService(EventCreationService eventCreationService) { - this.eventCreationService = eventCreationService; - } - - /** - * This method is used to publish OB events in the accelerator database. - * - * @param notificationCreationDTO Notification details DTO - * @return EventCreationResponse Response after event creation - */ - public EventCreationResponse publishOBEvent(NotificationCreationDTO notificationCreationDTO) { - - //validate if the resourceID is existing - ConsentResource consentResource = null; - ConsentCoreServiceImpl consentCoreService = EventNotificationServiceUtil.getConsentCoreServiceImpl(); - EventCreationResponse eventCreationResponse = new EventCreationResponse(); - - try { - consentResource = consentCoreService.getConsent(notificationCreationDTO.getResourceId(), - false); - - if (log.isDebugEnabled()) { - log.debug("Consent resource available for resource ID " + - consentResource.getConsentID().replaceAll("[\r\n]", "")); - } - } catch (ConsentManagementException e) { - log.error("Consent Management Exception when validating the consent resource", e); - eventCreationResponse.setErrorResponse(String.format("A resource was not found for the resource " + - "id : '%s' in the database. ", notificationCreationDTO.getResourceId())); - eventCreationResponse.setStatus(EventNotificationConstants.BAD_REQUEST); - return eventCreationResponse; - } - - //validate if the clientID is existing - try { - EventNotificationServiceUtil.validateClientId(notificationCreationDTO.getClientId()); - - } catch (OBEventNotificationException e) { - log.error("Invalid client ID", e); - eventCreationResponse.setErrorResponse(String.format("A client was not found" + - " for the client id : '%s' in the database. ", - notificationCreationDTO.getClientId().replaceAll("[\r\n]", ""))); - eventCreationResponse.setStatus(EventNotificationConstants.BAD_REQUEST); - return eventCreationResponse; - } - - String registrationResponse = ""; - try { - registrationResponse = eventCreationService.publishOBEventNotification(notificationCreationDTO); - JSONObject responseJSON = new JSONObject(); - responseJSON.put(EventNotificationConstants.NOTIFICATIONS_ID, registrationResponse); - eventCreationResponse.setStatus(EventNotificationConstants.CREATED); - eventCreationResponse.setResponseBody(responseJSON); - return eventCreationResponse; - - } catch (OBEventNotificationException e) { - log.error("OB Event Notification Creation error", e); - } - - eventCreationResponse.setStatus(EventNotificationConstants.BAD_REQUEST); - eventCreationResponse.setErrorResponse("Error in event creation request payload"); - return eventCreationResponse; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventPollingServiceHandler.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventPollingServiceHandler.java deleted file mode 100644 index 9372f04d..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventPollingServiceHandler.java +++ /dev/null @@ -1,160 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.handler; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.EventPollingDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.AggregatedPollingResponse; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationError; -import com.wso2.openbanking.accelerator.event.notifications.service.response.EventPollingResponse; -import com.wso2.openbanking.accelerator.event.notifications.service.service.EventPollingService; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.Locale; - -/** - * This is the service handler for event polling. - */ -public class DefaultEventPollingServiceHandler implements EventPollingServiceHandler { - - private static final Log log = LogFactory.getLog(DefaultEventPollingServiceHandler.class); - - public void setEventPollingService(EventPollingService eventPollingService) { - this.eventPollingService = eventPollingService; - } - - private EventPollingService eventPollingService = new EventPollingService(); - - - /** - * This method is used to Poll Events as per request params. - * @param eventPollingRequest JSON request for event polling - * @return EventPollingResponse - */ - public EventPollingResponse pollEvents(JSONObject eventPollingRequest) { - - EventPollingDTO eventPollingDTO = mapPollingRequest(eventPollingRequest); - EventPollingResponse eventPollingResponse = new EventPollingResponse(); - - //Validate clientID of the polling request - try { - EventNotificationServiceUtil.validateClientId(eventPollingDTO.getClientId()); - } catch (OBEventNotificationException e) { - log.error("Invalid client ID", e); - eventPollingResponse.setStatus(EventNotificationConstants.BAD_REQUEST); - eventPollingResponse.setErrorResponse(EventNotificationServiceUtil.getErrorDTO( - EventNotificationConstants.INVALID_REQUEST, String.format("A client was not found" + - " for the client id : '%s' in the database.. ", eventPollingDTO.getClientId()))); - return eventPollingResponse; - } - - //Poll events - try { - AggregatedPollingResponse aggregatedPollingResponse = eventPollingService.pollEvents(eventPollingDTO); - eventPollingResponse.setStatus(aggregatedPollingResponse.getStatus()); - eventPollingResponse.setResponseBody(getPollingResponseJSON(aggregatedPollingResponse)); - return eventPollingResponse; - } catch (OBEventNotificationException e) { - log.error("OB Event Notification error" , e); - eventPollingResponse.setStatus(EventNotificationConstants.BAD_REQUEST); - eventPollingResponse.setErrorResponse(EventNotificationServiceUtil.getErrorDTO( - EventNotificationConstants.INVALID_REQUEST, e.getMessage())); - return eventPollingResponse; - } - - } - - /** - * This method will map the eventPollingRequest JSON to EventPollingDTO. - * @param eventPollingRequest JSON request for event polling - * @return EventPollingDTO - */ - public EventPollingDTO mapPollingRequest(JSONObject eventPollingRequest) { - - EventPollingDTO eventPollingDTO = new EventPollingDTO(); - eventPollingDTO.setClientId(eventPollingRequest.get(EventNotificationConstants.X_WSO2_CLIENT_ID).toString()); - - if (eventPollingRequest.size() == 0) { - - eventPollingDTO.setMaxEvents(OpenBankingConfigParser.getInstance().getNumberOfSetsToReturn()); - - return eventPollingDTO; - } - - //Set acknowledged events to DTO - if (eventPollingRequest.containsKey(EventNotificationConstants.ACK.toLowerCase(Locale.ROOT))) { - JSONArray acknowledgedEvents = (JSONArray) eventPollingRequest. - get(EventNotificationConstants.ACK.toLowerCase(Locale.ROOT)); - acknowledgedEvents.forEach((event -> { - eventPollingDTO.setAck(event.toString()); - })); - } - - //Set error events to DTO - if (eventPollingRequest.containsKey(EventNotificationConstants.SET_ERRORS)) { - JSONObject errorEvents = (JSONObject) eventPollingRequest. - get(EventNotificationConstants.SET_ERRORS); - errorEvents.keySet().forEach(errorEvent -> { - JSONObject errorEventInformation = (JSONObject) errorEvents.get(errorEvent); - NotificationError notificationError = getNotificationError(errorEventInformation); - notificationError.setNotificationId(errorEvent); - eventPollingDTO.setErrors(errorEvent, notificationError); - }); - } - - //Set maxEvents count to return - if (eventPollingRequest.containsKey(EventNotificationConstants.MAX_EVENTS)) { - eventPollingDTO.setMaxEvents(Integer.parseInt(eventPollingRequest. - get(EventNotificationConstants.MAX_EVENTS).toString())); - } else { - eventPollingDTO.setMaxEvents(OpenBankingConfigParser.getInstance().getNumberOfSetsToReturn()); - } - - return eventPollingDTO; - } - - @Generated(message = "Private method tested when testing the invoked method") - private NotificationError getNotificationError(JSONObject errorEvent) { - - NotificationError notificationError = new NotificationError(); - notificationError.setErrorCode(errorEvent.get( - EventNotificationConstants.ERROR.toLowerCase(Locale.ROOT)).toString()); - notificationError.setErrorDescription( - errorEvent.get(EventNotificationConstants.DESCRIPTION).toString()); - return notificationError; - } - - @Generated(message = "Private method tested when testing the invoked method") - private JSONObject getPollingResponseJSON(AggregatedPollingResponse aggregatedPollingResponse) { - - JSONObject responseJSON = new JSONObject(); - responseJSON.put(EventNotificationConstants.SETS, aggregatedPollingResponse.getSets()); - responseJSON.put(EventNotificationConstants.MORE_AVAILABLE, - aggregatedPollingResponse.isMoreAvailable()); - return responseJSON; - } - -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventSubscriptionServiceHandler.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventSubscriptionServiceHandler.java deleted file mode 100644 index e8a281d4..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventSubscriptionServiceHandler.java +++ /dev/null @@ -1,345 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.handler; - -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.EventSubscriptionDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.EventSubscription; -import com.wso2.openbanking.accelerator.event.notifications.service.response.EventSubscriptionResponse; -import com.wso2.openbanking.accelerator.event.notifications.service.service.EventSubscriptionService; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import net.minidev.json.JSONObject; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.HttpStatus; - -import java.util.ArrayList; -import java.util.List; - -/** - * This is the default service handler for event notification subscription. - */ -public class DefaultEventSubscriptionServiceHandler implements EventSubscriptionServiceHandler { - private static final Log log = LogFactory.getLog(DefaultEventSubscriptionServiceHandler.class); - - private EventSubscriptionService eventSubscriptionService = new EventSubscriptionService(); - - public void setEventSubscriptionService(EventSubscriptionService eventSubscriptionService) { - this.eventSubscriptionService = eventSubscriptionService; - } - - /** - * This method is used to create event subscriptions. - * - * @param eventSubscriptionRequestDto Event Subscription DTO - * @return EventSubscriptionResponse Event Subscription Response - */ - public EventSubscriptionResponse createEventSubscription(EventSubscriptionDTO eventSubscriptionRequestDto) { - EventSubscriptionResponse eventSubscriptionResponse = new EventSubscriptionResponse(); - - EventSubscriptionResponse clientIdValidation = validateClientId(eventSubscriptionRequestDto.getClientId()); - // check whether clientIdValidation is not null, then return the error response - if (clientIdValidation != null) { - return clientIdValidation; - } - - EventSubscription eventSubscription = mapEventSubscriptionDtoToModel(eventSubscriptionRequestDto); - - try { - EventSubscription createEventSubscriptionResponse = eventSubscriptionService. - createEventSubscription(eventSubscription); - eventSubscriptionResponse.setStatus(HttpStatus.SC_CREATED); - eventSubscriptionResponse. - setResponseBody(mapSubscriptionModelToResponseJson(createEventSubscriptionResponse)); - return eventSubscriptionResponse; - } catch (OBEventNotificationException e) { - log.error("Error occurred while creating event subscription", e); - eventSubscriptionResponse.setStatus(HttpStatus.SC_INTERNAL_SERVER_ERROR); - eventSubscriptionResponse.setErrorResponse(EventNotificationServiceUtil.getErrorDTO( - EventNotificationConstants.INVALID_REQUEST, e.getMessage())); - return eventSubscriptionResponse; - } - - } - - /** - * This method is used to retrieve a single event subscription. - * - * @param clientId Client ID of the subscription created - * @param subscriptionId Subscription ID of the subscription created - * @return EventSubscriptionResponse Event Subscription Response containing subscription - * details for the given subscription ID - */ - public EventSubscriptionResponse getEventSubscription(String clientId, String subscriptionId) { - EventSubscriptionResponse eventSubscriptionResponse = new EventSubscriptionResponse(); - - EventSubscriptionResponse clientIdValidation = validateClientId(clientId); - // check whether clientIdValidation is not null, then return the error response - if (clientIdValidation != null) { - return clientIdValidation; - } - - try { - EventSubscription eventSubscription = eventSubscriptionService. - getEventSubscriptionBySubscriptionId(subscriptionId); - eventSubscriptionResponse.setStatus(HttpStatus.SC_OK); - eventSubscriptionResponse.setResponseBody(mapSubscriptionModelToResponseJson(eventSubscription)); - return eventSubscriptionResponse; - } catch (OBEventNotificationException e) { - log.error("Error occurred while retrieving event subscription", e); - if (e.getMessage().equals(EventNotificationConstants.EVENT_SUBSCRIPTION_NOT_FOUND)) { - eventSubscriptionResponse.setStatus(HttpStatus.SC_BAD_REQUEST); - eventSubscriptionResponse.setErrorResponse(EventNotificationServiceUtil.getErrorDTO( - EventNotificationConstants.INVALID_REQUEST, e.getMessage())); - } else { - eventSubscriptionResponse.setStatus(HttpStatus.SC_INTERNAL_SERVER_ERROR); - eventSubscriptionResponse.setErrorResponse(EventNotificationServiceUtil.getErrorDTO( - EventNotificationConstants.INVALID_REQUEST, e.getMessage())); - } - return eventSubscriptionResponse; - } - } - - /** - * This method is used to retrieve all event subscriptions of a client. - * - * @param clientId Client ID - * @return EventSubscriptionResponse Event Subscription Response containing all the subscriptions - */ - public EventSubscriptionResponse getAllEventSubscriptions(String clientId) { - EventSubscriptionResponse eventSubscriptionResponse = new EventSubscriptionResponse(); - - EventSubscriptionResponse clientIdValidation = validateClientId(clientId); - // check whether clientIdValidation is not null, then return the error response - if (clientIdValidation != null) { - return clientIdValidation; - } - - try { - List eventSubscriptionList = eventSubscriptionService. - getEventSubscriptionsByClientId(clientId); - List eventSubscriptionResponseList = new ArrayList<>(); - for (EventSubscription eventSubscription : eventSubscriptionList) { - eventSubscriptionResponseList.add(mapSubscriptionModelToResponseJson(eventSubscription)); - } - eventSubscriptionResponse.setStatus(HttpStatus.SC_OK); - eventSubscriptionResponse.setResponseBody(eventSubscriptionResponseList); - return eventSubscriptionResponse; - } catch (OBEventNotificationException e) { - log.error("Error occurred while retrieving event subscriptions", e); - eventSubscriptionResponse.setStatus(HttpStatus.SC_INTERNAL_SERVER_ERROR); - eventSubscriptionResponse.setErrorResponse(EventNotificationServiceUtil.getErrorDTO( - EventNotificationConstants.INVALID_REQUEST, e.getMessage())); - return eventSubscriptionResponse; - } - } - - /** - * This method is used to retrieve all event subscriptions by event type. - * - * @param clientId Client ID - * @param eventType Event Type to retrieve subscriptions - * @return EventSubscriptionResponse Event Subscription Response containing subscriptions per specified - * event type - */ - public EventSubscriptionResponse getEventSubscriptionsByEventType(String clientId, String eventType) { - EventSubscriptionResponse eventSubscriptionResponse = new EventSubscriptionResponse(); - - EventSubscriptionResponse clientIdValidation = validateClientId(clientId); - // check whether clientIdValidation is not null, then return the error response - if (clientIdValidation != null) { - return clientIdValidation; - } - - try { - List eventSubscriptionList = eventSubscriptionService. - getEventSubscriptionsByClientIdAndEventType(eventType); - List eventSubscriptionResponseList = new ArrayList<>(); - for (EventSubscription eventSubscription : eventSubscriptionList) { - eventSubscriptionResponseList.add(mapSubscriptionModelToResponseJson(eventSubscription)); - } - eventSubscriptionResponse.setStatus(HttpStatus.SC_OK); - eventSubscriptionResponse.setResponseBody(eventSubscriptionResponseList); - return eventSubscriptionResponse; - } catch (OBEventNotificationException e) { - log.error("Error occurred while retrieving event subscriptions", e); - eventSubscriptionResponse.setStatus(HttpStatus.SC_INTERNAL_SERVER_ERROR); - eventSubscriptionResponse.setErrorResponse(EventNotificationServiceUtil.getErrorDTO( - EventNotificationConstants.INVALID_REQUEST, e.getMessage())); - return eventSubscriptionResponse; - } - } - - /** - * This method is used to update an event subscription. - * - * @param eventSubscriptionUpdateRequestDto Event Subscription Update Request DTO - * @return EventSubscriptionResponse Event Subscription Response containing the updated subscription - */ - public EventSubscriptionResponse updateEventSubscription(EventSubscriptionDTO eventSubscriptionUpdateRequestDto) { - EventSubscriptionResponse eventSubscriptionResponse = new EventSubscriptionResponse(); - - EventSubscriptionResponse clientIdValidation = validateClientId(eventSubscriptionUpdateRequestDto. - getClientId()); - // check whether clientIdValidation is not null, then return the error response - if (clientIdValidation != null) { - return clientIdValidation; - } - - EventSubscription eventSubscription = mapEventSubscriptionDtoToModel(eventSubscriptionUpdateRequestDto); - - try { - Boolean isUpdated = eventSubscriptionService.updateEventSubscription(eventSubscription); - if (!isUpdated) { - eventSubscriptionResponse.setStatus(HttpStatus.SC_BAD_REQUEST); - eventSubscriptionResponse.setErrorResponse(EventNotificationServiceUtil.getErrorDTO( - EventNotificationConstants.INVALID_REQUEST, - "Event subscription not found.")); - return eventSubscriptionResponse; - } - eventSubscriptionResponse.setStatus(HttpStatus.SC_OK); - EventSubscription eventSubscriptionUpdateResponse = eventSubscriptionService. - getEventSubscriptionBySubscriptionId(eventSubscriptionUpdateRequestDto.getSubscriptionId()); - eventSubscriptionResponse. - setResponseBody(mapSubscriptionModelToResponseJson(eventSubscriptionUpdateResponse)); - return eventSubscriptionResponse; - } catch (OBEventNotificationException e) { - log.error("Error occurred while updating event subscription", e); - eventSubscriptionResponse.setStatus(HttpStatus.SC_INTERNAL_SERVER_ERROR); - eventSubscriptionResponse.setErrorResponse(EventNotificationServiceUtil.getErrorDTO( - EventNotificationConstants.INVALID_REQUEST, e.getMessage())); - return eventSubscriptionResponse; - } - } - - /** - * This method is used to delete an event subscription. - * - * @param clientId Client ID - * @param subscriptionId Subscription ID to be deleted - * @return EventSubscriptionResponse Event Subscription Response containing the deleted subscription - */ - public EventSubscriptionResponse deleteEventSubscription(String clientId, String subscriptionId) { - EventSubscriptionResponse eventSubscriptionResponse = new EventSubscriptionResponse(); - - EventSubscriptionResponse clientIdValidation = validateClientId(clientId); - // check whether clientIdValidation is not null, then return the error response - if (clientIdValidation != null) { - return clientIdValidation; - } - try { - Boolean isDeleted = eventSubscriptionService.deleteEventSubscription(subscriptionId); - if (!isDeleted) { - eventSubscriptionResponse.setStatus(HttpStatus.SC_BAD_REQUEST); - eventSubscriptionResponse.setErrorResponse(EventNotificationServiceUtil.getErrorDTO( - EventNotificationConstants.INVALID_REQUEST, - "Event subscription not found")); - return eventSubscriptionResponse; - } - eventSubscriptionResponse.setStatus(HttpStatus.SC_NO_CONTENT); - return eventSubscriptionResponse; - } catch (OBEventNotificationException e) { - log.error("Error occurred while deleting event subscription", e); - eventSubscriptionResponse.setStatus(HttpStatus.SC_INTERNAL_SERVER_ERROR); - eventSubscriptionResponse.setErrorResponse(EventNotificationServiceUtil.getErrorDTO( - EventNotificationConstants.INVALID_REQUEST, e.getMessage())); - return eventSubscriptionResponse; - } - } - - /** - * This method is used to validate the client ID. - * - * @param clientId Client ID - * @return EventSubscriptionResponse Return EventSubscriptionResponse if the client ID is - * invalid, if the client ID is valid, null will be returned. - */ - private EventSubscriptionResponse validateClientId(String clientId) { - try { - EventNotificationServiceUtil.validateClientId(clientId); - } catch (OBEventNotificationException e) { - log.error("Invalid client ID", e); - EventSubscriptionResponse eventSubscriptionResponse = new EventSubscriptionResponse(); - eventSubscriptionResponse.setStatus(HttpStatus.SC_BAD_REQUEST); - eventSubscriptionResponse.setErrorResponse(EventNotificationServiceUtil.getErrorDTO( - EventNotificationConstants.INVALID_REQUEST, e.getMessage())); - return eventSubscriptionResponse; - } - return null; - } - - /** - * This method will map the event subscription DTO to event subscription model - * to be passed to the dao layer. - * - * @param eventSubscriptionDTO Event Subscription DTO - * @return EventSubscription Event Subscription Model mapped - */ - private EventSubscription mapEventSubscriptionDtoToModel(EventSubscriptionDTO eventSubscriptionDTO) { - EventSubscription eventSubscription = new EventSubscription(); - - eventSubscription.setSubscriptionId(eventSubscriptionDTO.getSubscriptionId()); - - JSONObject payload = eventSubscriptionDTO.getRequestData(); - List eventTypes = new ArrayList<>(); - Object eventTypesObj = payload.get(EventNotificationConstants.EVENT_TYPE_PARAM); - if (eventTypesObj instanceof List) { - List eventTypesList = (List) eventTypesObj; - for (Object item : eventTypesList) { - if (item instanceof String) { - eventTypes.add((String) item); - } - } - } - eventSubscription.setEventTypes(eventTypes); - eventSubscription.setCallbackUrl(payload.get(EventNotificationConstants.CALLBACK_URL_PARAM) != null ? - payload.get(EventNotificationConstants.CALLBACK_URL_PARAM).toString() : null); - eventSubscription.setSpecVersion(payload.get(EventNotificationConstants.VERSION_PARAM) != null ? - payload.get(EventNotificationConstants.VERSION_PARAM).toString() : null); - eventSubscription.setClientId(eventSubscriptionDTO.getClientId()); - eventSubscription.setRequestData(payload.toJSONString()); - return eventSubscription; - } - - /** - * This method is used to create the response JSON object from the event subscription model. - * - * @param eventSubscription Event Subscription Model - * @return JSONObject containing mapped subscription - */ - public JSONObject mapSubscriptionModelToResponseJson(EventSubscription eventSubscription) { - JSONObject responsePayload = new JSONObject(); - - if (eventSubscription.getSubscriptionId() != null) { - responsePayload.put(EventNotificationConstants.SUBSCRIPTION_ID_PARAM, - eventSubscription.getSubscriptionId()); - } - if (eventSubscription.getCallbackUrl() != null) { - responsePayload.put(EventNotificationConstants.CALLBACK_URL_PARAM, eventSubscription.getCallbackUrl()); - } - if (eventSubscription.getSpecVersion() != null) { - responsePayload.put(EventNotificationConstants.VERSION_PARAM, eventSubscription.getSpecVersion()); - } - if (eventSubscription.getEventTypes() != null) { - responsePayload.put(EventNotificationConstants.EVENT_TYPE_PARAM, eventSubscription.getEventTypes()); - } - return responsePayload; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventCreationServiceHandler.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventCreationServiceHandler.java deleted file mode 100644 index 671a64c6..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventCreationServiceHandler.java +++ /dev/null @@ -1,38 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.handler; - -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationCreationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.response.EventCreationResponse; - -/** - * Event creation service handler is used to map the creation request and validate the date before - * calling the service. In need of a custom handling this class can be extended and the extended class - * can be added to the deployment.toml under event_creation_handler to execute the specific class. - */ -public interface EventCreationServiceHandler { - /** - * This method is used to publish OB events in the accelerator database. The method is a generic - * method that is used to persist data into the OB_NOTIFICATION and OB_NOTIFICATION_EVENT tables. - * @param notificationCreationDTO Notification details DTO - * @return For successful request the API will return a JSON with the notificationID - */ - EventCreationResponse publishOBEvent(NotificationCreationDTO notificationCreationDTO); - -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventNotificationPersistenceServiceHandler.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventNotificationPersistenceServiceHandler.java deleted file mode 100644 index 720d89a8..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventNotificationPersistenceServiceHandler.java +++ /dev/null @@ -1,61 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.handler; - -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationCreationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.response.EventCreationResponse; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import net.minidev.json.JSONObject; - -/** - * Handler class for persisting event notifications to the database. - */ -public class EventNotificationPersistenceServiceHandler { - private static EventNotificationPersistenceServiceHandler instance = - new EventNotificationPersistenceServiceHandler(); - private DefaultEventCreationServiceHandler defaultEventCreationServiceHandler; - - private EventNotificationPersistenceServiceHandler() { - this.defaultEventCreationServiceHandler = EventNotificationServiceUtil.getDefaultEventCreationServiceHandler(); - } - - public static EventNotificationPersistenceServiceHandler getInstance() { - return instance; - } - - /** - * This method is to persist authorization revoke event. - * - * @param clientId - client ID - * @param resourceId - resource ID - * @param notificationType - notification type - * @param notificationInfo - notification info - * @return EventCreationResponse - */ - public EventCreationResponse persistRevokeEvent(String clientId, - String resourceId, - String notificationType, JSONObject notificationInfo) { - NotificationCreationDTO notificationCreationDTO = - new NotificationCreationDTO(); - notificationCreationDTO.setClientId(clientId); - notificationCreationDTO.setResourceId(resourceId); - notificationCreationDTO.setEventPayload(notificationType, notificationInfo); - return defaultEventCreationServiceHandler.publishOBEvent(notificationCreationDTO); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventPollingServiceHandler.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventPollingServiceHandler.java deleted file mode 100644 index 30a02dfe..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventPollingServiceHandler.java +++ /dev/null @@ -1,48 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.handler; - -import com.wso2.openbanking.accelerator.event.notifications.service.dto.EventPollingDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.response.EventPollingResponse; -import net.minidev.json.JSONObject; - -/** - * EventPolling Service handler is used to validate and map the polling request to the DTO before calling the - * polling service. For custom validations this class can be extended and the extended class - * can be added to the deployment.toml under event_polling_handler to execute the specific class. - */ -public interface EventPollingServiceHandler { - /** - * This method follows the IETF Specification for SET delivery over HTTP. - * The method supports event acknowledgment in both positive and negative. - * Also, can be used to POLL for available OPEN notifications. - * @param eventPollingRequest JSON request for event polling - * @return EventPollingResponse to the polling endpoint. - */ - EventPollingResponse pollEvents(JSONObject eventPollingRequest); - - /** - * This method is used to map the eventPollingRequest to EventPollingDTO. - * @param eventPollingRequest JSON request for event polling - * @return eventPollingDTO with the request parameters. - */ - EventPollingDTO mapPollingRequest(JSONObject eventPollingRequest); - -} - diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventSubscriptionServiceHandler.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventSubscriptionServiceHandler.java deleted file mode 100644 index ad161311..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventSubscriptionServiceHandler.java +++ /dev/null @@ -1,94 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.handler; - -import com.wso2.openbanking.accelerator.event.notifications.service.dto.EventSubscriptionDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.model.EventSubscription; -import com.wso2.openbanking.accelerator.event.notifications.service.response.EventSubscriptionResponse; -import net.minidev.json.JSONObject; - -/** - * EventSubscription Service handler is used to validate subscription requests before calling the - * subscription service. For custom validations this class can be extended and the extended class - * can be added to the deployment.toml under event_subscription_handler to execute the specific class. - */ -public interface EventSubscriptionServiceHandler { - - /** - * This method is used to create event subscriptions in the accelerator database. The method is a generic - * method that is used to persist data into the NOTIFICATION_SUBSCRIPTION and NOTIFICATION_SUBSCRIPTION_EVENT - * tables. - * - * @param eventSubscriptionRequestDto The request DTO that contains the subscription details. - * @return For successful request the API will return a JSON with the subscriptionId - */ - EventSubscriptionResponse createEventSubscription(EventSubscriptionDTO eventSubscriptionRequestDto); - - /** - * This method is used to retrieve an event subscription by its subscription ID. - * - * @param clientId The client ID of the subscription. - * @param subscriptionId The subscription ID of the subscription. - * @return For successful request the API will return a JSON with the retrieved Subscription. - */ - EventSubscriptionResponse getEventSubscription(String clientId, String subscriptionId); - - /** - * This method is used to retrieve all event subscriptions of a client. - * - * @param clientId The client ID of the subscription. - * @return For successful request the API will return a JSON with the retrieved Subscriptions. - */ - EventSubscriptionResponse getAllEventSubscriptions(String clientId); - - /** - * This method is used to retrieve all event subscriptions by event type. - * - * @param clientId The client ID of the subscription. - * @param eventType The event type that needs to be subscribed by the retrieving subscriptions. - * @return For successful request the API will return a JSON with the retrieved Subscriptions. - */ - EventSubscriptionResponse getEventSubscriptionsByEventType(String clientId, String eventType); - - /** - * This method is used to update an event subscription. - * - * @param eventSubscriptionUpdateRequestDto The request DTO that contains the updating subscription details. - * @return For successful request the API will return a JSON with the updated Subscription. - */ - EventSubscriptionResponse updateEventSubscription(EventSubscriptionDTO eventSubscriptionUpdateRequestDto); - - /** - * This method is used to delete an event subscription. - * - * @param clientId The client ID of the subscription. - * @param subscriptionId The subscription ID of the subscription. - * @return For successful request the API will an OK response. - */ - EventSubscriptionResponse deleteEventSubscription(String clientId, String subscriptionId); - - /** - * This method is used to create the response JSON object from the event subscription model. - * - * @param eventSubscription The event subscription model. - * @return JSONObject - */ - JSONObject mapSubscriptionModelToResponseJson(EventSubscription eventSubscription); - -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/internal/EventNotificationComponent.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/internal/EventNotificationComponent.java deleted file mode 100644 index 7b813677..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/internal/EventNotificationComponent.java +++ /dev/null @@ -1,102 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.internal; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.event.notifications.service.realtime.service.RealtimeEventNotificationLoaderService; -import com.wso2.openbanking.accelerator.event.notifications.service.realtime.util. - activator.PeriodicalEventNotificationConsumerJobActivator; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.osgi.service.component.ComponentContext; -import org.osgi.service.component.annotations.Activate; -import org.osgi.service.component.annotations.Component; -import org.osgi.service.component.annotations.Reference; -import org.osgi.service.component.annotations.ReferenceCardinality; -import org.osgi.service.component.annotations.ReferencePolicy; -import org.wso2.carbon.identity.oauth2.OAuth2Service; - -/** - * The Component class for activating event notification osgi service. - */ -@Component( - name = "com.wso2.openbanking.accelerator.event.notifications.service.internal.EventNotificationComponent", - immediate = true) -public class EventNotificationComponent { - private static Log log = LogFactory.getLog(EventNotificationComponent.class); - - @Activate - protected void activate(ComponentContext context) { - if (log.isDebugEnabled()) { - log.debug("Event Notification Service Component Activated"); - } - - // Check if realtime event notification enabled - if (OpenBankingConfigParser.getInstance().isRealtimeEventNotificationEnabled()) { - /* - * Initialize the blocking queue for storing the realtime event notifications - * Initialize the quartz job for consuming the realtime event notifications - * Initialize the thread for producing the open state realtime event notifications - */ - new Thread(new RealtimeEventNotificationLoaderService()).start(); - new PeriodicalEventNotificationConsumerJobActivator().activate(); - } - } - - /** - * Setters for the descendent OSGI services of the EventNotificationComponent. - * This is added to run the EventNotification OSGI component after the Common module - * @param openBankingConfigurationService OpenBankingConfigurationService - */ - @Reference( - service = OpenBankingConfigurationService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetConfigService" - ) - public void setConfigService(OpenBankingConfigurationService openBankingConfigurationService) { - EventNotificationDataHolder.getInstance().setOpenBankingConfigurationService(openBankingConfigurationService); - } - - public void unsetConfigService(OpenBankingConfigurationService openBankingConfigurationService) { - EventNotificationDataHolder.getInstance().setOpenBankingConfigurationService(null); - } - - /** - * Setters for the descendent OSGI services of the EventNotificationComponent. - * This is added to run the EventNotification OSGI component after the OAuth2Service - */ - @Reference( - service = OAuth2Service.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetOAuth2Service" - ) - - /** - * Setters for the descendent OSGI services of the EventNotificationComponent. - * @param oAuth2Service OAuth2Service - */ - public void setOAuth2Service(OAuth2Service oAuth2Service) { - } - - public void unsetOAuth2Service(OAuth2Service oAuth2Service) { - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/internal/EventNotificationDataHolder.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/internal/EventNotificationDataHolder.java deleted file mode 100644 index 36317c3a..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/internal/EventNotificationDataHolder.java +++ /dev/null @@ -1,75 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.internal; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.event.notifications.service.realtime.model.RealtimeEventNotification; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.util.concurrent.LinkedBlockingQueue; - -/** - * Data holder for Open Banking Event Notifications. - */ -public class EventNotificationDataHolder { - private static Log log = LogFactory.getLog(EventNotificationDataHolder.class); - private static volatile EventNotificationDataHolder instance; - private volatile LinkedBlockingQueue realtimeEventNotificationQueue; - private OpenBankingConfigurationService openBankingConfigurationService; - - private EventNotificationDataHolder() { - this.realtimeEventNotificationQueue = new LinkedBlockingQueue<>(); - } - - /** - * Return a singleton instance of the data holder. - * - * @return A singleton instance of the data holder - */ - public static synchronized EventNotificationDataHolder getInstance() { - if (instance == null) { - synchronized (EventNotificationDataHolder.class) { - if (instance == null) { - instance = new EventNotificationDataHolder(); - } - } - } - return instance; - } - - public LinkedBlockingQueue getRealtimeEventNotificationQueue() { - return realtimeEventNotificationQueue; - } - - public OpenBankingConfigurationService getOpenBankingConfigurationService() { - - return openBankingConfigurationService; - } - - public void setOpenBankingConfigurationService( - OpenBankingConfigurationService openBankingConfigurationService) { - - this.openBankingConfigurationService = openBankingConfigurationService; - } - - public void setRealtimeEventNotificationQueue(LinkedBlockingQueue queue) { - this.realtimeEventNotificationQueue = queue; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/AggregatedPollingResponse.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/AggregatedPollingResponse.java deleted file mode 100644 index f94cce75..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/AggregatedPollingResponse.java +++ /dev/null @@ -1,64 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.model; - -import java.util.HashMap; -import java.util.Map; - -/** - * Default Polling Response Implementation. - */ -public class AggregatedPollingResponse { - - private Map sets = new HashMap<>(); - - //For more available parameter - private int count = 0; - - private String status; - - public String getStatus() { - return status; - } - - public void setStatus(String status) { - this.status = status; - } - - public Map getSets() { - return sets; - } - - public void setSets(Map sets) { - this.sets = sets; - } - - public int getCount() { - return count; - } - - public void setCount(int count) { - this.count = count; - } - - public Boolean isMoreAvailable() { - return count > 0 ? true : false; - } - -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/EventSubscription.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/EventSubscription.java deleted file mode 100644 index 0909f53c..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/EventSubscription.java +++ /dev/null @@ -1,99 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.model; - -import java.util.List; - -/** - * This is the Event Subscription Model. - */ -public class EventSubscription { - private String subscriptionId = null; - private String clientId = null; - private String callbackUrl = null; - private Long timeStamp = null; - private String specVersion = null; - private String status = null; - private List eventTypes = null; - private String requestData = null; - - public String getSubscriptionId() { - return subscriptionId; - } - - public void setSubscriptionId(String subscriptionId) { - this.subscriptionId = subscriptionId; - } - - public String getClientId() { - return clientId; - } - - public void setClientId(String clientId) { - this.clientId = clientId; - } - - public String getCallbackUrl() { - return callbackUrl; - } - - public void setCallbackUrl(String callbackUrl) { - this.callbackUrl = callbackUrl; - } - - public Long getTimeStamp() { - return timeStamp; - } - - public void setTimeStamp(Long timeStamp) { - this.timeStamp = timeStamp; - } - - public String getSpecVersion() { - return specVersion; - } - - public void setSpecVersion(String specVersion) { - this.specVersion = specVersion; - } - - public String getStatus() { - return status; - } - - public void setStatus(String status) { - this.status = status; - } - - public List getEventTypes() { - return eventTypes; - } - - public void setEventTypes(List eventTypes) { - this.eventTypes = eventTypes; - } - - public String getRequestData() { - return requestData; - } - - public void setRequestData(String requestData) { - this.requestData = requestData; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/Notification.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/Notification.java deleted file mode 100644 index 1a7aacf6..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/Notification.java +++ /dev/null @@ -1,126 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.model; - -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.ObjectMapper; -import com.nimbusds.jose.JOSEException; -import net.minidev.json.JSONObject; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; - -import java.io.IOException; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -/** - * This is the notification model. - */ -public class Notification { - private String iss = null; - private Long iat = null; - private String jti = null; - private String sub = null; - private String aud = null; - private String txn = null; - private Long toe = null; - private Map events = new HashMap(); - - public Long getIat() { - return iat; - } - - public void setIat(Long iat) { - this.iat = iat; - } - - public String getJti() { - return jti; - } - - public void setJti(String jti) { - this.jti = jti; - } - - public String getSub() { - return sub; - } - - public void setSub(String sub) { - this.sub = sub; - } - - public String getAud() { - return aud; - } - - public void setAud(String aud) { - this.aud = aud; - } - - public String getTxn() { - return txn; - } - - public void setTxn(String txn) { - this.txn = txn; - } - - public Long getToe() { - return toe; - } - - public void setToe(Long toe) { - this.toe = toe; - } - - public Map getEvents() { - return events; - } - - public void setEvents(List eventsList) { - - for (NotificationEvent notificationEvent : eventsList) { - this.events.put(notificationEvent.getEventType(), notificationEvent.getEventInformation()); - } - } - - public String getIss() { - return iss; - } - - public void setIss(String iss) { - this.iss = iss; - } - - /** - * This method is to convert the class to a JSONObject. - * @param notification Notification - * @return JSONObject - * @throws IOException IOException when converting the class to JSONObject - * @throws JOSEException JOSEException when converting the class to JSONObject - * @throws IdentityOAuth2Exception IdentityOAuth2Exception when converting the class to JSONObject - */ - public static JsonNode getJsonNode(Notification notification) - throws IOException, JOSEException, IdentityOAuth2Exception { - ObjectMapper objectMapper = new ObjectMapper(); - return objectMapper.convertValue(notification, JsonNode.class); - } - -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/NotificationError.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/NotificationError.java deleted file mode 100644 index 12938666..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/NotificationError.java +++ /dev/null @@ -1,52 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.model; - -/** - * The notification error model. - */ -public class NotificationError { - private String notificationId = null; - private String errorCode = null; - private String errorDescription = null; - - public String getNotificationId() { - return notificationId; - } - - public void setNotificationId(String notificationId) { - this.notificationId = notificationId; - } - - public String getErrorCode() { - return errorCode; - } - - public void setErrorCode(String errorCode) { - this.errorCode = errorCode; - } - - public String getErrorDescription() { - return errorDescription; - } - - public void setErrorDescription(String errorDescription) { - this.errorDescription = errorDescription; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/NotificationEvent.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/NotificationEvent.java deleted file mode 100644 index beb444d0..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/model/NotificationEvent.java +++ /dev/null @@ -1,64 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.model; - -import net.minidev.json.JSONObject; - -/** - * This is the notification even model class. - */ -public class NotificationEvent { - - private Integer eventId = null; - private String notificationId = null; - private String eventType = null; - private JSONObject eventInformation; - - public Integer getEventId() { - return eventId; - } - - public void setEventId(Integer eventId) { - this.eventId = eventId; - } - - public String getNotificationId() { - return notificationId; - } - - public void setNotificationId(String notificationId) { - this.notificationId = notificationId; - } - - public String getEventType() { - return eventType; - } - - public void setEventType(String eventType) { - this.eventType = eventType; - } - - public JSONObject getEventInformation() { - return eventInformation; - } - - public void setEventInformation(JSONObject eventInformation) { - this.eventInformation = eventInformation; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/persistence/EventPollingStoreInitializer.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/persistence/EventPollingStoreInitializer.java deleted file mode 100644 index a682b49a..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/persistence/EventPollingStoreInitializer.java +++ /dev/null @@ -1,76 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.persistence; - -import com.wso2.openbanking.accelerator.common.persistence.JDBCPersistenceManager; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.AggregatedPollingDAO; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.AggregatedPollingDAOImpl; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.MSSQLNotificationPollingSqlStatements; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.NotificationPollingSqlStatements; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.PostgreSqlPollingDAOImpl; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.SQLException; - -/** - * Initializer Class for EventPolling Service DB. - */ -@Generated(message = "Datastore initializer classes") -public class EventPollingStoreInitializer { - - private static Log log = LogFactory.getLog(EventPollingStoreInitializer.class); - private static final String MYSQL = "MySQL"; - private static final String POSTGRE = "PostgreSQL"; - private static final String MSSQL = "Microsoft"; - private static final String ORACLE = "Oracle"; - private static final String H2 = "h2"; - - public static AggregatedPollingDAO initializeAggregatedPollingDAO() throws OBEventNotificationException { - - AggregatedPollingDAO aggregatedPollingDAO; - try (Connection connection = JDBCPersistenceManager.getInstance().getDBConnection()) { - String driverName = connection.getMetaData().getDriverName(); - - if (driverName.contains(MYSQL) || driverName.contains(H2)) { - aggregatedPollingDAO = new AggregatedPollingDAOImpl(new NotificationPollingSqlStatements()); - } else if (driverName.contains(POSTGRE)) { - aggregatedPollingDAO = new PostgreSqlPollingDAOImpl(new NotificationPollingSqlStatements()); - } else if (driverName.contains(MSSQL)) { - aggregatedPollingDAO = new PostgreSqlPollingDAOImpl(new MSSQLNotificationPollingSqlStatements()); - } else if (driverName.contains(ORACLE)) { - aggregatedPollingDAO = new PostgreSqlPollingDAOImpl(new MSSQLNotificationPollingSqlStatements()); - } else { - throw new OBEventNotificationException("Unhandled DB driver: " + driverName + " detected"); - } - - } catch (SQLException e) { - throw new OBEventNotificationException("Error while getting the database connection : ", e); - } - return aggregatedPollingDAO; - } - - public static AggregatedPollingDAO getAggregatedPollingDAO() throws OBEventNotificationException { - - return initializeAggregatedPollingDAO(); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/persistence/EventPublisherStoreInitializer.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/persistence/EventPublisherStoreInitializer.java deleted file mode 100644 index 0a2d9938..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/persistence/EventPublisherStoreInitializer.java +++ /dev/null @@ -1,74 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.persistence; - -import com.wso2.openbanking.accelerator.common.persistence.JDBCPersistenceManager; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.EventPublisherDAO; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.EventPublisherDAOImpl; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.NotificationPublisherSqlStatements; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.SQLException; - -/** - * Initialize DB for Event Creation. - */ -@Generated(message = "Datastore initializer classes") -public class EventPublisherStoreInitializer { - - private static Log log = LogFactory.getLog(EventPublisherStoreInitializer.class); - private static final String MYSQL = "MySQL"; - private static final String POSTGRE = "PostgreSQL"; - private static final String MSSQL = "Microsoft"; - private static final String ORACLE = "Oracle"; - private static final String H2 = "h2"; - - public static EventPublisherDAO initializePublisherDAO() throws OBEventNotificationException { - - EventPublisherDAO eventPublisherDAO; - try (Connection connection = JDBCPersistenceManager.getInstance().getDBConnection()) { - String driverName = connection.getMetaData().getDriverName(); - - if (driverName.contains(MYSQL) || driverName.contains(H2)) { - eventPublisherDAO = new EventPublisherDAOImpl(new NotificationPublisherSqlStatements()); - } else if (driverName.contains(POSTGRE)) { - eventPublisherDAO = new EventPublisherDAOImpl(new NotificationPublisherSqlStatements()); - } else if (driverName.contains(MSSQL)) { - eventPublisherDAO = new EventPublisherDAOImpl(new NotificationPublisherSqlStatements()); - } else if (driverName.contains(ORACLE)) { - eventPublisherDAO = new EventPublisherDAOImpl(new NotificationPublisherSqlStatements()); - } else { - throw new OBEventNotificationException("Unhandled DB driver: " + driverName + " detected"); - } - } catch (SQLException e) { - throw new OBEventNotificationException("Error while getting the database connection : ", e); - } - - return eventPublisherDAO; - } - - public static EventPublisherDAO getEventCreationDao() throws OBEventNotificationException { - - return initializePublisherDAO(); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/persistence/EventSubscriptionStoreInitializer.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/persistence/EventSubscriptionStoreInitializer.java deleted file mode 100644 index a3d040bf..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/persistence/EventSubscriptionStoreInitializer.java +++ /dev/null @@ -1,75 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.persistence; - -import com.wso2.openbanking.accelerator.common.persistence.JDBCPersistenceManager; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.EventSubscriptionDAO; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.EventSubscriptionDAOImpl; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.EventSubscriptionSqlStatements; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.PostgreSqlEventSubscriptionDAOImpl; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.SQLException; - -/** - * Initializer Class for EventSubscription Service DB. - */ -@Generated(message = "Datastore initializer classes") -public class EventSubscriptionStoreInitializer { - - private static Log log = LogFactory.getLog(EventSubscriptionStoreInitializer.class); - private static final String MYSQL = "MySQL"; - private static final String POSTGRE = "PostgreSQL"; - private static final String MSSQL = "Microsoft"; - private static final String ORACLE = "Oracle"; - private static final String H2 = "h2"; - - public static EventSubscriptionDAO initializeSubscriptionDAO() throws OBEventNotificationException { - - EventSubscriptionDAO eventSubscriptionDao; - try (Connection connection = JDBCPersistenceManager.getInstance().getDBConnection()) { - String driverName = connection.getMetaData().getDriverName(); - - if (driverName.contains(MYSQL) || driverName.contains(H2)) { - eventSubscriptionDao = new EventSubscriptionDAOImpl(new EventSubscriptionSqlStatements()); - } else if (driverName.contains(POSTGRE)) { - eventSubscriptionDao = new PostgreSqlEventSubscriptionDAOImpl(new EventSubscriptionSqlStatements()); - } else if (driverName.contains(MSSQL)) { - eventSubscriptionDao = new EventSubscriptionDAOImpl(new EventSubscriptionSqlStatements()); - } else if (driverName.contains(ORACLE)) { - eventSubscriptionDao = new EventSubscriptionDAOImpl(new EventSubscriptionSqlStatements()); - } else { - throw new OBEventNotificationException("Unhandled DB driver: " + driverName + " detected"); - } - } catch (SQLException e) { - throw new OBEventNotificationException("Error while getting the database connection : ", e); - } - - return eventSubscriptionDao; - } - - public static EventSubscriptionDAO getEventSubscriptionDao() throws OBEventNotificationException { - - return initializeSubscriptionDAO(); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/model/RealtimeEventNotification.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/model/RealtimeEventNotification.java deleted file mode 100644 index 4ae5dae1..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/model/RealtimeEventNotification.java +++ /dev/null @@ -1,59 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.realtime.model; - -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.realtime.service.RealtimeEventNotificationRequestGenerator; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; - -/** - * Model class for real time event notifications. - */ -public class RealtimeEventNotification { - private String callbackUrl = null; - private String eventSET = null; // Security Event Token to hold the Event Notification Data - private NotificationDTO notificationDTO = null; - - public void setCallbackUrl(String callbackUrl) { - this.callbackUrl = callbackUrl; - } - - public void setEventSET(String notification) { - this.eventSET = notification; - } - - public void setNotificationDTO(NotificationDTO notificationDTO) { - this.notificationDTO = notificationDTO; - } - - public String getCallbackUrl() { - return callbackUrl; - } - - public String getJsonPayload() { - RealtimeEventNotificationRequestGenerator eventNotificationRequestGenerator = - EventNotificationServiceUtil.getRealtimeEventNotificationRequestGenerator(); - return eventNotificationRequestGenerator.getRealtimeEventNotificationPayload(notificationDTO, eventSET); - } - - public String getNotificationId() { - return notificationDTO.getNotificationId(); - } - -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/DefaultRealtimeEventNotificationRequestGenerator.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/DefaultRealtimeEventNotificationRequestGenerator.java deleted file mode 100644 index 7f9e90f1..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/DefaultRealtimeEventNotificationRequestGenerator.java +++ /dev/null @@ -1,40 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.realtime.service; - -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; - -import java.util.HashMap; -import java.util.Map; - -/** - * Default class for realtime event notification request generation. - * This is to generate the realtime event notification request payload and headers. - */ -public class DefaultRealtimeEventNotificationRequestGenerator implements RealtimeEventNotificationRequestGenerator { - @Override - public String getRealtimeEventNotificationPayload(NotificationDTO notificationDTO, String eventSET) { - return "{\"notificationId\": " + notificationDTO.getNotificationId() + ", \"SET\": " + eventSET + "}"; - } - - @Override - public Map getAdditionalHeaders() { - return new HashMap<>(); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/EventNotificationProducerService.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/EventNotificationProducerService.java deleted file mode 100644 index 72da33fb..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/EventNotificationProducerService.java +++ /dev/null @@ -1,81 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.realtime.service; - -import com.nimbusds.jose.JOSEException; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.internal.EventNotificationDataHolder; -import com.wso2.openbanking.accelerator.event.notifications.service.model.Notification; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationEvent; -import com.wso2.openbanking.accelerator.event.notifications.service.realtime.model.RealtimeEventNotification; -import com.wso2.openbanking.accelerator.event.notifications.service.service.EventNotificationGenerator; -import com.wso2.openbanking.accelerator.event.notifications.service.service.EventPollingService; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; - -import java.io.IOException; -import java.util.List; -import java.util.concurrent.LinkedBlockingQueue; - -/** - * This thread is used to produce the event notification and put it into the realtime event notification queue. - */ -public class EventNotificationProducerService implements Runnable { - private static final Log log = LogFactory.getLog(EventPollingService.class); - private final NotificationDTO notificationDTO; - private final List notificationEvents; - - public EventNotificationProducerService( - NotificationDTO notificationDTO, List notificationEvents) { - this.notificationDTO = notificationDTO; - this.notificationEvents = notificationEvents; - } - - @Override - public void run() { - String callbackUrl = EventNotificationServiceUtil.getCallbackURL(notificationDTO.getClientId()); - - LinkedBlockingQueue queue = EventNotificationDataHolder.getInstance(). - getRealtimeEventNotificationQueue(); - EventNotificationGenerator eventNotificationGenerator = EventNotificationServiceUtil. - getEventNotificationGenerator(); - RealtimeEventNotification realtimeEventNotification = new RealtimeEventNotification(); - realtimeEventNotification.setNotificationDTO(notificationDTO); - realtimeEventNotification.setCallbackUrl(callbackUrl); - - try { - Notification notification = eventNotificationGenerator.generateEventNotificationBody( - notificationDTO, notificationEvents); - realtimeEventNotification.setEventSET(eventNotificationGenerator.generateEventNotification( - Notification.getJsonNode(notification))); - - queue.put(realtimeEventNotification); // put the notification into the queue - } catch (InterruptedException e) { - log.error("Error when adding the Realtime Notification with notification ID " + - notificationDTO.getNotificationId() + " into the RealtimeEventNotification Queue", e); - } catch (OBEventNotificationException e) { - log.error("Error when generating the event notification", e); - } catch (IOException | JOSEException | IdentityOAuth2Exception e) { - log.error("Error while processing event notification JSON object", e); - } - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationLoaderService.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationLoaderService.java deleted file mode 100644 index f31ff3bf..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationLoaderService.java +++ /dev/null @@ -1,86 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.realtime.service; - -import com.nimbusds.jose.JOSEException; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.AggregatedPollingDAO; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.internal.EventNotificationDataHolder; -import com.wso2.openbanking.accelerator.event.notifications.service.model.Notification; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationEvent; -import com.wso2.openbanking.accelerator.event.notifications.service.persistence.EventPollingStoreInitializer; -import com.wso2.openbanking.accelerator.event.notifications.service.realtime.model.RealtimeEventNotification; -import com.wso2.openbanking.accelerator.event.notifications.service.service.EventNotificationGenerator; -import com.wso2.openbanking.accelerator.event.notifications.service.service.EventPollingService; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; - -import java.io.IOException; -import java.util.List; -import java.util.concurrent.LinkedBlockingQueue; - -/** - * This service is used to add open state event notifications to the realtime event notification queue. - * This service is called whenever the server starts. - */ -public class RealtimeEventNotificationLoaderService implements Runnable { - private static final Log log = LogFactory.getLog(EventPollingService.class); - - @Override - public void run() { - // Get all open state event notifications from the database and add them to the queue - try { - LinkedBlockingQueue queue = EventNotificationDataHolder.getInstance(). - getRealtimeEventNotificationQueue(); - AggregatedPollingDAO aggregatedPollingDAO = EventPollingStoreInitializer.getAggregatedPollingDAO(); - EventNotificationGenerator eventNotificationGenerator = EventNotificationServiceUtil. - getEventNotificationGenerator(); - List openNotifications = aggregatedPollingDAO.getNotificationsByStatus( - EventNotificationConstants.OPEN); - - for (NotificationDTO notificationDTO : openNotifications) { - //Get events by notificationId - List notificationEvents = aggregatedPollingDAO. - getEventsByNotificationID(notificationDTO.getNotificationId()); - - Notification responseNotification = eventNotificationGenerator. - generateEventNotificationBody(notificationDTO, notificationEvents); - - String callbackUrl = EventNotificationServiceUtil.getCallbackURL(notificationDTO.getClientId()); - - RealtimeEventNotification realtimeEventNotification = new RealtimeEventNotification(); - realtimeEventNotification.setCallbackUrl(callbackUrl); - realtimeEventNotification.setEventSET(eventNotificationGenerator.generateEventNotification( - Notification.getJsonNode(responseNotification))); - realtimeEventNotification.setNotificationDTO(notificationDTO); - queue.put(realtimeEventNotification); // put the notification into the queue - } - } catch (InterruptedException e) { - log.error("Error when adding the Realtime Notification into the RealtimeEventNotification Queue", e); - } catch (OBEventNotificationException e) { - log.error("Error when generating the event notification", e); - } catch (IOException | JOSEException | IdentityOAuth2Exception e) { - log.error("Error while processing event notification JSON object", e); - } - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationRequestGenerator.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationRequestGenerator.java deleted file mode 100644 index f9e6bf89..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationRequestGenerator.java +++ /dev/null @@ -1,48 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.realtime.service; - - -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; - -import java.util.Map; - -/** - * Interface for event notification request metadata generation. For custom class extensions the class name - * is to be referred from the realtime_event_notification_request_generator in deployment.toml - */ -public interface RealtimeEventNotificationRequestGenerator { - /** - * This method is to generate realtime event notification payload. To generate custom values - * for the body this method should be extended. - * - * @param notificationDTO Notification details DTO - * @param eventSET Event set - * @return String payload - */ - String getRealtimeEventNotificationPayload(NotificationDTO notificationDTO, String eventSET); - - /** - * This method is to generate realtime event notification request headers. To generate custom values - * for the body this method should be extended. - * - * @return Map of headers - */ - Map getAdditionalHeaders(); -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationSenderService.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationSenderService.java deleted file mode 100644 index 8b2ae6c9..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationSenderService.java +++ /dev/null @@ -1,195 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.realtime.service; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.HTTPClientUtils; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.AggregatedPollingDAO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.internal.EventNotificationComponent; -import com.wso2.openbanking.accelerator.event.notifications.service.persistence.EventPollingStoreInitializer; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.HttpResponse; -import org.apache.http.HttpStatus; -import org.apache.http.client.config.RequestConfig; -import org.apache.http.client.methods.HttpPost; -import org.apache.http.entity.ContentType; -import org.apache.http.entity.StringEntity; -import org.apache.http.impl.client.CloseableHttpClient; - -import java.io.IOException; -import java.net.URI; -import java.time.Duration; -import java.time.LocalTime; -import java.util.Map; - -/** - * This method is used to send the HTTP requests to the TPP provided callback URL. - * Exponential backoff and Circuit breaker based retry policy is used to retry failed POST requests. - */ -public class RealtimeEventNotificationSenderService implements Runnable { - - private static final Log log = LogFactory.getLog(EventNotificationComponent.class); - - private static final OpenBankingConfigParser configParser = OpenBankingConfigParser.getInstance(); - private static final int MAX_RETRIES = configParser.getRealtimeEventNotificationMaxRetries(); - private static final int INITIAL_BACKOFF_TIME_IN_SECONDS = - configParser.getRealtimeEventNotificationInitialBackoffTimeInSeconds(); - private static final String BACKOFF_FUNCTION = configParser.getRealtimeEventNotificationBackoffFunction(); - private static final int CIRCUIT_BREAKER_OPEN_TIMEOUT_IN_SECONDS = - configParser.getRealtimeEventNotificationCircuitBreakerOpenTimeoutInSeconds(); - private static final int TIMEOUT_IN_SECONDS = configParser.getRealtimeEventNotificationTimeoutInSeconds(); - - private CloseableHttpClient httpClient; - private RealtimeEventNotificationRequestGenerator httpRequestGenerator; - private String notificationId; - private String callbackUrl; - private String payloadJson; - - public RealtimeEventNotificationSenderService(String callbackUrl, String payloadJson, - String notificationId) { - try { - this.httpClient = HTTPClientUtils.getRealtimeEventNotificationHttpsClient(); - } catch (OpenBankingException e) { - log.error("Failed to initialize the HTTP client for the realtime event notification", e); - } - this.httpRequestGenerator = EventNotificationServiceUtil.getRealtimeEventNotificationRequestGenerator(); - this.notificationId = notificationId; - this.callbackUrl = callbackUrl; - this.payloadJson = payloadJson; - } - - public void run() { - try { - postWithRetry(); - } catch (OBEventNotificationException e) { - log.error("Failed to send the Real-time event notification with notificationId: " - + notificationId, e); - } - } - - /** - * This method is used to send the HTTP requests to the TPP provided callback URL. - * Exponential backoff and Circuit breaker based retry policy is used to retry failed POST requests. - * - * @throws OBEventNotificationException - */ - private void postWithRetry() throws OBEventNotificationException { - AggregatedPollingDAO aggregatedPollingDAO = - EventPollingStoreInitializer.getAggregatedPollingDAO(); - int retryCount = 0; - long backoffTimeMs = INITIAL_BACKOFF_TIME_IN_SECONDS * 1000L; - boolean circuitBreakerOpen = false; - LocalTime startTime = LocalTime.now(); - - while (retryCount <= MAX_RETRIES && !circuitBreakerOpen) { - try { - // This if closure will execute only if the initial POST request is failed. - // This includes the retry policy and will execute according to the configurations. - if (retryCount > 0) { - if (log.isDebugEnabled()) { - log.debug("HTTP request Retry #" + retryCount + " - waiting for " - + backoffTimeMs + " ms before trying again"); - } - Thread.sleep(backoffTimeMs); - - switch (BACKOFF_FUNCTION) { - case "CONSTANT": - // Backoff time will not be changed - // Retries will happen in constant time frames - break; - case "LINEAR": - // Backoff time will be doubled after each retry - // nextWaitingTime = 2 x previousWaitingTime - backoffTimeMs *= 2; - break; - case "EX": - // Backoff time will be increased exponentially - // nextWaitingTime = startWaitingTime x e^(retryCount) - backoffTimeMs = (long) - (INITIAL_BACKOFF_TIME_IN_SECONDS - * 1000 * Math.exp(retryCount)); - break; - default: - log.error("Invalid backoff function for the realtime event notification retry policy: " - + BACKOFF_FUNCTION); - throw new IllegalArgumentException( - "Invalid backoff function for the realtime event notification retry policy: " - + BACKOFF_FUNCTION); - } - } - - HttpPost httpPost = new HttpPost(URI.create(callbackUrl)); - - for (Map.Entry entry : httpRequestGenerator.getAdditionalHeaders().entrySet()) { - String headerName = entry.getKey(); - String headerValue = entry.getValue(); - httpPost.setHeader(headerName, headerValue); - } - - httpPost.setEntity(new StringEntity(payloadJson, ContentType.APPLICATION_JSON)); - RequestConfig requestConfig = RequestConfig.custom() - .setConnectTimeout(TIMEOUT_IN_SECONDS * 1000) - .setConnectionRequestTimeout(TIMEOUT_IN_SECONDS * 1000) - .setSocketTimeout(TIMEOUT_IN_SECONDS * 1000) - .build(); - httpPost.setConfig(requestConfig); - - HttpResponse response = httpClient.execute(httpPost); - int statusCode = response.getStatusLine().getStatusCode(); - if (statusCode == HttpStatus.SC_OK) { - if (log.isDebugEnabled()) { - log.debug("Real-time event notification with notificationId: " + notificationId - + " sent successfully"); - } - aggregatedPollingDAO.updateNotificationStatusById(notificationId, EventNotificationConstants.ACK); - return; - } else { - if (log.isDebugEnabled()) { - log.debug("Real-time event notification with notificationId: " + notificationId - + " sent failed with status code: " + statusCode); - } - } - } catch (IOException | InterruptedException e) { - log.error("Real-time event notification with notificationId: " + notificationId - + " sent failed" + e); - } - - // Circuit breaker will be opened if the retrying time exceeds the configured circuit breaker timeout. - if (Duration.between(startTime, LocalTime.now()).toMillis() - > CIRCUIT_BREAKER_OPEN_TIMEOUT_IN_SECONDS * 1000) { - circuitBreakerOpen = true; - if (log.isDebugEnabled()) { - log.debug("Circuit breaker open for the realtime event notification with notificationId: " - + notificationId); - } - } - retryCount++; - } - - // If the circuit breaker is opened or the maximum retry count is exceeded, - // the notification status will be updated as ERROR. - aggregatedPollingDAO.updateNotificationStatusById(notificationId, EventNotificationConstants.ERROR); - - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/util/activator/PeriodicalEventNotificationConsumerJobActivator.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/util/activator/PeriodicalEventNotificationConsumerJobActivator.java deleted file mode 100644 index 4fcc8bed..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/util/activator/PeriodicalEventNotificationConsumerJobActivator.java +++ /dev/null @@ -1,92 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.realtime.util.activator; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.event.notifications.service.realtime.util.job.EventNotificationConsumerJob; -import com.wso2.openbanking.accelerator.event.notifications.service.realtime. - util.scheduler.PeriodicalEventNotificationConsumerJobScheduler; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.quartz.CronExpression; -import org.quartz.JobDetail; -import org.quartz.Scheduler; -import org.quartz.SchedulerException; -import org.quartz.SimpleScheduleBuilder; -import org.quartz.Trigger; - -import java.text.ParseException; -import java.util.Date; - -import static org.quartz.JobBuilder.newJob; -import static org.quartz.TriggerBuilder.newTrigger; - -/** - * Scheduled Task definition and trigger to perform realtime event notification sending based on the cron string. - */ -@Generated(message = "Excluding from code coverage") -public class PeriodicalEventNotificationConsumerJobActivator { - - private static Log log = LogFactory.getLog(PeriodicalEventNotificationConsumerJobActivator.class); - private static final String PERIODIC_CRON_EXPRESSION = OpenBankingConfigParser - .getInstance().getRealtimeEventNotificationSchedulerCronExpression(); - - public void activate() { - int cronInSeconds = 60; - - try { - CronExpression cron = new CronExpression(PERIODIC_CRON_EXPRESSION); - - Date nextValidTime = cron.getNextValidTimeAfter(new Date()); - Date secondValidTime = cron.getNextValidTimeAfter(nextValidTime); - - cronInSeconds = (int) (secondValidTime.getTime() - nextValidTime.getTime()) / 1000; - - } catch (ParseException e) { - log.error("Error while parsing the event notification scheduler cron expression : " - + PERIODIC_CRON_EXPRESSION, e); - } - - JobDetail job = newJob(EventNotificationConsumerJob.class) - .withIdentity("RealtimeEventNotificationJob", "group2") - .build(); - - Trigger trigger = newTrigger() - .withIdentity("periodicalEvenNotificationTrigger", "group2") - .withSchedule(SimpleScheduleBuilder.simpleSchedule() - .withIntervalInSeconds(cronInSeconds) - .repeatForever()) - .build(); - - try { - Scheduler scheduler = PeriodicalEventNotificationConsumerJobScheduler.getInstance().getScheduler(); - // this check is to remove already stored jobs in clustered mode. - if (scheduler.checkExists(job.getKey())) { - scheduler.deleteJob(job.getKey()); - } - - scheduler.scheduleJob(job, trigger); - log.info("Periodical Realtime Event Notification sender Started with cron : " - + PERIODIC_CRON_EXPRESSION); - } catch (SchedulerException e) { - log.error("Error while starting Periodical Realtime Event Notification sender", e); - } - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/util/job/EventNotificationConsumerJob.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/util/job/EventNotificationConsumerJob.java deleted file mode 100644 index d5591990..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/util/job/EventNotificationConsumerJob.java +++ /dev/null @@ -1,91 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.realtime.util.job; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.event.notifications.service.internal.EventNotificationDataHolder; -import com.wso2.openbanking.accelerator.event.notifications.service.realtime.model.RealtimeEventNotification; -import com.wso2.openbanking.accelerator.event.notifications.service.realtime.service.RealtimeEventNotificationSenderService; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.quartz.DisallowConcurrentExecution; -import org.quartz.Job; -import org.quartz.JobExecutionContext; -import org.quartz.JobExecutionException; - -import java.util.ArrayList; -import java.util.concurrent.ExecutorService; -import java.util.concurrent.Executors; -import java.util.concurrent.LinkedBlockingQueue; - -/** - * Scheduled Task to send realtime event notifications to callback Urls. - * This task is scheduled to run periodically. - * This task consumes all the notifications in the queue and send them to the callback urls. - */ -@Generated(message = "Excluding from code coverage") -@DisallowConcurrentExecution -public class EventNotificationConsumerJob implements Job { - - private static final Log log = LogFactory.getLog(EventNotificationConsumerJob.class); - private static final int THREAD_POOL_SIZE = OpenBankingConfigParser - .getInstance().getEventNotificationThreadpoolSize(); - - @Override - public void execute(JobExecutionContext jobExecutionContext) throws JobExecutionException { - ArrayList notifications = consumeNotifications(); - // send notifications to the callback urls - int threads = Math.min(notifications.size(), THREAD_POOL_SIZE); - int threadPoolSize = Math.max(threads, 2); - - ExecutorService executor = Executors.newFixedThreadPool(threadPoolSize); - - for (RealtimeEventNotification notification : notifications) { - String callbackUrl = notification.getCallbackUrl(); - String payload = notification.getJsonPayload(); - Runnable worker = new RealtimeEventNotificationSenderService(callbackUrl, - payload, notification.getNotificationId()); - executor.execute(worker); - } - - executor.shutdown(); - while (!executor.isTerminated()) { } - } - - private static ArrayList consumeNotifications() { - - LinkedBlockingQueue queue = EventNotificationDataHolder.getInstance() - .getRealtimeEventNotificationQueue(); - ArrayList notifications = new ArrayList<>(); - - // consume all notifications in the queue - int key = 0; - while (!queue.isEmpty() && key < THREAD_POOL_SIZE) { - key++; - try { - RealtimeEventNotification notification = queue.take(); - notifications.add(notification); - } catch (InterruptedException ex) { - log.error("Error while consuming notifications from the event notification queue", ex); - } - } - return notifications; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/util/scheduler/PeriodicalEventNotificationConsumerJobScheduler.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/util/scheduler/PeriodicalEventNotificationConsumerJobScheduler.java deleted file mode 100644 index 2b58cd55..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/util/scheduler/PeriodicalEventNotificationConsumerJobScheduler.java +++ /dev/null @@ -1,77 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.realtime.util.scheduler; - -import com.wso2.openbanking.accelerator.common.util.Generated; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.quartz.Scheduler; -import org.quartz.SchedulerException; -import org.quartz.impl.StdSchedulerFactory; - -/** - * Periodic realtime event notification job scheduler class. - * This class initialize the scheduler and schedule configured jobs and triggers. - */ -@Generated(message = "Excluding from code coverage") -public class PeriodicalEventNotificationConsumerJobScheduler { - private static volatile PeriodicalEventNotificationConsumerJobScheduler instance; - private static volatile Scheduler scheduler; - private static Log log = LogFactory.getLog(PeriodicalEventNotificationConsumerJobScheduler.class); - - private PeriodicalEventNotificationConsumerJobScheduler() { - initScheduler(); - } - - public static synchronized PeriodicalEventNotificationConsumerJobScheduler getInstance() { - - if (instance == null) { - synchronized (PeriodicalEventNotificationConsumerJobScheduler.class) { - if (instance == null) { - instance = new PeriodicalEventNotificationConsumerJobScheduler(); - } - } - } - return instance; - } - - private void initScheduler() { - - if (instance != null) { - return; - } - synchronized (PeriodicalEventNotificationConsumerJobScheduler.class) { - try { - scheduler = StdSchedulerFactory.getDefaultScheduler(); - scheduler.start(); - } catch (SchedulerException e) { - log.error("Exception while initializing the Real-time Event notification scheduler", e); - } - } - } - - /** - * Returns the scheduler. - * - * @return Scheduler scheduler. - */ - public Scheduler getScheduler() { - return scheduler; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/response/EventCreationResponse.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/response/EventCreationResponse.java deleted file mode 100644 index 658cad48..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/response/EventCreationResponse.java +++ /dev/null @@ -1,55 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.response; - -import net.minidev.json.JSONObject; - -/** - * This class is to pass the event creation response to the api endpoint. - */ -public class EventCreationResponse { - - private String status; - private JSONObject responseBody; - private String errorResponse; - - public String getErrorResponse() { - return errorResponse; - } - - public void setErrorResponse(String errorResponse) { - this.errorResponse = errorResponse; - } - - public String getStatus() { - return status; - } - - public void setStatus(String status) { - this.status = status; - } - - public JSONObject getResponseBody() { - return responseBody; - } - - public void setResponseBody(JSONObject responseBody) { - this.responseBody = responseBody; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/response/EventPollingResponse.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/response/EventPollingResponse.java deleted file mode 100644 index af8b7ee0..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/response/EventPollingResponse.java +++ /dev/null @@ -1,55 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.response; - -import net.minidev.json.JSONObject; - -/** - * This class is used to map the Event Polling service response to the API response. - */ -public class EventPollingResponse { - - private String status; - private JSONObject responseBody; - private Object errorResponse; - - public String getStatus() { - return status; - } - - public void setStatus(String status) { - this.status = status; - } - - public JSONObject getResponseBody() { - return responseBody; - } - - public void setResponseBody(JSONObject responseBody) { - this.responseBody = responseBody; - } - - public Object getErrorResponse() { - return errorResponse; - } - - public void setErrorResponse(Object errorResponse) { - this.errorResponse = errorResponse; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/response/EventSubscriptionResponse.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/response/EventSubscriptionResponse.java deleted file mode 100644 index 327ff5f0..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/response/EventSubscriptionResponse.java +++ /dev/null @@ -1,54 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.response; - -/** - * This class is used to map the Event Subscription service response to the API response. - */ -public class EventSubscriptionResponse { - - private int status; - private Object responseBody; - private Object errorResponse; - - public int getStatus() { - return status; - } - - public void setStatus(int status) { - this.status = status; - } - - public Object getResponseBody() { - return responseBody; - } - - public void setResponseBody(Object responseBody) { - this.responseBody = responseBody; - } - - public Object getErrorResponse() { - return errorResponse; - } - - public void setErrorResponse(Object errorResponse) { - this.errorResponse = errorResponse; - } - -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/DefaultEventNotificationGenerator.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/DefaultEventNotificationGenerator.java deleted file mode 100644 index 5303ae05..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/DefaultEventNotificationGenerator.java +++ /dev/null @@ -1,87 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.service; - -import com.fasterxml.jackson.databind.JsonNode; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.Notification; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationEvent; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.registry.core.utils.UUIDGenerator; - -import java.time.Instant; -import java.util.List; - -/** - * Default Event Notification Response Generator Class. - */ -public class DefaultEventNotificationGenerator implements EventNotificationGenerator { - - private static Log log = LogFactory.getLog(DefaultEventNotificationGenerator.class); - - @Override - public Notification generateEventNotificationBody(NotificationDTO notificationDTO, - List notificationEventList) - throws OBEventNotificationException { - - Notification notification = new Notification(); - //get current time in milliseconds - Long currentTime = Instant.now().getEpochSecond(); - - //generate transaction Identifier - String transactionIdentifier = UUIDGenerator.generateUUID(); - - notification.setIss(OpenBankingConfigParser.getInstance().getEventNotificationTokenIssuer()); - notification.setIat(currentTime); - notification.setAud(notificationDTO.getClientId()); - notification.setJti(notificationDTO.getNotificationId()); - notification.setTxn(transactionIdentifier); - notification.setToe(notificationDTO.getUpdatedTimeStamp()); - notification.setSub(generateSubClaim(notificationDTO)); - notification.setEvents(notificationEventList); - return notification; - } - - @Generated(message = "Excluded from tests as using a util method from a different package") - public String generateEventNotification(JsonNode jsonNode) - throws OBEventNotificationException { - - String payload = EventNotificationServiceUtil.getCustomNotificationPayload(jsonNode); - try { - return IdentityCommonUtil.signJWTWithDefaultKey(payload); - } catch (Exception e) { - log.error("Error while signing the JWT token", e); - throw new OBEventNotificationException("Error while signing the JWT token", e); - } - - } - - @Generated(message = "Private method tested when the used method is tested") - private String generateSubClaim(NotificationDTO notificationDTO) { - String sub = notificationDTO.getClientId(); - return sub; - } - -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventCreationService.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventCreationService.java deleted file mode 100644 index 3ccc63ec..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventCreationService.java +++ /dev/null @@ -1,121 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.service; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.util.DatabaseUtil; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.EventPublisherDAO; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationCreationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationEvent; -import com.wso2.openbanking.accelerator.event.notifications.service.persistence.EventPublisherStoreInitializer; -import com.wso2.openbanking.accelerator.event.notifications.service.realtime.service.EventNotificationProducerService; -import net.minidev.json.JSONObject; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.registry.core.utils.UUIDGenerator; - - -import java.sql.Connection; -import java.util.ArrayList; -import java.util.Map; - -/** - * This is the event creation service class. - */ -public class EventCreationService { - - private static Log log = LogFactory.getLog(EventCreationService.class); - - /** - * The publishOBEventNotification methods will call the dao layer to persist the event - * notifications for event polling request. - * @param notificationCreationDTO Notification Creation DTO - * @return Event Response - * @throws OBEventNotificationException Exception when persisting event notification data - */ - public String publishOBEventNotification(NotificationCreationDTO notificationCreationDTO) - throws OBEventNotificationException { - - Connection connection = DatabaseUtil.getDBConnection(); - NotificationDTO notification = getNotification(notificationCreationDTO); - ArrayList eventsList = getEvents(notificationCreationDTO.getEventPayload()); - - EventPublisherDAO eventPublisherDAO = EventPublisherStoreInitializer.getEventCreationDao(); - String eventResponse = null; - - try { - eventResponse = eventPublisherDAO.persistEventNotification(connection, notification, eventsList); - DatabaseUtil.commitTransaction(connection); - - // Check whether the real time event notification is enabled. - if (OpenBankingConfigParser.getInstance().isRealtimeEventNotificationEnabled()) { - new Thread(new EventNotificationProducerService(notification, eventsList)).start(); - } - return eventResponse; - } catch (OBEventNotificationException e) { - throw new OBEventNotificationException("Error when persisting event notification data", e); - } finally { - log.debug(EventNotificationConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - /** - * The getEvents method is used to get the NotificationEvents Array list from payload. - * - * @param notificationEvents Notification Events to convert - * @return Event notification List - */ - @Generated(message = "Private methods invoked when calling referred method") - private ArrayList getEvents(Map notificationEvents) { - - ArrayList eventsList = new ArrayList<>(); - notificationEvents.keySet().forEach(key -> { - Object eventInfo = notificationEvents.get(key); - NotificationEvent notificationEvent = new NotificationEvent(); - notificationEvent.setEventType(key); - notificationEvent.setEventInformation((JSONObject) eventInfo); - eventsList.add(notificationEvent); - }); - - return eventsList; - } - - /** - * The getNotification method is used to get the NotificationDAO from payload. - * - * @param notificationCreationDTO Notification Creation DTO - * @return Notification Details - */ - @Generated(message = "Private methods invoked when calling referred method") - private NotificationDTO getNotification(NotificationCreationDTO notificationCreationDTO) { - - NotificationDTO notification = new NotificationDTO(); - notification.setNotificationId(UUIDGenerator.generateUUID()); - notification.setClientId(notificationCreationDTO.getClientId()); - notification.setResourceId(notificationCreationDTO.getResourceId()); - notification.setStatus(EventNotificationConstants.OPEN); - - return notification; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventNotificationGenerator.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventNotificationGenerator.java deleted file mode 100644 index 8b0e50ea..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventNotificationGenerator.java +++ /dev/null @@ -1,48 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.service; - -import com.fasterxml.jackson.databind.JsonNode; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.Notification; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationEvent; - -import java.util.List; - -/** - * Interface for event notification generation. For custom class extensions the class name - * is to be referred from the event_notification_generator in deployment.toml - */ -public interface EventNotificationGenerator { - - /** - * This method is to generate event notification body. To generate custom values - * for the body this method should be extended. - * @param notificationDTO Notification details DTO - * @param notificationEventList List of notification events - * - * @return Event Notification Body - * @throws OBEventNotificationException Exception when generating event notification body - */ - Notification generateEventNotificationBody(NotificationDTO notificationDTO, List - notificationEventList) throws OBEventNotificationException; - - String generateEventNotification(JsonNode jsonNode) throws OBEventNotificationException; -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventPollingService.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventPollingService.java deleted file mode 100644 index 8490a9f8..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventPollingService.java +++ /dev/null @@ -1,153 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.service; - -import com.nimbusds.jose.JOSEException; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.AggregatedPollingDAO; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.EventPollingDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.AggregatedPollingResponse; -import com.wso2.openbanking.accelerator.event.notifications.service.model.Notification; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationError; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationEvent; -import com.wso2.openbanking.accelerator.event.notifications.service.persistence.EventPollingStoreInitializer; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; - -import java.io.IOException; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -/** - * This is the event polling service. - */ -public class EventPollingService { - - private static Log log = LogFactory.getLog(EventPollingService.class); - - /** - * The pollEvents methods will return the Aggregated Polling Response for - * event polling request. - * @param eventPollingDTO Event polling request DTO - * @return AggregatedPollingResponse Aggregated Polling Response - * @throws OBEventNotificationException Exception when polling events - */ - public AggregatedPollingResponse pollEvents(EventPollingDTO eventPollingDTO) - throws OBEventNotificationException { - - AggregatedPollingResponse aggregatedPollingResponse = new AggregatedPollingResponse(); - AggregatedPollingDAO aggregatedPollingDAO = EventPollingStoreInitializer.getAggregatedPollingDAO(); - - EventNotificationGenerator eventNotificationGenerator = EventNotificationServiceUtil. - getEventNotificationGenerator(); - - Map sets = new HashMap<>(); - - //Short polling - if (eventPollingDTO.getReturnImmediately()) { - - //Update notifications with ack - for (String notificationId : eventPollingDTO.getAck()) { - aggregatedPollingDAO.updateNotificationStatusById(notificationId, EventNotificationConstants.ACK); - } - - //Update notifications with err - for (Map.Entry entry: eventPollingDTO.getErrors().entrySet()) { - //Check if the notification is in OPEN status - if (aggregatedPollingDAO.getNotificationStatus(entry.getKey())) { - aggregatedPollingDAO.updateNotificationStatusById( - entry.getKey(), EventNotificationConstants.ERROR); - aggregatedPollingDAO.storeErrorNotification(entry.getValue()); - } - } - - //Retrieve notifications - int maxEvents = eventPollingDTO.getMaxEvents(); - - if (maxEvents == 0) { - aggregatedPollingResponse.setSets(sets); - aggregatedPollingResponse.setStatus(EventNotificationConstants.OK); - } else { - - int setsToReturn = OpenBankingConfigParser.getInstance().getNumberOfSetsToReturn(); - - List notificationList; - - if (maxEvents < setsToReturn) { - notificationList = aggregatedPollingDAO.getNotificationsByClientIdAndStatus( - eventPollingDTO.getClientId(), EventNotificationConstants.OPEN, maxEvents); - - } else { - notificationList = aggregatedPollingDAO.getNotificationsByClientIdAndStatus( - eventPollingDTO.getClientId(), EventNotificationConstants.OPEN, setsToReturn); - } - - if (notificationList.isEmpty()) { - if (log.isDebugEnabled()) { - log.debug(String.format("No OB Event Notifications available for for the client " + - "with ID : '%s'.", eventPollingDTO.getClientId().replaceAll("[\r\n]", ""))); - } - aggregatedPollingResponse.setStatus(EventNotificationConstants.NOT_FOUND); - } else { - if (log.isDebugEnabled()) { - log.debug(String.format("OB Event Notifications available for the client " + - "with ID : '%s'.", eventPollingDTO.getClientId().replaceAll("[\r\n]", ""))); - } - aggregatedPollingResponse.setStatus(EventNotificationConstants.OK); - - for (NotificationDTO notificationDTO : notificationList) { - - try { - //Get events by notificationId - List notificationEvents = aggregatedPollingDAO. - getEventsByNotificationID(notificationDTO.getNotificationId()); - - Notification responseNotification = eventNotificationGenerator. - generateEventNotificationBody(notificationDTO, notificationEvents); - sets.put(notificationDTO.getNotificationId(), - eventNotificationGenerator.generateEventNotification(Notification.getJsonNode( - responseNotification))); - log.info("Retrieved OB event notifications"); - } catch (OBEventNotificationException | - IOException | JOSEException | IdentityOAuth2Exception e) { - log.debug("Error when retrieving OB event notifications.", e); - throw new OBEventNotificationException("Error when retrieving OB event notifications.", e); - } - } - aggregatedPollingResponse.setSets(sets); - } - } - - int count = aggregatedPollingDAO.getNotificationCountByClientIdAndStatus(eventPollingDTO.getClientId(), - EventNotificationConstants.OPEN) - aggregatedPollingResponse.getSets().size(); - - aggregatedPollingResponse.setCount(count); - - return aggregatedPollingResponse; - } - - return null; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventSubscriptionService.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventSubscriptionService.java deleted file mode 100644 index 1d561c01..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventSubscriptionService.java +++ /dev/null @@ -1,241 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.service; - -import com.wso2.openbanking.accelerator.common.util.DatabaseUtil; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.EventSubscriptionDAO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.EventSubscription; -import com.wso2.openbanking.accelerator.event.notifications.service.persistence.EventSubscriptionStoreInitializer; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.util.List; - -/** - * This is the event subscription service class. - */ -public class EventSubscriptionService { - private static Log log = LogFactory.getLog(EventSubscriptionService.class); - - /** - * This method will call the dao layer to persist the event subscription. - * - * @param eventSubscription event subscription object that needs to be persisted - * @return event subscription object that is persisted - * @throws OBEventNotificationException if an error occurred while persisting the event subscription - */ - public EventSubscription createEventSubscription(EventSubscription eventSubscription) - throws OBEventNotificationException { - - EventSubscriptionDAO eventSubscriptionDao = EventSubscriptionStoreInitializer.getEventSubscriptionDao(); - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - //store event subscription data in the database - EventSubscription storeEventSubscriptionResult = eventSubscriptionDao. - storeEventSubscription(connection, eventSubscription); - //store subscribed event types in the database - if (eventSubscription.getEventTypes() != null && !eventSubscription.getEventTypes().isEmpty()) { - List storedEventTypes = eventSubscriptionDao.storeSubscribedEventTypes(connection, - storeEventSubscriptionResult.getSubscriptionId(), eventSubscription.getEventTypes()); - storeEventSubscriptionResult.setEventTypes(storedEventTypes); - } - log.debug("Event subscription created successfully."); - DatabaseUtil.commitTransaction(connection); - return storeEventSubscriptionResult; - } catch (OBEventNotificationException e) { - log.error("Error while creating event subscription.", e); - DatabaseUtil.rollbackTransaction(connection); - throw new OBEventNotificationException(EventNotificationConstants.ERROR_STORING_EVENT_SUBSCRIPTION, e); - } finally { - DatabaseUtil.closeConnection(connection); - } - } - - /** - * This method will call the dao layer to retrieve a single event subscription. - * - * @param subscriptionId subscription id of the event subscription - * @return event subscription object that is retrieved - * @throws OBEventNotificationException if an error occurred while retrieving the event subscription - */ - public EventSubscription getEventSubscriptionBySubscriptionId(String subscriptionId) - throws OBEventNotificationException { - - EventSubscriptionDAO eventSubscriptionDao = EventSubscriptionStoreInitializer.getEventSubscriptionDao(); - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - return eventSubscriptionDao.getEventSubscriptionBySubscriptionId(connection, subscriptionId); - } catch (OBEventNotificationException e) { - log.error("Error while retrieving event subscription.", e); - throw new OBEventNotificationException(e.getMessage(), e); - } finally { - DatabaseUtil.closeConnection(connection); - } - } - - /** - * This method will call the dao layer to retrieve all event subscriptions of a client. - * - * @param clientId client id of the event subscription - * @return list of event subscriptions that are retrieved - * @throws OBEventNotificationException if an error occurred while retrieving the event subscriptions - */ - public List getEventSubscriptionsByClientId(String clientId) - throws OBEventNotificationException { - - Connection connection = DatabaseUtil.getDBConnection(); - try { - EventSubscriptionDAO eventSubscriptionDao = EventSubscriptionStoreInitializer.getEventSubscriptionDao(); - return eventSubscriptionDao.getEventSubscriptionsByClientId(connection, clientId); - } catch (OBEventNotificationException e) { - log.error("Error while retrieving event subscriptions.", e); - throw new OBEventNotificationException(e.getMessage(), e); - } finally { - DatabaseUtil.closeConnection(connection); - } - } - - /** - * This method will call the dao layer to retrieve all event subscriptions by event type. - * - * @param eventType event type that needs to be subscribed by the retrieving event subscriptions. - * @return list of event subscriptions that are retrieved - * @throws OBEventNotificationException if an error occurred while retrieving the event subscriptions - */ - public List getEventSubscriptionsByClientIdAndEventType(String eventType) - throws OBEventNotificationException { - - - Connection connection = DatabaseUtil.getDBConnection(); - try { - EventSubscriptionDAO eventSubscriptionDao = EventSubscriptionStoreInitializer.getEventSubscriptionDao(); - return eventSubscriptionDao.getEventSubscriptionsByEventType(connection, eventType); - } catch (OBEventNotificationException e) { - log.error("Error while retrieving event subscriptions.", e); - throw new OBEventNotificationException(e.getMessage(), e); - } finally { - DatabaseUtil.closeConnection(connection); - } - } - - /** - * This method will call the dao layer to update an event subscription. - * - * @param eventSubscription event subscription object that needs to be updated - * @return true if the event subscription is updated successfully - * @throws OBEventNotificationException if an error occurred while updating the event subscription - */ - public Boolean updateEventSubscription(EventSubscription eventSubscription) - throws OBEventNotificationException { - - Connection connection = DatabaseUtil.getDBConnection(); - - EventSubscriptionDAO eventSubscriptionDao = EventSubscriptionStoreInitializer.getEventSubscriptionDao(); - - //get the stored event subscription - EventSubscription retrievedEventSubscription = eventSubscriptionDao. - getEventSubscriptionBySubscriptionId(connection, eventSubscription.getSubscriptionId()); - - //update request data column - try { - JSONParser parser = new JSONParser(JSONParser.DEFAULT_PERMISSIVE_MODE); - JSONObject storedRequestData = (JSONObject) parser.parse(retrievedEventSubscription.getRequestData()); - JSONObject receivedRequestData = (JSONObject) parser.parse(eventSubscription.getRequestData()); - for (String key : storedRequestData.keySet()) { - if (receivedRequestData.containsKey(key)) { - storedRequestData.put(key, receivedRequestData.get(key)); - } - } - eventSubscription.setRequestData(storedRequestData.toJSONString()); - } catch (ParseException e) { - log.error("Error while Parsing the stored request Object", e); - throw new OBEventNotificationException("Error while Parsing the stored request Object", e); - } - - //update event subscription - try { - boolean isUpdated = eventSubscriptionDao.updateEventSubscription(connection, eventSubscription); - - //update subscribed event types - if (isUpdated && eventSubscription.getEventTypes() != null && - !eventSubscription.getEventTypes().isEmpty()) { - //delete the existing subscribed event types - eventSubscriptionDao.deleteSubscribedEventTypes(connection, eventSubscription.getSubscriptionId()); - //store the updated subscribed event types - List storedEventTypes = eventSubscriptionDao.storeSubscribedEventTypes(connection, - eventSubscription.getSubscriptionId(), eventSubscription.getEventTypes()); - eventSubscription.setEventTypes(storedEventTypes); - } else if (!isUpdated) { - log.debug("Event subscription update failed."); - DatabaseUtil.rollbackTransaction(connection); - } - log.debug("Event subscription updated successfully."); - DatabaseUtil.commitTransaction(connection); - return isUpdated; - } catch (OBEventNotificationException e) { - log.error("Error while updating event subscription.", e); - DatabaseUtil.rollbackTransaction(connection); - throw new OBEventNotificationException(e.getMessage(), e); - } finally { - DatabaseUtil.closeConnection(connection); - } - } - - /** - * This method will call the dao layer to delete an event subscription. - * - * @param subscriptionId subscription id of the event subscription - * @return true if the event subscription is deleted successfully - * @throws OBEventNotificationException if an error occurred while deleting the event subscription - */ - public Boolean deleteEventSubscription(String subscriptionId) throws OBEventNotificationException { - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - EventSubscriptionDAO eventSubscriptionDao = EventSubscriptionStoreInitializer.getEventSubscriptionDao(); - boolean isDeleted = eventSubscriptionDao.deleteEventSubscription(connection, subscriptionId); - if (isDeleted) { - log.debug("Event subscription deleted successfully."); - DatabaseUtil.commitTransaction(connection); - } else { - log.debug("Event subscription deletion failed."); - DatabaseUtil.rollbackTransaction(connection); - } - return isDeleted; - } catch (OBEventNotificationException e) { - log.error("Error while deleting event subscription.", e); - throw new OBEventNotificationException(e.getMessage(), e); - } finally { - DatabaseUtil.closeConnection(connection); - } - } - - -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/ExtendedEventNotificationGenerator.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/ExtendedEventNotificationGenerator.java deleted file mode 100644 index 5dff1cb9..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/service/ExtendedEventNotificationGenerator.java +++ /dev/null @@ -1,46 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.service; - -import com.fasterxml.jackson.databind.JsonNode; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.Notification; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationEvent; - -import java.util.List; - -/** - * Custom Event Polling Generator Class. - */ -@Generated(message = "Extended Implementation excluded from tests") -public class ExtendedEventNotificationGenerator implements EventNotificationGenerator { - - @Override - public Notification generateEventNotificationBody(NotificationDTO notificationDAO, - List notificationEventList) throws OBEventNotificationException { - return null; - } - - @Override - public String generateEventNotification(JsonNode jsonNode) { - return null; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/util/EventNotificationServiceUtil.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/util/EventNotificationServiceUtil.java deleted file mode 100644 index a83a9cc7..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/java/com/wso2/openbanking/accelerator/event/notifications/service/util/EventNotificationServiceUtil.java +++ /dev/null @@ -1,171 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.util; - -import com.fasterxml.jackson.databind.JsonNode; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.common.util.OpenBankingUtils; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.EventNotificationErrorDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.handler.DefaultEventCreationServiceHandler; -import com.wso2.openbanking.accelerator.event.notifications.service.realtime.service.RealtimeEventNotificationRequestGenerator; -import com.wso2.openbanking.accelerator.event.notifications.service.service.EventNotificationGenerator; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; -import org.wso2.carbon.identity.oauth2.util.OAuth2Util; - -import java.util.Optional; - -/** - * Default event notification validations. - */ -public class EventNotificationServiceUtil { - - private static final Log log = LogFactory.getLog(EventNotificationServiceUtil.class); - private static volatile ConsentCoreServiceImpl consentCoreService; - - /** - * This method is used to send the polling generator as per config. - * - * @return EventNotificationGenerator - */ - public static EventNotificationGenerator getEventNotificationGenerator() { - - EventNotificationGenerator eventNotificationGenerator = (EventNotificationGenerator) - OpenBankingUtils.getClassInstanceFromFQN(OpenBankingConfigParser.getInstance() - .getEventNotificationGenerator()); - return eventNotificationGenerator; - } - - /** - * This method is used to send the default realtime event notification request generator. - * - * @return RealtimeEventNotificationRequestGenerator - */ - public static RealtimeEventNotificationRequestGenerator getRealtimeEventNotificationRequestGenerator() { - - RealtimeEventNotificationRequestGenerator realtimeEventNotificationRequestGenerator = - (RealtimeEventNotificationRequestGenerator) OpenBankingUtils - .getClassInstanceFromFQN(OpenBankingConfigParser.getInstance(). - getRealtimeEventNotificationRequestGenerator()); - return realtimeEventNotificationRequestGenerator; - } - - /** - * Method to modify event notification payload with custom eventValues. - * - * @param jsonNode Json Node to convert - * @return String eventNotificationPayload - */ - public static String getCustomNotificationPayload(JsonNode jsonNode) { - - String payload = jsonNode.toString(); - return payload; - } - - /** - * Method to get event JSON from eventInformation payload string. - * @param eventInformation String event Information - * @return JSONObject converted event json - * @throws ParseException Exception when parsing event information - */ - public static JSONObject getEventJSONFromString(String eventInformation) throws ParseException { - - JSONParser parser = new JSONParser(); - return (JSONObject) parser.parse(eventInformation); - } - - /** - * Validate if the client ID is existing. - * @param clientId client ID of the TPP - * @throws OBEventNotificationException Exception when validating client ID - */ - @Generated(message = "Excluded since this needs OAuth2Util service provider") - public static void validateClientId(String clientId) throws OBEventNotificationException { - - if (StringUtils.isNotEmpty(clientId)) { - Optional serviceProvider; - try { - serviceProvider = Optional.ofNullable(OAuth2Util.getServiceProvider(clientId)); - if (!serviceProvider.isPresent()) { - log.error(EventNotificationConstants.INVALID_CLIENT_ID); - throw new OBEventNotificationException(EventNotificationConstants.INVALID_CLIENT_ID); - } - } catch (IdentityOAuth2Exception e) { - log.error(EventNotificationConstants.INVALID_CLIENT_ID, e); - throw new OBEventNotificationException(EventNotificationConstants.INVALID_CLIENT_ID); - } - } - } - - @Generated(message = "Creating a single instance for ConsentCoreService") - public static synchronized ConsentCoreServiceImpl getConsentCoreServiceImpl() { - if (consentCoreService == null) { - synchronized (ConsentCoreServiceImpl.class) { - if (consentCoreService == null) { - consentCoreService = new ConsentCoreServiceImpl(); - } - } - } - return consentCoreService; - } - - /** - * Get the callback URL of the TPP from the Subscription Object. - * - * @param clientID client ID of the TPP - * @return callback URL of the TPP - */ - public static String getCallbackURL(String clientID) { - - return "http://localhost:8080/sample-tpp-server"; - } - - /** - * Get the default event creation service handler. - * - * @return DefaultEventCreationServiceHandler - */ - public static DefaultEventCreationServiceHandler getDefaultEventCreationServiceHandler() { - return new DefaultEventCreationServiceHandler(); - } - - /** - * Method to map Event subscription Service error to API response. - * - * @param error Error code - * @param errorDescription Error description - * @return EventNotificationErrorDTO - */ - public static EventNotificationErrorDTO getErrorDTO(String error, String errorDescription) { - EventNotificationErrorDTO eventNotificationErrorDTO = new EventNotificationErrorDTO(); - eventNotificationErrorDTO.setError(error); - eventNotificationErrorDTO.setErrorDescription(errorDescription); - return eventNotificationErrorDTO; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/resources/findbugs-exclude.xml b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/resources/findbugs-exclude.xml deleted file mode 100644 index 1ea4c491..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/resources/findbugs-exclude.xml +++ /dev/null @@ -1,46 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/resources/findbugs-include.xml b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/resources/findbugs-include.xml deleted file mode 100644 index 8932a22e..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/constants/EventNotificationTestConstants.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/constants/EventNotificationTestConstants.java deleted file mode 100644 index 236d744b..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/constants/EventNotificationTestConstants.java +++ /dev/null @@ -1,72 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.constants; - -import java.util.Arrays; -import java.util.List; - -/** - * Test constant class for EventNotification Tests. - */ -public class EventNotificationTestConstants { - - public static final String SAMPLE_CLIENT_ID = "19_FTbAvbZm9YC9QRBYw8E0hVnAa"; - public static final String SAMPLE_CLIENT_ID_2 = "19_FTbAvbZm9YC9QRBYw8E0hVnAb"; - public static final String SAMPLE_RESOURCE_ID = "85d81bdb-111e-4553-8c0c-0cd2dd780515"; - public static final String SAMPLE_NOTIFICATION_ID = "c2fcb77a-274d-4851-b392-a2c0af312fd7"; - public static final String SAMPLE_NOTIFICATION_ID_2 = "c2fcb77a-274d-4851-b392-a2c0af312fb7"; - - public static final String SAMPLE_ERROR_NOTIFICATION_ID = "d3fcb77a-274d-4851-b392-a2c0af312fd8"; - public static final Long UPDATED_TIME = 1646389384L; - public static final String SAMPLE_NOTIFICATION_EVENT_TYPE_1 = "urn_uk_org_openbanking_events_resource-update"; - public static final String SAMPLE_NOTIFICATION_EVENT_TYPE_2 = - "urn_uk_org_openbanking_events_consent-authorization-revoked"; - public static final Boolean SAMPLE_RETURN_IMMEDIATETLY = true; - public static final int SAMPLE_MAX_EVENTS = 5; - - public static final String ERROR_CODE = "authentication_failed"; - - public static final String ERROR_DESCRIPTION = "The SET could not be authenticated"; - - public static final String SAMPLE_SET = "eyJhbGciOiJSUzI1NiJ9.eyJzdWIiOiJ4MEpFRzM5VGJ6dmMyRXhvbG1TaWZaR1Np" + - "TzhhIiwiYXVkIjoieDBKRUczOVRienZjMkV4b2xtU2lmWkdTaU84YSIsImlzcyI6Ind3dy53c28yLmNvbSIsInR4biI6ImUwY2Y2NG" + - "RlLTlkMGUtNDBmYy04ZWUyLTFhNTNmYzNiOWY4ZiIsInRvZSI6MTY2NDI3MDYzNzAwMCwiaWF0IjoxNjY0Mjc2NzA5LCJqdGkiOi" + - "I0ZjMxMjAwNy00ZDNmLTQwZTQtYTUyNS0wZjZlZThiYjU0ZDkiLCJldmVudHMiOnsidXJuX3VrX29yZ19vcGVuYmFua2luZ19ldm" + - "VudHNfcmVzb3VyY2UtdXBkYXRlIjp7ImtleTIiOiJ2YWx1ZSIsInJlc291cmNlSUQiOiJmNmRlMWE3NC0xMmY1LTQ5NWQtYWRjNC0" + - "xNzI4YWQwMDAyZTAnIiwia2V5MyI6InZhbHVlIn0sInVybl91a19vcmdfb3BlbmJhbmtpbmdfZXZlbnRzX2NvbnNlbnQtYXV0aG9y" + - "aXphdGlvbi1yZXZva2VkIjp7ImtleTIiOiJ2YWx1ZSIsInJlc291cmNlSUQiOiJmNmRlMWE3NC0xMmY1LTQ5NWQtYWRjNC0xNzI4Y" + - "WQwMDAyZTAnIiwia2V5MyI6InZhbHVlIn19fQ.VrkFxa2fyRhf4rP1plhedKIXNTjsrbvVveDLLHZotll2GIbxm0lCCElGUXNh463" + - "R9_HXIjfyi61b0yN2gRZKiwhPftIe9AUdFj2e2hheiE_UTiVDo9RiEvo2drvE_-ri4MN0mKHPx2GdIGKx3WTo84Ike3VZitpi8WTL7" + - "Ap1mIK1RITOd9QGO2iAwXj5NQPy9iXDV9ynQTmblLeiessUAmI3WyoYEw82P-7M0yHWCf_ztFfg6w_s9uyrak8HFmsmHeQb86frLI" + - "i4UKGiGvAVM-dBF8BAEq5eFZ2TBYWDrugk4HrSdFz7AblReTzL8vF7XFlEocFQSQ1Y_k1hXQCn4g"; - - public static final String INVALID_CLIENT_ERROR = "\"A client was not\" +\n" + - " \" found for the client id : '19_FTbAvbZm9YC9QRBYw8E0hVnAa' in the database.\""; - - public static final String SAMPLE_CALLBACK_URL = "https://localhost:8080/callback"; - - public static final String SAMPLE_NOTIFICATION_PAYLOAD = - "{\"notificationId\": " + SAMPLE_NOTIFICATION_ID + ", \"SET\": " + SAMPLE_SET + "}"; - - public static final String SAMPLE_SPEC_VERSION = "3.1"; - public static final List SAMPLE_NOTIFICATION_EVENT_TYPES = Arrays.asList(SAMPLE_NOTIFICATION_EVENT_TYPE_1, - SAMPLE_NOTIFICATION_EVENT_TYPE_2); - public static final String SAMPLE_SUBSCRIPTION_ID_1 = "550e8400-e29b-41d4-a716-446655440000"; - public static final String SAMPLE_SUBSCRIPTION_ID_2 = "9e65ebe4-2251-4a89-ba74-54060e76f51d"; -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/AggregatedPollingDAOImplTests.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/AggregatedPollingDAOImplTests.java deleted file mode 100644 index b3435533..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/AggregatedPollingDAOImplTests.java +++ /dev/null @@ -1,263 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dao; - -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.util.DatabaseUtil; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationTestConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationError; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationEvent; -import com.wso2.openbanking.accelerator.event.notifications.service.utils.EventNotificationTestUtils; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -import java.io.IOException; -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.sql.Timestamp; -import java.util.List; -import java.util.Map; - -import static org.mockito.Matchers.anyString; -import static org.powermock.api.mockito.PowerMockito.when; -/** - * Test class for AggregatedPollingDAOImpl. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest(DatabaseUtil.class) -public class AggregatedPollingDAOImplTests extends PowerMockTestCase { - - private static Connection mockedConnection; - private static Connection mockedExceptionConnection; - private PreparedStatement mockedPreparedStatement; - - - AggregatedPollingDAOImpl aggregatedPollingDAOImpl = new AggregatedPollingDAOImpl( - new NotificationPollingSqlStatements()); - - @BeforeClass - public void initTest() throws Exception { - - mockedConnection = Mockito.mock(Connection.class); - mockedExceptionConnection = Mockito.mock(Connection.class); - mockedPreparedStatement = Mockito.mock(PreparedStatement.class); - } - - @BeforeMethod - public void mock() throws ConsentManagementException { - - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()).thenReturn(mockedConnection); - } - - @Test - public void testGetNotificationsByStatus() throws OBEventNotificationException, SQLException { - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedResultSet.next()).thenReturn(true).thenReturn(true).thenReturn(false); - when(mockedResultSet.isBeforeFirst()).thenReturn(true); - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedResultSet.getString(EventNotificationConstants.NOTIFICATION_ID)).thenReturn( - EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - when(mockedResultSet.getString(EventNotificationConstants.CLIENT_ID)).thenReturn( - EventNotificationTestConstants.SAMPLE_CLIENT_ID); - when(mockedResultSet.getString(EventNotificationConstants.RESOURCE_ID)).thenReturn( - EventNotificationTestConstants.SAMPLE_RESOURCE_ID); - when(mockedResultSet.getString(EventNotificationConstants.STATUS)).thenReturn("OPEN"); - when(mockedResultSet.getTimestamp(EventNotificationConstants.UPDATED_TIMESTAMP)).thenReturn( - new Timestamp(System.currentTimeMillis())); - - List eventsList = aggregatedPollingDAOImpl.getNotificationsByStatus("OPEN"); - Assert.assertEquals(EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID, - eventsList.get(0).getNotificationId()); - } - - @Test - public void testGetEventsByNotificationID() throws OBEventNotificationException, SQLException, IOException { - - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedResultSet.next()).thenReturn(true).thenReturn(true).thenReturn(false); - when(mockedResultSet.isBeforeFirst()).thenReturn(true); - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedResultSet.getString(EventNotificationConstants.NOTIFICATION_ID)).thenReturn( - EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - when(mockedResultSet.getString(EventNotificationConstants.EVENT_TYPE)).thenReturn( - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1); - when(mockedResultSet.getString(EventNotificationConstants.EVENT_INFO)).thenReturn( - EventNotificationTestUtils.getSampleEventInformation().toString()); - - List eventsList = aggregatedPollingDAOImpl.getEventsByNotificationID( - EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - Assert.assertEquals(EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID, - eventsList.get(0).getNotificationId()); - - } - - @Test - public void testGetNotificationsByClientIdAndStatus() throws SQLException, OBEventNotificationException { - - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedResultSet.next()).thenReturn(true).thenReturn(true).thenReturn(false); - when(mockedResultSet.isBeforeFirst()).thenReturn(true); - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedResultSet.getString(EventNotificationConstants.NOTIFICATION_ID)).thenReturn( - EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - when(mockedResultSet.getString(EventNotificationConstants.CLIENT_ID)).thenReturn( - EventNotificationTestConstants.SAMPLE_CLIENT_ID); - when(mockedResultSet.getString(EventNotificationConstants.RESOURCE_ID)).thenReturn( - EventNotificationTestConstants.SAMPLE_RESOURCE_ID); - when(mockedResultSet.getString(EventNotificationConstants.STATUS)).thenReturn("OPEN"); - when(mockedResultSet.getTimestamp(EventNotificationConstants.UPDATED_TIMESTAMP)).thenReturn( - new Timestamp(System.currentTimeMillis())); - - List eventsList = aggregatedPollingDAOImpl.getNotificationsByClientIdAndStatus( - EventNotificationTestConstants.SAMPLE_CLIENT_ID, "OPEN", 5); - Assert.assertEquals(EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID, - eventsList.get(0).getNotificationId()); - } - - @Test - public void testGetEventsStatus() throws SQLException, IOException, OBEventNotificationException { - - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedResultSet.next()).thenReturn(true); - when(mockedResultSet.getString("STATUS")).thenReturn(EventNotificationConstants.OPEN); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - - boolean status = aggregatedPollingDAOImpl.getNotificationStatus(EventNotificationTestConstants. - SAMPLE_NOTIFICATION_ID); - - Assert.assertTrue(status); - - } - - @Test - public void testStoreErrorNotifications() throws SQLException, OBEventNotificationException { - - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()).thenReturn(mockedConnection); - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenReturn(1); - - Map errors = aggregatedPollingDAOImpl. - storeErrorNotification(EventNotificationTestUtils.getNotificationError()); - - Assert.assertTrue(errors.containsKey(EventNotificationTestConstants.SAMPLE_ERROR_NOTIFICATION_ID)); - } - - @Test - public void testGetEventsStatusACK() throws SQLException, IOException, OBEventNotificationException { - - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedResultSet.next()).thenReturn(true); - when(mockedResultSet.getString("STATUS")).thenReturn(EventNotificationConstants.ACK); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - - boolean status = aggregatedPollingDAOImpl.getNotificationStatus(EventNotificationTestConstants. - SAMPLE_NOTIFICATION_ID); - - Assert.assertFalse(status); - - } - - @Test(expectedExceptions = OBEventNotificationException.class) - public void testGetEventsStatusDBError() throws SQLException, OBEventNotificationException { - - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()).thenReturn(mockedExceptionConnection); - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedExceptionConnection.prepareStatement(anyString())).thenThrow(new SQLException()); - - boolean status = aggregatedPollingDAOImpl.getNotificationStatus(EventNotificationTestConstants. - SAMPLE_NOTIFICATION_ID); - } - - @Test - public void testGetNotificationCount() throws SQLException, OBEventNotificationException { - - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedResultSet.next()).thenReturn(true); - when(mockedResultSet.getInt("NOTIFICATION_COUNT")).thenReturn(4); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - - int eventCount = aggregatedPollingDAOImpl.getNotificationCountByClientIdAndStatus( - EventNotificationTestConstants.SAMPLE_CLIENT_ID, EventNotificationConstants.OPEN); - - Assert.assertEquals(eventCount, 4); - } - - @Test - public void testGetNotificationCountNoEvents() throws SQLException, OBEventNotificationException { - - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - - int eventCount = aggregatedPollingDAOImpl.getNotificationCountByClientIdAndStatus( - EventNotificationTestConstants.SAMPLE_CLIENT_ID, EventNotificationConstants.OPEN); - - Assert.assertEquals(eventCount, 0); - } - - @Test - public void testUpdateNotificationStatusById() throws SQLException, OBEventNotificationException { - - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()).thenReturn(mockedConnection); - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenReturn(1); - - Boolean updatedStatus = aggregatedPollingDAOImpl.updateNotificationStatusById( - EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID, "ACK"); - - Assert.assertTrue(updatedStatus); - } - - @Test - public void testUpdateNotificationStatusByIdError() throws SQLException, OBEventNotificationException { - - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()).thenReturn(mockedConnection); - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenReturn(0); - - Boolean updatedStatus = aggregatedPollingDAOImpl.updateNotificationStatusById( - EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID, "ACK"); - - Assert.assertFalse(updatedStatus); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventPublisherDAOImplTests.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventPublisherDAOImplTests.java deleted file mode 100644 index c5413a2e..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventPublisherDAOImplTests.java +++ /dev/null @@ -1,96 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dao; - -import com.wso2.openbanking.accelerator.common.util.DatabaseUtil; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationTestConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.utils.EventNotificationTestUtils; -import net.minidev.json.parser.ParseException; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.SQLException; - -import static org.mockito.Matchers.anyString; -import static org.powermock.api.mockito.PowerMockito.when; -/** - * Test class for EventPublisherDAOImpl. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest(DatabaseUtil.class) -public class EventPublisherDAOImplTests extends PowerMockTestCase { - - private static Connection mockedConnection; - private static Connection mockedExceptionConnection; - private PreparedStatement mockedPreparedStatement; - - EventPublisherDAOImpl eventPublisherDAOImpl = new EventPublisherDAOImpl(new NotificationPublisherSqlStatements()); - - @BeforeClass - public void initTest() throws Exception { - - mockedConnection = Mockito.mock(Connection.class); - mockedExceptionConnection = Mockito.mock(Connection.class); - mockedPreparedStatement = Mockito.mock(PreparedStatement.class); - } - - @Test - public void testPersistEventNotification() throws SQLException, ParseException, OBEventNotificationException { - - PowerMockito.mockStatic(DatabaseUtil.class); - int[] noOfRows = new int[5]; - PowerMockito.when(DatabaseUtil.getDBConnection()).thenReturn(mockedConnection); - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenReturn(1); - when(mockedPreparedStatement.executeBatch()).thenReturn(noOfRows); - - String notificationId = eventPublisherDAOImpl.persistEventNotification(mockedConnection, - EventNotificationTestUtils.getSampleNotificationDTO(), - EventNotificationTestUtils.getSampleEventList()); - - Assert.assertEquals(notificationId, EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - } - - @Test(expectedExceptions = OBEventNotificationException.class) - public void testPersistEventNotificationDBError() throws SQLException, - ParseException, OBEventNotificationException { - - PowerMockito.mockStatic(DatabaseUtil.class); - int[] noOfRows = new int[5]; - PowerMockito.when(DatabaseUtil.getDBConnection()).thenReturn(mockedExceptionConnection); - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenThrow(new SQLException()); - //when(mockedPreparedStatement.executeBatch()).thenReturn(noOfRows); - - String notificationId = eventPublisherDAOImpl.persistEventNotification(mockedConnection, - EventNotificationTestUtils.getSampleNotificationDTO(), - EventNotificationTestUtils.getSampleEventList()); - - Assert.assertEquals(notificationId, EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventSubscriptionDAOImplTests.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventSubscriptionDAOImplTests.java deleted file mode 100644 index 5579e440..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/EventSubscriptionDAOImplTests.java +++ /dev/null @@ -1,390 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.dao; - -import com.wso2.openbanking.accelerator.common.util.DatabaseUtil; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationTestConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.EventSubscription; -import com.wso2.openbanking.accelerator.event.notifications.service.utils.EventNotificationTestUtils; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.util.List; - -import static org.mockito.Matchers.anyString; -import static org.powermock.api.mockito.PowerMockito.when; - -/** - * Test class for EventSubscriptionDAOImpl. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest(DatabaseUtil.class) -public class EventSubscriptionDAOImplTests extends PowerMockTestCase { - private static Connection mockedConnection; - private PreparedStatement mockedPreparedStatement; - - EventSubscriptionDAOImpl eventSubscriptionDAOImpl = new EventSubscriptionDAOImpl( - new EventSubscriptionSqlStatements()); - - @BeforeMethod - public void mock() throws OBEventNotificationException { - mockedConnection = Mockito.mock(Connection.class); - mockedPreparedStatement = Mockito.mock(PreparedStatement.class); - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()).thenReturn(mockedConnection); - } - - @Test - public void testStoreEventSubscription() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenReturn(1); - EventSubscription sampleEventSubscription = EventNotificationTestUtils.getSampleEventSubscription(); - - EventSubscription result = eventSubscriptionDAOImpl.storeEventSubscription(mockedConnection, - sampleEventSubscription); - - Assert.assertNotNull(sampleEventSubscription.getSubscriptionId()); - Assert.assertNotNull(sampleEventSubscription.getTimeStamp()); - Assert.assertEquals("CREATED", result.getStatus()); - } - - @Test(expectedExceptions = OBEventNotificationException.class) - public void testStoreEventSubscriptionDBError() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenThrow(new SQLException()); - - eventSubscriptionDAOImpl.storeEventSubscription(mockedConnection, - EventNotificationTestUtils.getSampleEventSubscription()); - } - - @Test - public void testStoreSubscribedEventTypes() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeBatch()).thenReturn(new int[]{1, 1, 1}); - List sampleEventTypes = EventNotificationTestUtils.getSampleStoredEventTypes(); - - List result = eventSubscriptionDAOImpl.storeSubscribedEventTypes(mockedConnection, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, sampleEventTypes); - - Assert.assertEquals(sampleEventTypes, result); - } - - @Test - public void testStoreSubscribedEventTypesFailure() throws SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeBatch()).thenReturn(new int[]{0, 1, 1}); - List sampleEventTypes = EventNotificationTestUtils.getSampleStoredEventTypes(); - - Assert.assertThrows(OBEventNotificationException.class, () -> eventSubscriptionDAOImpl. - storeSubscribedEventTypes(mockedConnection, EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, - sampleEventTypes)); - } - - @Test - public void testStoreSubscribedEventTypesDBError() throws SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeBatch()).thenThrow(new SQLException()); - List sampleEventTypes = EventNotificationTestUtils.getSampleStoredEventTypes(); - - Assert.assertThrows(OBEventNotificationException.class, () -> eventSubscriptionDAOImpl. - storeSubscribedEventTypes(mockedConnection, EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, - sampleEventTypes)); - } - - @Test - public void testGetEventSubscriptionBySubscriptionId() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedResultSet.next()).thenReturn(true, true, true, false); - when(mockedResultSet.getString(EventNotificationConstants.EVENT_TYPE)). - thenReturn(EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2); - - EventSubscription result = eventSubscriptionDAOImpl.getEventSubscriptionBySubscriptionId( - mockedConnection, EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - List eventTypes = result.getEventTypes(); - Assert.assertTrue(eventTypes.contains(EventNotificationTestConstants. - SAMPLE_NOTIFICATION_EVENT_TYPE_1)); - } - - @Test - public void testGetEventSubscriptionBySubscriptionIdNotFound() throws SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedResultSet.next()).thenReturn(false); - - Assert.assertThrows(OBEventNotificationException.class, () -> eventSubscriptionDAOImpl. - getEventSubscriptionBySubscriptionId(mockedConnection, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1)); - } - - @Test - public void testGetEventSubscriptionBySubscriptionIdDBError() throws SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeQuery()).thenThrow(new SQLException()); - - Assert.assertThrows(OBEventNotificationException.class, () -> eventSubscriptionDAOImpl. - getEventSubscriptionBySubscriptionId(mockedConnection, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1)); - } - - @Test - public void testGetEventSubscriptionsByClientId() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedResultSet.isBeforeFirst()).thenReturn(true); - when(mockedResultSet.next()).thenReturn(true, true, true, true, true, true, true, false); - when(mockedResultSet.getString(EventNotificationConstants.SUBSCRIPTION_ID)). - thenReturn(EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2); - when(mockedResultSet.getString(EventNotificationConstants.EVENT_TYPE)). - thenReturn(EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2); - - List eventSubscriptions = eventSubscriptionDAOImpl.getEventSubscriptionsByClientId( - mockedConnection, EventNotificationTestConstants.SAMPLE_CLIENT_ID); - - Assert.assertNotNull(eventSubscriptions); - Assert.assertEquals(2, eventSubscriptions.size()); // We expect one EventSubscription object - - EventSubscription subscription = eventSubscriptions.get(0); - Assert.assertNotNull(subscription.getEventTypes()); - Assert.assertEquals(subscription.getEventTypes().size(), 2); - Assert.assertTrue(subscription.getEventTypes().contains(EventNotificationTestConstants. - SAMPLE_NOTIFICATION_EVENT_TYPE_1)); - Assert.assertTrue(subscription.getEventTypes().contains(EventNotificationTestConstants. - SAMPLE_NOTIFICATION_EVENT_TYPE_2)); - - EventSubscription subscription2 = eventSubscriptions.get(1); - Assert.assertNotNull(subscription2.getEventTypes()); - Assert.assertEquals(subscription2.getEventTypes().size(), 2); - Assert.assertTrue(subscription2.getEventTypes().contains(EventNotificationTestConstants. - SAMPLE_NOTIFICATION_EVENT_TYPE_1)); - Assert.assertTrue(subscription2.getEventTypes().contains(EventNotificationTestConstants. - SAMPLE_NOTIFICATION_EVENT_TYPE_2)); - } - - @Test - public void testGetEventSubscriptionsByClientIdNoSubscriptions() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedResultSet.isBeforeFirst()).thenReturn(false); - - List eventSubscriptions = eventSubscriptionDAOImpl.getEventSubscriptionsByClientId( - mockedConnection, EventNotificationTestConstants.SAMPLE_CLIENT_ID); - - Assert.assertNotNull(eventSubscriptions); - Assert.assertTrue(eventSubscriptions.isEmpty()); // We expect an empty list since no data was found - } - - @Test - public void testGetEventSubscriptionsByEventType() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedResultSet.isBeforeFirst()).thenReturn(true); - when(mockedResultSet.next()).thenReturn(true, true, true, true, true, true, true, false); - when(mockedResultSet.getString(EventNotificationConstants.SUBSCRIPTION_ID)). - thenReturn(EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2); - when(mockedResultSet.getString(EventNotificationConstants.EVENT_TYPE)). - thenReturn(EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2); - - List eventSubscriptions = eventSubscriptionDAOImpl. - getEventSubscriptionsByEventType(mockedConnection, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1); - - Assert.assertEquals(2, eventSubscriptions.size()); - - EventSubscription subscription1 = eventSubscriptions.get(0); - Assert.assertEquals(EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, subscription1.getSubscriptionId()); - Assert.assertEquals(subscription1.getEventTypes().size(), 2); - Assert.assertEquals(EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - subscription1.getEventTypes().get(0)); - - EventSubscription subscription2 = eventSubscriptions.get(1); - Assert.assertEquals(EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2, subscription2.getSubscriptionId()); - Assert.assertEquals(subscription2.getEventTypes().size(), 2); - Assert.assertEquals(EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - subscription2.getEventTypes().get(0)); - } - - @Test - public void testGetEventSubscriptionsByEventTypeNoSubscriptions() throws OBEventNotificationException, - SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedResultSet.isBeforeFirst()).thenReturn(false); - - List eventSubscriptions = eventSubscriptionDAOImpl. - getEventSubscriptionsByEventType(mockedConnection, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1); - - Assert.assertEquals(0, eventSubscriptions.size()); - } - - @Test(expectedExceptions = OBEventNotificationException.class) - public void testGetEventSubscriptionsByEventType_SQLException() throws OBEventNotificationException, - SQLException { - // Mock the behavior of the PreparedStatement and ResultSet - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedResultSet.isBeforeFirst()).thenThrow(new SQLException()); - - // Call the method under test (expecting an exception to be thrown) - eventSubscriptionDAOImpl.getEventSubscriptionsByEventType(mockedConnection, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1); - } - - @Test - public void testUpdateEventSubscription() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenReturn(1); - - Boolean isUpdated = eventSubscriptionDAOImpl.updateEventSubscription(mockedConnection, - EventNotificationTestUtils.getSampleEventSubscriptionToBeUpdated()); - Assert.assertTrue(isUpdated); - } - - @Test - public void testUpdateEventSubscriptionFailed() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenReturn(0); - - Boolean isUpdated = eventSubscriptionDAOImpl.updateEventSubscription(mockedConnection, - EventNotificationTestUtils.getSampleEventSubscriptionToBeUpdated()); - Assert.assertFalse(isUpdated); - } - - @Test - public void testUpdateEventSubscriptionDBError() throws SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenThrow(new SQLException()); - - Assert.assertThrows(OBEventNotificationException.class, - () -> eventSubscriptionDAOImpl.updateEventSubscription(mockedConnection, - EventNotificationTestUtils.getSampleEventSubscriptionToBeUpdated())); - } - - @Test - public void testDeleteEventSubscription() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenReturn(1); - - Boolean isDeleted = eventSubscriptionDAOImpl.deleteEventSubscription(mockedConnection, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - - Assert.assertTrue(isDeleted); - } - - @Test - public void testDeleteEventSubscriptionFails() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenReturn(0); - - Boolean isDeleted = eventSubscriptionDAOImpl.deleteEventSubscription(mockedConnection, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - - Assert.assertFalse(isDeleted); - } - - @Test(expectedExceptions = OBEventNotificationException.class) - public void testDeleteEventSubscriptionDBError() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenThrow(new SQLException()); - - eventSubscriptionDAOImpl.deleteEventSubscription(mockedConnection, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - } - - @Test - public void testDeleteSubscribedEventTypes() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenReturn(1); - - boolean isDeleted = eventSubscriptionDAOImpl.deleteSubscribedEventTypes(mockedConnection, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - - Assert.assertTrue(isDeleted); - } - - @Test - public void testDeleteSubscribedEventTypesFails() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenReturn(0); - - boolean isDeleted = eventSubscriptionDAOImpl.deleteSubscribedEventTypes(mockedConnection, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - - Assert.assertFalse(isDeleted); - } - - @Test(expectedExceptions = OBEventNotificationException.class) - public void testDeleteSubscribedEventTypesDBError() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenThrow(new SQLException()); - - eventSubscriptionDAOImpl.deleteSubscribedEventTypes(mockedConnection, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - } -} - diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/PostgreSqlEventSubscriptionDAOImplTests.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/PostgreSqlEventSubscriptionDAOImplTests.java deleted file mode 100644 index e2aa9aa1..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/dao/PostgreSqlEventSubscriptionDAOImplTests.java +++ /dev/null @@ -1,382 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). All Rights Reserved. - * - * This software is the property of WSO2 LLC. and its suppliers, if any. - * Dissemination of any information or reproduction of any material contained - * herein in any form is strictly forbidden, unless permitted by WSO2 expressly. - * You may not alter or remove any copyright or other notice from copies of this content. - */ - -package com.wso2.openbanking.accelerator.event.notifications.service.dao; - -import com.wso2.openbanking.accelerator.common.util.DatabaseUtil; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationTestConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.EventSubscription; -import com.wso2.openbanking.accelerator.event.notifications.service.utils.EventNotificationTestUtils; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.util.List; - -import static org.mockito.Matchers.anyInt; -import static org.mockito.Matchers.anyString; -import static org.powermock.api.mockito.PowerMockito.when; - -/** - * Test class for EventSubscriptionDAOImpl. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest(DatabaseUtil.class) -public class PostgreSqlEventSubscriptionDAOImplTests extends PowerMockTestCase { - private static Connection mockedConnection; - private PreparedStatement mockedPreparedStatement; - - PostgreSqlEventSubscriptionDAOImpl eventSubscriptionDAOImpl = new PostgreSqlEventSubscriptionDAOImpl( - new EventSubscriptionSqlStatements()); - - @BeforeMethod - public void mock() throws OBEventNotificationException { - mockedConnection = Mockito.mock(Connection.class); - mockedPreparedStatement = Mockito.mock(PreparedStatement.class); - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()).thenReturn(mockedConnection); - } - - @Test - public void testStoreEventSubscription() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenReturn(1); - EventSubscription sampleEventSubscription = EventNotificationTestUtils.getSampleEventSubscription(); - - EventSubscription result = eventSubscriptionDAOImpl.storeEventSubscription(mockedConnection, - sampleEventSubscription); - - Assert.assertNotNull(sampleEventSubscription.getSubscriptionId()); - Assert.assertNotNull(sampleEventSubscription.getTimeStamp()); - Assert.assertEquals("CREATED", result.getStatus()); - } - - @Test(expectedExceptions = OBEventNotificationException.class) - public void testStoreEventSubscriptionDBError() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenThrow(new SQLException()); - - eventSubscriptionDAOImpl.storeEventSubscription(mockedConnection, - EventNotificationTestUtils.getSampleEventSubscription()); - } - - @Test - public void testStoreSubscribedEventTypes() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeBatch()).thenReturn(new int[]{1, 1, 1}); - List sampleEventTypes = EventNotificationTestUtils.getSampleStoredEventTypes(); - - List result = eventSubscriptionDAOImpl.storeSubscribedEventTypes(mockedConnection, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, sampleEventTypes); - - Assert.assertEquals(sampleEventTypes, result); - } - - @Test - public void testStoreSubscribedEventTypesFailure() throws SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeBatch()).thenReturn(new int[]{0, 1, 1}); - List sampleEventTypes = EventNotificationTestUtils.getSampleStoredEventTypes(); - - Assert.assertThrows(OBEventNotificationException.class, () -> eventSubscriptionDAOImpl. - storeSubscribedEventTypes(mockedConnection, EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, - sampleEventTypes)); - } - - @Test - public void testStoreSubscribedEventTypesDBError() throws SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeBatch()).thenThrow(new SQLException()); - List sampleEventTypes = EventNotificationTestUtils.getSampleStoredEventTypes(); - - Assert.assertThrows(OBEventNotificationException.class, () -> eventSubscriptionDAOImpl. - storeSubscribedEventTypes(mockedConnection, EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, - sampleEventTypes)); - } - - @Test - public void testGetEventSubscriptionBySubscriptionId() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString(), anyInt(), anyInt())).thenReturn(mockedPreparedStatement); - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedResultSet.next()).thenReturn(true, true, true, false); - when(mockedResultSet.getString(EventNotificationConstants.EVENT_TYPE)). - thenReturn(EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2); - - EventSubscription result = eventSubscriptionDAOImpl.getEventSubscriptionBySubscriptionId( - mockedConnection, EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - List eventTypes = result.getEventTypes(); - Assert.assertTrue(eventTypes.contains(EventNotificationTestConstants. - SAMPLE_NOTIFICATION_EVENT_TYPE_1)); - } - - @Test - public void testGetEventSubscriptionBySubscriptionIdNotFound() throws SQLException { - when(mockedConnection.prepareStatement(anyString(), anyInt(), anyInt())).thenReturn(mockedPreparedStatement); - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedResultSet.next()).thenReturn(false); - - Assert.assertThrows(OBEventNotificationException.class, () -> eventSubscriptionDAOImpl. - getEventSubscriptionBySubscriptionId(mockedConnection, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1)); - } - - @Test - public void testGetEventSubscriptionBySubscriptionIdDBError() throws SQLException { - when(mockedConnection.prepareStatement(anyString(), anyInt(), anyInt())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeQuery()).thenThrow(new SQLException()); - - Assert.assertThrows(OBEventNotificationException.class, () -> eventSubscriptionDAOImpl. - getEventSubscriptionBySubscriptionId(mockedConnection, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1)); - } - - @Test - public void testGetEventSubscriptionsByClientId() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString(), anyInt(), anyInt())).thenReturn(mockedPreparedStatement); - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedResultSet.isBeforeFirst()).thenReturn(true); - when(mockedResultSet.next()).thenReturn(true, true, true, true, true, true, true, false); - when(mockedResultSet.getString(EventNotificationConstants.SUBSCRIPTION_ID)). - thenReturn(EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2); - when(mockedResultSet.getString(EventNotificationConstants.EVENT_TYPE)). - thenReturn(EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2); - - List eventSubscriptions = eventSubscriptionDAOImpl.getEventSubscriptionsByClientId( - mockedConnection, EventNotificationTestConstants.SAMPLE_CLIENT_ID); - - Assert.assertNotNull(eventSubscriptions); - Assert.assertEquals(2, eventSubscriptions.size()); // We expect one EventSubscription object - - EventSubscription subscription = eventSubscriptions.get(0); - Assert.assertNotNull(subscription.getEventTypes()); - Assert.assertEquals(subscription.getEventTypes().size(), 2); - Assert.assertTrue(subscription.getEventTypes().contains(EventNotificationTestConstants. - SAMPLE_NOTIFICATION_EVENT_TYPE_1)); - Assert.assertTrue(subscription.getEventTypes().contains(EventNotificationTestConstants. - SAMPLE_NOTIFICATION_EVENT_TYPE_2)); - - EventSubscription subscription2 = eventSubscriptions.get(1); - Assert.assertNotNull(subscription2.getEventTypes()); - Assert.assertEquals(subscription2.getEventTypes().size(), 2); - Assert.assertTrue(subscription2.getEventTypes().contains(EventNotificationTestConstants. - SAMPLE_NOTIFICATION_EVENT_TYPE_1)); - Assert.assertTrue(subscription2.getEventTypes().contains(EventNotificationTestConstants. - SAMPLE_NOTIFICATION_EVENT_TYPE_2)); - } - - @Test - public void testGetEventSubscriptionsByClientIdNoSubscriptions() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString(), anyInt(), anyInt())).thenReturn(mockedPreparedStatement); - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedResultSet.isBeforeFirst()).thenReturn(false); - - List eventSubscriptions = eventSubscriptionDAOImpl.getEventSubscriptionsByClientId( - mockedConnection, EventNotificationTestConstants.SAMPLE_CLIENT_ID); - - Assert.assertNotNull(eventSubscriptions); - Assert.assertTrue(eventSubscriptions.isEmpty()); // We expect an empty list since no data was found - } - - @Test - public void testGetEventSubscriptionsByEventType() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedResultSet.isBeforeFirst()).thenReturn(true); - when(mockedResultSet.next()).thenReturn(true, true, true, true, true, true, true, false); - when(mockedResultSet.getString(EventNotificationConstants.SUBSCRIPTION_ID)). - thenReturn(EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2); - when(mockedResultSet.getString(EventNotificationConstants.EVENT_TYPE)). - thenReturn(EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_2); - - List eventSubscriptions = eventSubscriptionDAOImpl. - getEventSubscriptionsByEventType(mockedConnection, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1); - - Assert.assertEquals(2, eventSubscriptions.size()); - - EventSubscription subscription1 = eventSubscriptions.get(0); - Assert.assertEquals(EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1, subscription1.getSubscriptionId()); - Assert.assertEquals(subscription1.getEventTypes().size(), 2); - Assert.assertEquals(EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - subscription1.getEventTypes().get(0)); - - EventSubscription subscription2 = eventSubscriptions.get(1); - Assert.assertEquals(EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2, subscription2.getSubscriptionId()); - Assert.assertEquals(subscription2.getEventTypes().size(), 2); - Assert.assertEquals(EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - subscription2.getEventTypes().get(0)); - } - - @Test - public void testGetEventSubscriptionsByEventTypeNoSubscriptions() throws OBEventNotificationException, - SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedResultSet.isBeforeFirst()).thenReturn(false); - - List eventSubscriptions = eventSubscriptionDAOImpl. - getEventSubscriptionsByEventType(mockedConnection, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1); - - Assert.assertEquals(0, eventSubscriptions.size()); - } - - @Test(expectedExceptions = OBEventNotificationException.class) - public void testGetEventSubscriptionsByEventType_SQLException() throws OBEventNotificationException, - SQLException { - // Mock the behavior of the PreparedStatement and ResultSet - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - ResultSet mockedResultSet = Mockito.mock(ResultSet.class); - when(mockedPreparedStatement.executeQuery()).thenReturn(mockedResultSet); - when(mockedResultSet.isBeforeFirst()).thenThrow(new SQLException()); - - // Call the method under test (expecting an exception to be thrown) - eventSubscriptionDAOImpl.getEventSubscriptionsByEventType(mockedConnection, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1); - } - - @Test - public void testUpdateEventSubscription() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenReturn(1); - - Boolean isUpdated = eventSubscriptionDAOImpl.updateEventSubscription(mockedConnection, - EventNotificationTestUtils.getSampleEventSubscriptionToBeUpdated()); - Assert.assertTrue(isUpdated); - } - - @Test - public void testUpdateEventSubscriptionFailed() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenReturn(0); - - Boolean isUpdated = eventSubscriptionDAOImpl.updateEventSubscription(mockedConnection, - EventNotificationTestUtils.getSampleEventSubscriptionToBeUpdated()); - Assert.assertFalse(isUpdated); - } - - @Test - public void testUpdateEventSubscriptionDBError() throws SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenThrow(new SQLException()); - - Assert.assertThrows(OBEventNotificationException.class, - () -> eventSubscriptionDAOImpl.updateEventSubscription(mockedConnection, - EventNotificationTestUtils.getSampleEventSubscriptionToBeUpdated())); - } - - @Test - public void testDeleteEventSubscription() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenReturn(1); - - Boolean isDeleted = eventSubscriptionDAOImpl.deleteEventSubscription(mockedConnection, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - - Assert.assertTrue(isDeleted); - } - - @Test - public void testDeleteEventSubscriptionFails() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenReturn(0); - - Boolean isDeleted = eventSubscriptionDAOImpl.deleteEventSubscription(mockedConnection, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - - Assert.assertFalse(isDeleted); - } - - @Test(expectedExceptions = OBEventNotificationException.class) - public void testDeleteEventSubscriptionDBError() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenThrow(new SQLException()); - - eventSubscriptionDAOImpl.deleteEventSubscription(mockedConnection, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - } - - @Test - public void testDeleteSubscribedEventTypes() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenReturn(1); - - boolean isDeleted = eventSubscriptionDAOImpl.deleteSubscribedEventTypes(mockedConnection, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - - Assert.assertTrue(isDeleted); - } - - @Test - public void testDeleteSubscribedEventTypesFails() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenReturn(0); - - boolean isDeleted = eventSubscriptionDAOImpl.deleteSubscribedEventTypes(mockedConnection, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - - Assert.assertFalse(isDeleted); - } - - @Test(expectedExceptions = OBEventNotificationException.class) - public void testDeleteSubscribedEventTypesDBError() throws OBEventNotificationException, SQLException { - when(mockedConnection.prepareStatement(anyString())).thenReturn(mockedPreparedStatement); - when(mockedPreparedStatement.executeUpdate()).thenThrow(new SQLException()); - - eventSubscriptionDAOImpl.deleteSubscribedEventTypes(mockedConnection, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - } -} - diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventCreationServiceHandlerTest.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventCreationServiceHandlerTest.java deleted file mode 100644 index 5137e1b5..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventCreationServiceHandlerTest.java +++ /dev/null @@ -1,162 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.handler; - -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationTestConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.response.EventCreationResponse; -import com.wso2.openbanking.accelerator.event.notifications.service.service.EventCreationService; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import com.wso2.openbanking.accelerator.event.notifications.service.utils.EventNotificationTestUtils; -import org.junit.Before; -import org.mockito.Mock; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.Test; - -import java.sql.SQLException; - -import static org.mockito.Matchers.anyBoolean; -import static org.mockito.Matchers.anyString; -import static org.mockito.Mockito.when; -import static org.powermock.api.mockito.PowerMockito.mockStatic; -/** - * Test class for DefaultEventCreationServiceHandler. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({EventNotificationServiceUtil.class, ConsentCoreServiceImpl.class}) -public class DefaultEventCreationServiceHandlerTest extends PowerMockTestCase { - - @Mock - ConsentCoreServiceImpl consentCoreServiceImpl; - - @Before - public void setUp() throws SQLException, ConsentManagementException { - - ConsentResource consentResource = new ConsentResource(); - when(consentCoreServiceImpl.getConsent(anyString(), false)).thenReturn(consentResource); - - } - - DefaultEventCreationServiceHandler defaultEventCreationServiceHandler = new DefaultEventCreationServiceHandler(); - - @Test - public void testPublishOBEvents() throws Exception { - - EventCreationService eventCreationService = Mockito.mock(EventCreationService.class); - Mockito.when(eventCreationService.publishOBEventNotification(Mockito.anyObject())). - thenReturn(EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - - defaultEventCreationServiceHandler.setEventCreationService(eventCreationService); - ConsentResource consentResource = new ConsentResource(); - consentResource.setConsentID("0ba972a9-08cd-4cad-b7e2-20655bcbd9e0"); - when(consentCoreServiceImpl.getConsent(anyString(), anyBoolean())).thenReturn(consentResource); - - mockStatic(EventNotificationServiceUtil.class); - PowerMockito.doNothing().when(EventNotificationServiceUtil.class, "validateClientId", anyString()); - PowerMockito.when(EventNotificationServiceUtil.getConsentCoreServiceImpl()).thenReturn(consentCoreServiceImpl); - - EventCreationResponse eventCreationResponse = - defaultEventCreationServiceHandler.publishOBEvent( - EventNotificationTestUtils.getNotificationCreationDTO()); - - Assert.assertEquals(eventCreationResponse.getStatus(), EventNotificationConstants.CREATED); - Assert.assertEquals(eventCreationResponse.getResponseBody().get(EventNotificationConstants.NOTIFICATIONS_ID), - EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - - } - - @Test - public void testPublishOBEventConsentException() throws Exception { - - EventCreationService eventCreationService = Mockito.mock(EventCreationService.class); - Mockito.when(eventCreationService.publishOBEventNotification(Mockito.anyObject())). - thenReturn(EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - - defaultEventCreationServiceHandler.setEventCreationService(eventCreationService); - when(consentCoreServiceImpl.getConsent(anyString(), anyBoolean())).thenThrow(new ConsentManagementException( - "Consent resource doesn't exist")); - - mockStatic(EventNotificationServiceUtil.class); - PowerMockito.doNothing().when(EventNotificationServiceUtil.class, "validateClientId", anyString()); - PowerMockito.when(EventNotificationServiceUtil.getConsentCoreServiceImpl()).thenReturn(consentCoreServiceImpl); - - EventCreationResponse eventCreationResponse = - defaultEventCreationServiceHandler.publishOBEvent(EventNotificationTestUtils. - getNotificationCreationDTO()); - - Assert.assertEquals(eventCreationResponse.getStatus(), EventNotificationConstants.BAD_REQUEST); - } - - @Test - public void testPublishOBEventInvalidClient() throws Exception { - - EventCreationService eventCreationService = Mockito.mock(EventCreationService.class); - Mockito.when(eventCreationService.publishOBEventNotification(Mockito.anyObject())). - thenReturn(EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - - defaultEventCreationServiceHandler.setEventCreationService(eventCreationService); - ConsentResource consentResource = new ConsentResource(); - consentResource.setConsentID("0ba972a9-08cd-4cad-b7e2-20655bcbd9e0"); - when(consentCoreServiceImpl.getConsent(anyString(), anyBoolean())).thenReturn(consentResource); - - PowerMockito.mockStatic(EventNotificationServiceUtil.class); - PowerMockito.doThrow(new OBEventNotificationException("Invalid client ID")). - when(EventNotificationServiceUtil.class); - EventNotificationServiceUtil.validateClientId(anyString()); - PowerMockito.when(EventNotificationServiceUtil.getConsentCoreServiceImpl()).thenReturn(consentCoreServiceImpl); - - EventCreationResponse eventCreationResponse = - defaultEventCreationServiceHandler.publishOBEvent( - EventNotificationTestUtils.getNotificationCreationDTO()); - - Assert.assertEquals(eventCreationResponse.getStatus(), EventNotificationConstants.BAD_REQUEST); - } - - @Test - public void testPublishOBEventsServiceException() throws Exception { - - EventCreationService eventCreationService = Mockito.mock(EventCreationService.class); - Mockito.when(eventCreationService.publishOBEventNotification(Mockito.anyObject())).thenThrow(new - OBEventNotificationException("Error when persisting events")); - - defaultEventCreationServiceHandler.setEventCreationService(eventCreationService); - ConsentResource consentResource = new ConsentResource(); - consentResource.setConsentID("0ba972a9-08cd-4cad-b7e2-20655bcbd9e0"); - when(consentCoreServiceImpl.getConsent(anyString(), anyBoolean())).thenReturn(consentResource); - - mockStatic(EventNotificationServiceUtil.class); - PowerMockito.doNothing().when(EventNotificationServiceUtil.class, "validateClientId", anyString()); - PowerMockito.when(EventNotificationServiceUtil.getConsentCoreServiceImpl()).thenReturn(consentCoreServiceImpl); - - EventCreationResponse eventCreationResponse = - defaultEventCreationServiceHandler.publishOBEvent( - EventNotificationTestUtils.getNotificationCreationDTO()); - - Assert.assertEquals(eventCreationResponse.getStatus(), EventNotificationConstants.BAD_REQUEST); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventPollingServiceHandlerTests.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventPollingServiceHandlerTests.java deleted file mode 100644 index 3700bc50..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventPollingServiceHandlerTests.java +++ /dev/null @@ -1,173 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.handler; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationTestConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.EventPollingDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.response.EventPollingResponse; -import com.wso2.openbanking.accelerator.event.notifications.service.service.EventPollingService; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import com.wso2.openbanking.accelerator.event.notifications.service.utils.EventNotificationTestUtils; -import net.minidev.json.JSONObject; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; - -import static org.mockito.Matchers.anyString; -import static org.powermock.api.mockito.PowerMockito.doNothing; -import static org.powermock.api.mockito.PowerMockito.mockStatic; -/** - * Test class for DefaultEventPollingServiceHandler. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({OpenBankingConfigParser.class, EventNotificationServiceUtil.class}) -public class DefaultEventPollingServiceHandlerTests extends PowerMockTestCase { - - @BeforeMethod - public void mock() { - - EventNotificationTestUtils.mockConfigParser(); - - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - - DefaultEventPollingServiceHandler defaultEventPollingServiceHandler = new DefaultEventPollingServiceHandler(); - - - @Test - public void testMapPollingRequest() { - - EventPollingDTO eventPollingDTO = defaultEventPollingServiceHandler.mapPollingRequest( - EventNotificationTestUtils.getEventRequest()); - - Assert.assertEquals(eventPollingDTO.getReturnImmediately(), - EventNotificationTestConstants.SAMPLE_RETURN_IMMEDIATETLY); - Assert.assertEquals(eventPollingDTO.getMaxEvents(), EventNotificationTestConstants.SAMPLE_MAX_EVENTS); - } - -// @Test -// public void mapPollingForEmptyRequest() { -// JSONObject eventPollingRequest = new JSONObject(); -// eventPollingRequest.put(EventNotificationConstants.X_WSO2_CLIENT_ID, -// EventNotificationTestConstants.SAMPLE_CLIENT_ID); -// EventPollingDTO eventPollingDTO = defaultEventPollingServiceHandler.mapPollingRequest(eventPollingRequest); -// -// Assert.assertEquals(eventPollingDTO., 0); -// } - - @Test - public void testMapPollingRequestWithError() { - - JSONObject eventRequest = EventNotificationTestUtils.getEventRequest(); - eventRequest.put(EventNotificationConstants.SET_ERRORS, EventNotificationTestUtils.getPollingError()); - - EventPollingDTO eventPollingDTO = defaultEventPollingServiceHandler.mapPollingRequest(eventRequest); - - Assert.assertEquals(eventPollingDTO.getMaxEvents(), EventNotificationTestConstants.SAMPLE_MAX_EVENTS); - } - - - @Test - public void testPollEvents() throws Exception { - - JSONObject eventPollingRequest = new JSONObject(); - eventPollingRequest.put(EventNotificationConstants.X_WSO2_CLIENT_ID, - EventNotificationTestConstants.SAMPLE_CLIENT_ID); - eventPollingRequest.put(EventNotificationConstants.MAX_EVENTS, 5); - - EventPollingService eventPollingService = Mockito.mock(EventPollingService.class); - Mockito.when(eventPollingService.pollEvents(Mockito.anyObject())).thenReturn(EventNotificationTestUtils. - getAggregatedPollingResponse()); - - defaultEventPollingServiceHandler.setEventPollingService(eventPollingService); - - mockStatic(EventNotificationServiceUtil.class); - doNothing().when(EventNotificationServiceUtil.class, "validateClientId", anyString()); - - EventPollingResponse eventPollingResponse = - defaultEventPollingServiceHandler.pollEvents(eventPollingRequest); - - Assert.assertEquals(eventPollingResponse.getStatus(), "OK"); - Assert.assertTrue(eventPollingResponse.getResponseBody().containsKey("moreAvailable")); - Assert.assertTrue(eventPollingResponse.getResponseBody().containsKey("sets")); - } - - @Test - public void testPollEventsInvalidClient() throws Exception { - JSONObject eventPollingRequest = new JSONObject(); - eventPollingRequest.put(EventNotificationConstants.X_WSO2_CLIENT_ID, - EventNotificationTestConstants.SAMPLE_CLIENT_ID); - eventPollingRequest.put(EventNotificationConstants.MAX_EVENTS, 5); - - EventPollingService eventPollingService = Mockito.mock(EventPollingService.class); - Mockito.when(eventPollingService.pollEvents(Mockito.anyObject())).thenReturn(EventNotificationTestUtils. - getAggregatedPollingResponse()); - - defaultEventPollingServiceHandler.setEventPollingService(eventPollingService); - - mockStatic(EventNotificationServiceUtil.class); - PowerMockito.doThrow(new OBEventNotificationException("Invalid client ID")).when( - EventNotificationServiceUtil.class); - EventNotificationServiceUtil.validateClientId(anyString()); - - EventPollingResponse eventPollingResponse = - defaultEventPollingServiceHandler.pollEvents(eventPollingRequest); - - Assert.assertEquals(eventPollingResponse.getStatus(), EventNotificationConstants.BAD_REQUEST); - } - - @Test - public void testPollEventsServiceError() throws Exception { - - JSONObject eventPollingRequest = new JSONObject(); - eventPollingRequest.put(EventNotificationConstants.X_WSO2_CLIENT_ID, - EventNotificationTestConstants.SAMPLE_CLIENT_ID); - eventPollingRequest.put(EventNotificationConstants.MAX_EVENTS, 5); - - EventPollingService eventPollingService = Mockito.mock(EventPollingService.class); - Mockito.when(eventPollingService.pollEvents(Mockito.anyObject())).thenThrow(new - OBEventNotificationException("Error when polling events")); - - defaultEventPollingServiceHandler.setEventPollingService(eventPollingService); - - mockStatic(EventNotificationServiceUtil.class); - doNothing().when(EventNotificationServiceUtil.class, "validateClientId", anyString()); - - EventPollingResponse eventPollingResponse = - defaultEventPollingServiceHandler.pollEvents(eventPollingRequest); - - Assert.assertEquals(eventPollingResponse.getStatus(), EventNotificationConstants.BAD_REQUEST); - - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventSubscriptionServiceHandlerTests.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventSubscriptionServiceHandlerTests.java deleted file mode 100644 index f8cd9f2f..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/DefaultEventSubscriptionServiceHandlerTests.java +++ /dev/null @@ -1,298 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.handler; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationTestConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.response.EventSubscriptionResponse; -import com.wso2.openbanking.accelerator.event.notifications.service.service.EventSubscriptionService; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import com.wso2.openbanking.accelerator.event.notifications.service.utils.EventNotificationTestUtils; -import net.minidev.json.JSONObject; -import org.eclipse.jetty.http.HttpStatus; -import org.mockito.Mockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; - -import static org.mockito.Matchers.anyString; -import static org.powermock.api.mockito.PowerMockito.doNothing; -import static org.powermock.api.mockito.PowerMockito.mockStatic; - -/** - * Test class for DefaultEventSubscriptionServiceHandler. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({OpenBankingConfigParser.class, EventNotificationServiceUtil.class}) -public class DefaultEventSubscriptionServiceHandlerTests extends PowerMockTestCase { - @BeforeMethod - public void mock() { - EventNotificationTestUtils.mockConfigParser(); - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - - DefaultEventSubscriptionServiceHandler defaultEventSubscriptionServiceHandler = - new DefaultEventSubscriptionServiceHandler(); - - @Test - public void testCreateEventSubscription() throws Exception { - EventSubscriptionService eventSubscriptionService = Mockito.mock(EventSubscriptionService.class); - Mockito.when(eventSubscriptionService.createEventSubscription(Mockito.anyObject())) - .thenReturn(EventNotificationTestUtils.getSampleStoredEventSubscription()); - - defaultEventSubscriptionServiceHandler.setEventSubscriptionService(eventSubscriptionService); - - mockStatic(EventNotificationServiceUtil.class); - doNothing().when(EventNotificationServiceUtil.class, "validateClientId", anyString()); - - EventSubscriptionResponse eventSubscriptionCreationResponse = defaultEventSubscriptionServiceHandler - .createEventSubscription(EventNotificationTestUtils.getSampleEventSubscriptionDTO()); - - Assert.assertEquals(eventSubscriptionCreationResponse.getStatus(), HttpStatus.CREATED_201); - } - - @Test - public void testCreateEventSubscriptionServiceError() throws Exception { - EventSubscriptionService eventSubscriptionService = Mockito.mock(EventSubscriptionService.class); - Mockito.when(eventSubscriptionService.createEventSubscription(Mockito.anyObject())) - .thenThrow(new OBEventNotificationException(EventNotificationConstants. - ERROR_STORING_EVENT_SUBSCRIPTION)); - - defaultEventSubscriptionServiceHandler.setEventSubscriptionService(eventSubscriptionService); - - mockStatic(EventNotificationServiceUtil.class); - doNothing().when(EventNotificationServiceUtil.class, "validateClientId", anyString()); - - EventSubscriptionResponse eventSubscriptionCreationResponse = defaultEventSubscriptionServiceHandler - .createEventSubscription(EventNotificationTestUtils.getSampleEventSubscriptionDTO()); - - Assert.assertEquals(eventSubscriptionCreationResponse.getStatus(), - HttpStatus.INTERNAL_SERVER_ERROR_500); - } - - @Test - public void testGetEventSubscription() throws Exception { - EventSubscriptionService eventSubscriptionService = Mockito.mock(EventSubscriptionService.class); - Mockito.when(eventSubscriptionService.getEventSubscriptionBySubscriptionId(Mockito.anyObject())) - .thenReturn(EventNotificationTestUtils.getSampleStoredEventSubscription()); - - defaultEventSubscriptionServiceHandler.setEventSubscriptionService(eventSubscriptionService); - - mockStatic(EventNotificationServiceUtil.class); - doNothing().when(EventNotificationServiceUtil.class, "validateClientId", anyString()); - - EventSubscriptionResponse eventSubscriptionRetrieveResponse = defaultEventSubscriptionServiceHandler - .getEventSubscription(EventNotificationTestConstants.SAMPLE_CLIENT_ID, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - - Assert.assertEquals(eventSubscriptionRetrieveResponse.getStatus(), HttpStatus.OK_200); - } - - @Test - public void testGetEventSubscriptionServiceError() throws Exception { - EventSubscriptionService eventSubscriptionService = Mockito.mock(EventSubscriptionService.class); - Mockito.when(eventSubscriptionService.getEventSubscriptionBySubscriptionId(Mockito.anyObject())) - .thenThrow(new OBEventNotificationException(EventNotificationConstants. - ERROR_RETRIEVING_EVENT_SUBSCRIPTIONS)); - - defaultEventSubscriptionServiceHandler.setEventSubscriptionService(eventSubscriptionService); - - mockStatic(EventNotificationServiceUtil.class); - doNothing().when(EventNotificationServiceUtil.class, "validateClientId", anyString()); - - EventSubscriptionResponse eventSubscriptionRetrieveResponse = defaultEventSubscriptionServiceHandler - .getEventSubscription(EventNotificationTestConstants.SAMPLE_CLIENT_ID, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - - Assert.assertEquals(eventSubscriptionRetrieveResponse.getStatus(), HttpStatus.INTERNAL_SERVER_ERROR_500); - } - - @Test - public void testGetAllEventSubscriptions() throws Exception { - EventSubscriptionService eventSubscriptionService = Mockito.mock(EventSubscriptionService.class); - Mockito.when(eventSubscriptionService.getEventSubscriptionsByClientId(Mockito.anyObject())) - .thenReturn(EventNotificationTestUtils.getSampleStoredEventSubscriptions()); - - defaultEventSubscriptionServiceHandler.setEventSubscriptionService(eventSubscriptionService); - - mockStatic(EventNotificationServiceUtil.class); - doNothing().when(EventNotificationServiceUtil.class, "validateClientId", anyString()); - - EventSubscriptionResponse eventSubscriptionRetrieveResponse = defaultEventSubscriptionServiceHandler - .getAllEventSubscriptions(EventNotificationTestConstants.SAMPLE_CLIENT_ID); - - Assert.assertEquals(eventSubscriptionRetrieveResponse.getStatus(), HttpStatus.OK_200); - } - - @Test - public void testGetAllEventSubscriptionsServiceError() throws Exception { - EventSubscriptionService eventSubscriptionService = Mockito.mock(EventSubscriptionService.class); - Mockito.when(eventSubscriptionService.getEventSubscriptionsByClientId(Mockito.anyObject())) - .thenThrow(new OBEventNotificationException(EventNotificationConstants. - ERROR_RETRIEVING_EVENT_SUBSCRIPTIONS)); - - defaultEventSubscriptionServiceHandler.setEventSubscriptionService(eventSubscriptionService); - - mockStatic(EventNotificationServiceUtil.class); - doNothing().when(EventNotificationServiceUtil.class, "validateClientId", anyString()); - - EventSubscriptionResponse eventSubscriptionRetrieveResponse = defaultEventSubscriptionServiceHandler - .getAllEventSubscriptions(EventNotificationTestConstants.SAMPLE_CLIENT_ID); - - Assert.assertEquals(eventSubscriptionRetrieveResponse.getStatus(), HttpStatus.INTERNAL_SERVER_ERROR_500); - } - - @Test - public void testGetEventSubscriptionsByEventType() throws Exception { - EventSubscriptionService eventSubscriptionService = Mockito.mock(EventSubscriptionService.class); - Mockito.when(eventSubscriptionService.getEventSubscriptionsByClientIdAndEventType(Mockito.anyString())) - .thenReturn(EventNotificationTestUtils.getSampleStoredEventSubscriptions()); - - defaultEventSubscriptionServiceHandler.setEventSubscriptionService(eventSubscriptionService); - - mockStatic(EventNotificationServiceUtil.class); - doNothing().when(EventNotificationServiceUtil.class, "validateClientId", anyString()); - - EventSubscriptionResponse eventSubscriptionRetrieveResponse = defaultEventSubscriptionServiceHandler - .getEventSubscriptionsByEventType(EventNotificationTestConstants.SAMPLE_CLIENT_ID, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1); - - Assert.assertEquals(eventSubscriptionRetrieveResponse.getStatus(), HttpStatus.OK_200); - } - - @Test - public void testGetEventSubscriptionsByEventTypeServiceError() throws Exception { - EventSubscriptionService eventSubscriptionService = Mockito.mock(EventSubscriptionService.class); - Mockito.when(eventSubscriptionService.getEventSubscriptionsByClientIdAndEventType(Mockito.anyString())) - .thenThrow(new OBEventNotificationException(EventNotificationConstants. - ERROR_RETRIEVING_EVENT_SUBSCRIPTIONS)); - - defaultEventSubscriptionServiceHandler.setEventSubscriptionService(eventSubscriptionService); - - mockStatic(EventNotificationServiceUtil.class); - doNothing().when(EventNotificationServiceUtil.class, "validateClientId", anyString()); - - EventSubscriptionResponse eventSubscriptionRetrieveResponse = defaultEventSubscriptionServiceHandler - .getEventSubscriptionsByEventType(EventNotificationTestConstants.SAMPLE_CLIENT_ID, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1); - - Assert.assertEquals(eventSubscriptionRetrieveResponse.getStatus(), HttpStatus.INTERNAL_SERVER_ERROR_500); - } - - @Test - public void testUpdateEventSubscription() throws Exception { - EventSubscriptionService eventSubscriptionService = Mockito.mock(EventSubscriptionService.class); - Mockito.when(eventSubscriptionService.updateEventSubscription(Mockito.anyObject())) - .thenReturn(true); - Mockito.when(eventSubscriptionService.getEventSubscriptionBySubscriptionId(Mockito.anyObject())) - .thenReturn(EventNotificationTestUtils.getSampleStoredEventSubscription()); - - defaultEventSubscriptionServiceHandler.setEventSubscriptionService(eventSubscriptionService); - - mockStatic(EventNotificationServiceUtil.class); - doNothing().when(EventNotificationServiceUtil.class, "validateClientId", anyString()); - - EventSubscriptionResponse eventSubscriptionUpdateResponse = defaultEventSubscriptionServiceHandler - .updateEventSubscription(EventNotificationTestUtils.getSampleEventSubscriptionUpdateDTO()); - - Assert.assertEquals(eventSubscriptionUpdateResponse.getStatus(), HttpStatus.OK_200); - } - - @Test - public void testUpdateEventSubscriptionServiceError() throws Exception { - EventSubscriptionService eventSubscriptionService = Mockito.mock(EventSubscriptionService.class); - Mockito.when(eventSubscriptionService.updateEventSubscription(Mockito.anyObject())) - .thenReturn(true); - Mockito.when(eventSubscriptionService.getEventSubscriptionBySubscriptionId(Mockito.anyObject())) - .thenThrow(new OBEventNotificationException(EventNotificationConstants. - ERROR_UPDATING_EVENT_SUBSCRIPTION)); - - defaultEventSubscriptionServiceHandler.setEventSubscriptionService(eventSubscriptionService); - - mockStatic(EventNotificationServiceUtil.class); - doNothing().when(EventNotificationServiceUtil.class, "validateClientId", anyString()); - - EventSubscriptionResponse eventSubscriptionUpdateResponse = defaultEventSubscriptionServiceHandler - .updateEventSubscription(EventNotificationTestUtils.getSampleEventSubscriptionUpdateDTO()); - - Assert.assertEquals(eventSubscriptionUpdateResponse.getStatus(), HttpStatus.INTERNAL_SERVER_ERROR_500); - } - - @Test - public void testDeleteEventSubscription() throws Exception { - EventSubscriptionService eventSubscriptionService = Mockito.mock(EventSubscriptionService.class); - Mockito.when(eventSubscriptionService.deleteEventSubscription(Mockito.anyObject())) - .thenReturn(true); - - defaultEventSubscriptionServiceHandler.setEventSubscriptionService(eventSubscriptionService); - - mockStatic(EventNotificationServiceUtil.class); - doNothing().when(EventNotificationServiceUtil.class, "validateClientId", anyString()); - - EventSubscriptionResponse eventSubscriptionDeletionResponse = defaultEventSubscriptionServiceHandler - .deleteEventSubscription(EventNotificationTestConstants.SAMPLE_CLIENT_ID, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - - Assert.assertEquals(eventSubscriptionDeletionResponse.getStatus(), HttpStatus.NO_CONTENT_204); - } - - @Test - public void testDeleteEventSubscriptionServiceError() throws Exception { - EventSubscriptionService eventSubscriptionService = Mockito.mock(EventSubscriptionService.class); - Mockito.when(eventSubscriptionService.deleteEventSubscription(Mockito.anyObject())) - .thenThrow(new OBEventNotificationException(EventNotificationConstants. - ERROR_DELETING_EVENT_SUBSCRIPTION)); - - defaultEventSubscriptionServiceHandler.setEventSubscriptionService(eventSubscriptionService); - - mockStatic(EventNotificationServiceUtil.class); - doNothing().when(EventNotificationServiceUtil.class, "validateClientId", anyString()); - - EventSubscriptionResponse eventSubscriptionDeletionResponse = defaultEventSubscriptionServiceHandler - .deleteEventSubscription(EventNotificationTestConstants.SAMPLE_CLIENT_ID, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - - Assert.assertEquals(eventSubscriptionDeletionResponse.getStatus(), HttpStatus.INTERNAL_SERVER_ERROR_500); - } - - @Test - public void testMapSubscriptionModelToResponseJson() { - - JSONObject eventSubscriptionCreationResponse = defaultEventSubscriptionServiceHandler - .mapSubscriptionModelToResponseJson(EventNotificationTestUtils.getSampleStoredEventSubscription()); - - Assert.assertEquals(eventSubscriptionCreationResponse.get(EventNotificationConstants.SUBSCRIPTION_ID_PARAM), - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - Assert.assertEquals(eventSubscriptionCreationResponse.get(EventNotificationConstants.CALLBACK_URL_PARAM), - EventNotificationTestConstants.SAMPLE_CALLBACK_URL); - Assert.assertEquals(eventSubscriptionCreationResponse.get(EventNotificationConstants.VERSION_PARAM), - EventNotificationTestConstants.SAMPLE_SPEC_VERSION); - Assert.assertNotNull(eventSubscriptionCreationResponse.get(EventNotificationConstants.EVENT_TYPE_PARAM)); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventNotificationPersistenceServiceHandlerTests.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventNotificationPersistenceServiceHandlerTests.java deleted file mode 100644 index 82a2ea0f..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/handler/EventNotificationPersistenceServiceHandlerTests.java +++ /dev/null @@ -1,68 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.handler; - -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationTestConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.response.EventCreationResponse; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.testng.Assert; -import org.testng.annotations.Test; - -import static org.mockito.Matchers.any; -import static org.mockito.Mockito.doReturn; - -/** - * Test class for EventNotificationPersistenceServiceHandler. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({EventNotificationServiceUtil.class, DefaultEventCreationServiceHandler.class}) -public class EventNotificationPersistenceServiceHandlerTests { - - @Test - public void testPersistRevokeEvent() { - DefaultEventCreationServiceHandler defaultEventCreationServiceHandlerMock = - Mockito.mock(DefaultEventCreationServiceHandler.class); - - PowerMockito.mockStatic(EventNotificationServiceUtil.class); - PowerMockito.when(EventNotificationServiceUtil.getDefaultEventCreationServiceHandler()) - .thenReturn(defaultEventCreationServiceHandlerMock); - - EventCreationResponse eventCreationResponse = new EventCreationResponse(); - eventCreationResponse.setStatus(EventNotificationConstants.OK); - doReturn(eventCreationResponse).when(defaultEventCreationServiceHandlerMock).publishOBEvent(any()); - - - - EventNotificationPersistenceServiceHandler revokeEventPersistenceServiceHandler = - EventNotificationPersistenceServiceHandler.getInstance(); - - EventCreationResponse response = - revokeEventPersistenceServiceHandler.persistRevokeEvent( - EventNotificationTestConstants.SAMPLE_CLIENT_ID, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, any()); - - Assert.assertEquals(response.getStatus(), EventNotificationConstants.OK); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/DefaultRealtimeEventNotificationPayloadGeneratorTests.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/DefaultRealtimeEventNotificationPayloadGeneratorTests.java deleted file mode 100644 index af4ac605..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/DefaultRealtimeEventNotificationPayloadGeneratorTests.java +++ /dev/null @@ -1,61 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.realtime.service; - -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationTestConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.testng.Assert; -import org.testng.annotations.Test; - -import java.util.HashMap; - -/** - * This class defines unit tests for DefaultRealtimeEventNotificationPayloadGenerator. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -public class DefaultRealtimeEventNotificationPayloadGeneratorTests { - - @Test - public void testGetRealtimeEventNotificationPayload() { - NotificationDTO notificationDTO = new NotificationDTO(); - notificationDTO.setClientId(EventNotificationTestConstants.SAMPLE_CLIENT_ID); - notificationDTO.setNotificationId(EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - - - RealtimeEventNotificationRequestGenerator defaultRealtimeEventNotificationRequestGenerator - = new DefaultRealtimeEventNotificationRequestGenerator(); - String result = defaultRealtimeEventNotificationRequestGenerator - .getRealtimeEventNotificationPayload(notificationDTO, - EventNotificationTestConstants.SAMPLE_SET); - - Assert.assertEquals(result, EventNotificationTestConstants.SAMPLE_NOTIFICATION_PAYLOAD); - } - - @Test - public void testGetAdditionalHeaders() { - RealtimeEventNotificationRequestGenerator defaultRealtimeEventNotificationRequestGenerator - = new DefaultRealtimeEventNotificationRequestGenerator(); - HashMap result = (HashMap) - defaultRealtimeEventNotificationRequestGenerator.getAdditionalHeaders(); - - Assert.assertEquals(0, result.size()); - } - -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/EventNotificationProducerServiceTests.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/EventNotificationProducerServiceTests.java deleted file mode 100644 index 7f58cdf0..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/EventNotificationProducerServiceTests.java +++ /dev/null @@ -1,99 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.realtime.service; - -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationTestConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.internal.EventNotificationDataHolder; -import com.wso2.openbanking.accelerator.event.notifications.service.model.Notification; -import com.wso2.openbanking.accelerator.event.notifications.service.realtime.model.RealtimeEventNotification; -import com.wso2.openbanking.accelerator.event.notifications.service.service.DefaultEventNotificationGenerator; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import com.wso2.openbanking.accelerator.event.notifications.service.utils.EventNotificationTestUtils; -import net.minidev.json.parser.ParseException; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.Test; - -import java.util.concurrent.LinkedBlockingQueue; - -import static org.mockito.Matchers.any; -import static org.mockito.Mockito.doReturn; - -/** - * Test class for EventNotificationProducerService. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({EventNotificationDataHolder.class, DefaultEventNotificationGenerator.class, - EventNotificationServiceUtil.class, DefaultRealtimeEventNotificationRequestGenerator.class}) -public class EventNotificationProducerServiceTests extends PowerMockTestCase { - @Test - public void testRun() throws OBEventNotificationException, ParseException, InterruptedException { - LinkedBlockingQueue eventQueue = new LinkedBlockingQueue<>(); - String callbackUrl = EventNotificationTestConstants.SAMPLE_CALLBACK_URL; - - DefaultEventNotificationGenerator mockedEventNotificationGenerator = - Mockito.mock(DefaultEventNotificationGenerator.class); - DefaultRealtimeEventNotificationRequestGenerator mockedRealtimeEventNotificationRequestGenerator = - Mockito.mock(DefaultRealtimeEventNotificationRequestGenerator.class); - - PowerMockito.mockStatic(EventNotificationServiceUtil.class); - PowerMockito.when(EventNotificationServiceUtil.getEventNotificationGenerator()).thenReturn( - mockedEventNotificationGenerator); - PowerMockito.when(EventNotificationServiceUtil.getRealtimeEventNotificationRequestGenerator()) - .thenReturn(mockedRealtimeEventNotificationRequestGenerator); - PowerMockito.when(EventNotificationServiceUtil.getCallbackURL(Mockito.any())).thenReturn(callbackUrl); - - EventNotificationDataHolder eventNotificationDataHolderMock = Mockito.mock(EventNotificationDataHolder.class); - Mockito.when(eventNotificationDataHolderMock.getRealtimeEventNotificationQueue()).thenReturn(eventQueue); - PowerMockito.mockStatic(EventNotificationDataHolder.class); - PowerMockito.when(EventNotificationDataHolder.getInstance()).thenReturn(eventNotificationDataHolderMock); - - NotificationDTO notificationDTO = new NotificationDTO(); - notificationDTO.setClientId(EventNotificationTestConstants.SAMPLE_CLIENT_ID); - notificationDTO.setNotificationId(EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - - Notification testNotification = new Notification(); - String testEventSET = EventNotificationTestConstants.SAMPLE_SET; - String testPayload = EventNotificationTestConstants.SAMPLE_NOTIFICATION_PAYLOAD; - - doReturn(testNotification).when(mockedEventNotificationGenerator).generateEventNotificationBody(any(), any()); - doReturn(testEventSET).when(mockedEventNotificationGenerator).generateEventNotification(any()); - doReturn(testPayload).when(mockedRealtimeEventNotificationRequestGenerator) - .getRealtimeEventNotificationPayload(any(), any()); - - EventNotificationProducerService eventNotificationProducerService = - new EventNotificationProducerService(notificationDTO, - EventNotificationTestUtils.getSampleNotificationsList()); - - new Thread(eventNotificationProducerService).start(); - - Thread.sleep(5000); - RealtimeEventNotification notification = eventQueue.take(); - - Assert.assertEquals(notification.getJsonPayload(), testPayload); - Assert.assertEquals(notification.getNotificationId(), EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - Assert.assertEquals(notification.getCallbackUrl(), callbackUrl); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationLoaderServiceTest.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationLoaderServiceTest.java deleted file mode 100644 index b0a2595e..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationLoaderServiceTest.java +++ /dev/null @@ -1,117 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.realtime.service; - -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationTestConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.AggregatedPollingDAOImpl; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.internal.EventNotificationDataHolder; -import com.wso2.openbanking.accelerator.event.notifications.service.model.Notification; -import com.wso2.openbanking.accelerator.event.notifications.service.persistence.EventPollingStoreInitializer; -import com.wso2.openbanking.accelerator.event.notifications.service.realtime.model.RealtimeEventNotification; -import com.wso2.openbanking.accelerator.event.notifications.service.service.DefaultEventNotificationGenerator; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.Test; - -import java.util.ArrayList; -import java.util.List; -import java.util.concurrent.LinkedBlockingQueue; - -import static org.mockito.Matchers.any; -import static org.mockito.Mockito.doReturn; - -/** - * Test class for RealtimeEventNotificationLoaderService. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({EventNotificationDataHolder.class, DefaultEventNotificationGenerator.class, - EventNotificationServiceUtil.class, EventPollingStoreInitializer.class, AggregatedPollingDAOImpl.class, - DefaultRealtimeEventNotificationRequestGenerator.class}) -public class RealtimeEventNotificationLoaderServiceTest extends PowerMockTestCase { - @Test - public void testRun() throws OBEventNotificationException, InterruptedException { - LinkedBlockingQueue eventQueue = new LinkedBlockingQueue<>(); - - DefaultEventNotificationGenerator mockedEventNotificationGenerator = - Mockito.mock(DefaultEventNotificationGenerator.class); - DefaultRealtimeEventNotificationRequestGenerator mockedRealtimeEventNotificationRequestGenerator = - Mockito.mock(DefaultRealtimeEventNotificationRequestGenerator.class); - - PowerMockito.mockStatic(EventNotificationServiceUtil.class); - PowerMockito.when(EventNotificationServiceUtil.getEventNotificationGenerator()).thenReturn( - mockedEventNotificationGenerator); - PowerMockito.when(EventNotificationServiceUtil.getRealtimeEventNotificationRequestGenerator()).thenReturn( - mockedRealtimeEventNotificationRequestGenerator); - - EventNotificationDataHolder eventNotificationDataHolderMock = Mockito.mock(EventNotificationDataHolder.class); - Mockito.when(eventNotificationDataHolderMock.getRealtimeEventNotificationQueue()).thenReturn(eventQueue); - PowerMockito.mockStatic(EventNotificationDataHolder.class); - PowerMockito.when(EventNotificationDataHolder.getInstance()).thenReturn(eventNotificationDataHolderMock); - - NotificationDTO notificationDTO1 = new NotificationDTO(); - notificationDTO1.setClientId(EventNotificationTestConstants.SAMPLE_CLIENT_ID); - notificationDTO1.setNotificationId(EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - - NotificationDTO notificationDTO2 = new NotificationDTO(); - notificationDTO2.setClientId(EventNotificationTestConstants.SAMPLE_CLIENT_ID_2); - notificationDTO2.setNotificationId(EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID_2); - - List notifications = new ArrayList<>(); - notifications.add(notificationDTO1); - notifications.add(notificationDTO2); - - AggregatedPollingDAOImpl mockAggregatedPollingDAOImpl = Mockito.mock(AggregatedPollingDAOImpl.class); - doReturn(notifications).when(mockAggregatedPollingDAOImpl).getNotificationsByStatus( - EventNotificationConstants.OPEN); - PowerMockito.mockStatic(EventPollingStoreInitializer.class); - PowerMockito.when(EventPollingStoreInitializer.getAggregatedPollingDAO()) - .thenReturn(mockAggregatedPollingDAOImpl); - - - Notification testNotification = new Notification(); - String testEventSET = EventNotificationTestConstants.SAMPLE_SET; - String testPayload = - "{\"notificationId\": " + notificationDTO1.getNotificationId() + ", \"SET\": " + testEventSET + "}"; - - doReturn(testNotification).when(mockedEventNotificationGenerator).generateEventNotificationBody(any(), any()); - doReturn(testEventSET).when(mockedEventNotificationGenerator).generateEventNotification(any()); - doReturn(testPayload).when(mockedRealtimeEventNotificationRequestGenerator) - .getRealtimeEventNotificationPayload(any(), any()); - - new Thread(new RealtimeEventNotificationLoaderService()).start(); - - Thread.sleep(5000); - RealtimeEventNotification notification1 = eventQueue.take(); - RealtimeEventNotification notification2 = eventQueue.take(); - - Assert.assertEquals(notification1.getNotificationId(), EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - Assert.assertEquals(notification2.getNotificationId(), EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID_2); - Assert.assertEquals(notification1.getJsonPayload(), testPayload); - Assert.assertEquals(notification2.getJsonPayload(), testPayload); - } - -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationSenderServiceTests.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationSenderServiceTests.java deleted file mode 100644 index 3c312ad7..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/realtime/service/RealtimeEventNotificationSenderServiceTests.java +++ /dev/null @@ -1,138 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.realtime.service; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.HTTPClientUtils; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationTestConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.AggregatedPollingDAOImpl; -import com.wso2.openbanking.accelerator.event.notifications.service.persistence.EventPollingStoreInitializer; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import org.apache.http.HttpStatus; -import org.apache.http.StatusLine; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpPost; -import org.apache.http.impl.client.CloseableHttpClient; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.Test; - -import java.io.IOException; -import java.util.HashMap; -import java.util.Map; - -import static org.mockito.Matchers.any; -import static org.mockito.Mockito.doReturn; - -/** - * Test class for RealtimeEventNotificationSenderService. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({OpenBankingConfigParser.class, HTTPClientUtils.class, CloseableHttpClient.class, HttpPost.class, - AggregatedPollingDAOImpl.class, EventPollingStoreInitializer.class, CloseableHttpResponse.class, - StatusLine.class, EventNotificationServiceUtil.class, DefaultRealtimeEventNotificationRequestGenerator.class}) -public class RealtimeEventNotificationSenderServiceTests extends PowerMockTestCase { - private static final int MAX_RETRIES = 1; - private static final int INITIAL_BACKOFF_TIME_IN_SECONDS = 1; - private static final String BACKOFF_FUNCTION = "EX"; - private static final int CIRCUIT_BREAKER_OPEN_TIMEOUT_IN_SECONDS = 30; - private static final int TIMEOUT_IN_SECONDS = 1; - - @BeforeClass - public void initTest() { - OpenBankingConfigParser configParser = Mockito.mock(OpenBankingConfigParser.class); - Mockito.when(configParser.getRealtimeEventNotificationMaxRetries()).thenReturn(MAX_RETRIES); - Mockito.when(configParser.getRealtimeEventNotificationInitialBackoffTimeInSeconds()) - .thenReturn(INITIAL_BACKOFF_TIME_IN_SECONDS); - Mockito.when(configParser.getRealtimeEventNotificationBackoffFunction()).thenReturn(BACKOFF_FUNCTION); - Mockito.when(configParser.getRealtimeEventNotificationCircuitBreakerOpenTimeoutInSeconds()) - .thenReturn(CIRCUIT_BREAKER_OPEN_TIMEOUT_IN_SECONDS); - Mockito.when(configParser.getRealtimeEventNotificationTimeoutInSeconds()).thenReturn(TIMEOUT_IN_SECONDS); - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(configParser); - } - - @Test - public void testRunBad() throws OpenBankingException, IOException { - AggregatedPollingDAOImpl aggregatedPollingDAOMock = Mockito.mock(AggregatedPollingDAOImpl.class); - Mockito.when(aggregatedPollingDAOMock.updateNotificationStatusById(EventNotificationTestConstants - .SAMPLE_NOTIFICATION_ID, EventNotificationConstants.ACK)).thenReturn(true); - PowerMockito.mockStatic(EventPollingStoreInitializer.class); - PowerMockito.when(EventPollingStoreInitializer.getAggregatedPollingDAO()).thenReturn(aggregatedPollingDAOMock); - - RealtimeEventNotificationRequestGenerator mockRequestGenerator = - Mockito.mock(DefaultRealtimeEventNotificationRequestGenerator.class); - Map mockHeaders = new HashMap<>(); - Mockito.when(mockRequestGenerator.getAdditionalHeaders()).thenReturn(mockHeaders); - PowerMockito.mockStatic(EventNotificationServiceUtil.class); - PowerMockito.when(EventNotificationServiceUtil.getRealtimeEventNotificationRequestGenerator()) - .thenReturn(mockRequestGenerator); - - CloseableHttpResponse mockResponse = Mockito.mock(CloseableHttpResponse.class); - StatusLine mockSL = Mockito.mock(StatusLine.class); - Mockito.when(mockResponse.getStatusLine()).thenReturn(mockSL); - Mockito.when(mockSL.getStatusCode()).thenReturn(HttpStatus.SC_BAD_REQUEST); - - CloseableHttpClient httpClientMock = Mockito.mock(CloseableHttpClient.class); - doReturn(mockResponse).when(httpClientMock).execute(any()); - PowerMockito.mockStatic(HTTPClientUtils.class); - PowerMockito.when(HTTPClientUtils.getRealtimeEventNotificationHttpsClient()).thenReturn(httpClientMock); - - new Thread(new RealtimeEventNotificationSenderService(EventNotificationTestConstants.SAMPLE_CALLBACK_URL, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_PAYLOAD, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID)).start(); - } - - @Test - public void testRun() throws OpenBankingException, IOException { - AggregatedPollingDAOImpl aggregatedPollingDAOMock = Mockito.mock(AggregatedPollingDAOImpl.class); - Mockito.when(aggregatedPollingDAOMock.updateNotificationStatusById(EventNotificationTestConstants - .SAMPLE_NOTIFICATION_ID, EventNotificationConstants.ACK)).thenReturn(true); - PowerMockito.mockStatic(EventPollingStoreInitializer.class); - PowerMockito.when(EventPollingStoreInitializer.getAggregatedPollingDAO()).thenReturn(aggregatedPollingDAOMock); - - RealtimeEventNotificationRequestGenerator mockRequestGenerator = - Mockito.mock(DefaultRealtimeEventNotificationRequestGenerator.class); - Map mockHeaders = new HashMap<>(); - Mockito.when(mockRequestGenerator.getAdditionalHeaders()).thenReturn(mockHeaders); - PowerMockito.mockStatic(EventNotificationServiceUtil.class); - PowerMockito.when(EventNotificationServiceUtil.getRealtimeEventNotificationRequestGenerator()) - .thenReturn(mockRequestGenerator); - - CloseableHttpResponse mockResponse = Mockito.mock(CloseableHttpResponse.class); - StatusLine mockSL = Mockito.mock(StatusLine.class); - Mockito.when(mockResponse.getStatusLine()).thenReturn(mockSL); - Mockito.when(mockSL.getStatusCode()).thenReturn(HttpStatus.SC_OK); - - CloseableHttpClient httpClientMock = Mockito.mock(CloseableHttpClient.class); - doReturn(mockResponse).when(httpClientMock).execute(any()); - PowerMockito.mockStatic(HTTPClientUtils.class); - PowerMockito.when(HTTPClientUtils.getRealtimeEventNotificationHttpsClient()).thenReturn(httpClientMock); - - new Thread(new RealtimeEventNotificationSenderService(EventNotificationTestConstants.SAMPLE_CALLBACK_URL, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_PAYLOAD, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID)).start(); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/service/DefaultEventNotificationGeneratorTests.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/service/DefaultEventNotificationGeneratorTests.java deleted file mode 100644 index ace067c6..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/service/DefaultEventNotificationGeneratorTests.java +++ /dev/null @@ -1,71 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.service; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationTestConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.Notification; -import com.wso2.openbanking.accelerator.event.notifications.service.utils.EventNotificationTestUtils; -import net.minidev.json.parser.ParseException; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; -/** - * Test class for DefaultEventNotificationGenerator. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({OpenBankingConfigParser.class}) -public class DefaultEventNotificationGeneratorTests { - - @BeforeMethod - public void mock() { - - EventNotificationTestUtils.mockConfigParser(); - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - - @Test - public void testGenerateEventNotificationBody() throws ParseException, OBEventNotificationException { - - NotificationDTO notificationDAO = new NotificationDTO(); - notificationDAO.setClientId(EventNotificationTestConstants.SAMPLE_CLIENT_ID); - notificationDAO.setNotificationId(EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - - DefaultEventNotificationGenerator defaultEventNotificationGenerator = new DefaultEventNotificationGenerator(); - - Notification notification = defaultEventNotificationGenerator.generateEventNotificationBody(notificationDAO, - EventNotificationTestUtils.getSampleNotificationsList()); - - Assert.assertEquals(notification.getAud(), EventNotificationTestConstants.SAMPLE_CLIENT_ID); - Assert.assertEquals(notification.getJti(), EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - Assert.assertNotNull(notification.getEvents().get(EventNotificationTestConstants. - SAMPLE_NOTIFICATION_EVENT_TYPE_1)); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventCreationServiceTests.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventCreationServiceTests.java deleted file mode 100644 index 0b36d156..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventCreationServiceTests.java +++ /dev/null @@ -1,102 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.service; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.util.DatabaseUtil; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationTestConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.EventPublisherDAO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.persistence.EventPublisherStoreInitializer; -import com.wso2.openbanking.accelerator.event.notifications.service.utils.EventNotificationTestUtils; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -import java.sql.Connection; - -import static org.mockito.Matchers.anyObject; -import static org.powermock.api.mockito.PowerMockito.when; -/** - * Tests for Default event notification validations. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({DatabaseUtil.class, EventPublisherStoreInitializer.class, OpenBankingConfigParser.class}) -public class EventCreationServiceTests extends PowerMockTestCase { - - private static Connection mockedConnection; - - private static EventPublisherDAO mockedEventPublisherDAO; - - @BeforeClass - public void initTest() throws Exception { - - mockedConnection = Mockito.mock(Connection.class); - } - - @BeforeMethod - public void mock() throws ConsentManagementException, OBEventNotificationException { - - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()).thenReturn(mockedConnection); - } - - @Test - public void testPublishOBEventNotification() throws OBEventNotificationException { - - mockedEventPublisherDAO = Mockito.mock(EventPublisherDAO.class); - OpenBankingConfigParser openBankingConfigParserMock = Mockito.mock(OpenBankingConfigParser.class); - Mockito.when(openBankingConfigParserMock.isRealtimeEventNotificationEnabled()).thenReturn(false); - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - PowerMockito.mockStatic(EventPublisherStoreInitializer.class); - PowerMockito.when(EventPublisherStoreInitializer.getEventCreationDao()).thenReturn( - mockedEventPublisherDAO); - when(mockedEventPublisherDAO.persistEventNotification(anyObject(), anyObject(), anyObject())).thenReturn( - EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - - EventCreationService eventCreationService = new EventCreationService(); - String notificationId = eventCreationService.publishOBEventNotification( - EventNotificationTestUtils.getNotificationCreationDTO()); - - Assert.assertEquals(EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID, notificationId); - } - - @Test(expectedExceptions = OBEventNotificationException.class) - public void testEventPublisherException() throws OBEventNotificationException { - - mockedEventPublisherDAO = Mockito.mock(EventPublisherDAO.class); - PowerMockito.mockStatic(EventPublisherStoreInitializer.class); - PowerMockito.when(EventPublisherStoreInitializer.getEventCreationDao()).thenReturn( - mockedEventPublisherDAO); - when(mockedEventPublisherDAO.persistEventNotification(anyObject(), anyObject(), anyObject())).thenThrow( - OBEventNotificationException.class); - - EventCreationService eventCreationService = new EventCreationService(); - String notificationId = eventCreationService.publishOBEventNotification( - EventNotificationTestUtils.getNotificationCreationDTO()); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventPollingServiceTests.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventPollingServiceTests.java deleted file mode 100644 index 0dc4f394..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventPollingServiceTests.java +++ /dev/null @@ -1,113 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.service; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.util.DatabaseUtil; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.AggregatedPollingDAO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.AggregatedPollingResponse; -import com.wso2.openbanking.accelerator.event.notifications.service.persistence.EventPollingStoreInitializer; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import com.wso2.openbanking.accelerator.event.notifications.service.utils.EventNotificationTestUtils; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -import java.sql.Connection; -/** - * Test class for EventPollingService. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({DatabaseUtil.class, EventPollingStoreInitializer.class, EventNotificationServiceUtil.class, - OpenBankingConfigParser.class}) -public class EventPollingServiceTests extends PowerMockTestCase { - private static Connection mockedConnection; - private static AggregatedPollingDAO mockedAggregatedPollingDAO; - private static EventNotificationGenerator mockedEventNotificationGenerator; - - @BeforeClass - public void initTest() throws Exception { - - mockedConnection = Mockito.mock(Connection.class); - } - - @BeforeMethod - public void mock() throws ConsentManagementException, OBEventNotificationException { - - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()).thenReturn(mockedConnection); - EventNotificationTestUtils.mockConfigParser(); - } - - @Test - public void testPollEventsNoNotifications() throws OBEventNotificationException { - - mockedAggregatedPollingDAO = Mockito.mock(AggregatedPollingDAO.class); - mockedEventNotificationGenerator = Mockito.mock(EventNotificationGenerator.class); - - PowerMockito.mockStatic(EventPollingStoreInitializer.class); - PowerMockito.when(EventPollingStoreInitializer.getAggregatedPollingDAO()).thenReturn( - mockedAggregatedPollingDAO); - PowerMockito.mockStatic(EventNotificationServiceUtil.class); - PowerMockito.when(EventNotificationServiceUtil.getEventNotificationGenerator()).thenReturn( - mockedEventNotificationGenerator); - PowerMockito.when(mockedAggregatedPollingDAO.getNotificationStatus(Mockito.anyString())).thenReturn(true); - - EventPollingService eventPollingService = new EventPollingService(); - - AggregatedPollingResponse aggregatedPollingResponse = eventPollingService.pollEvents( - EventNotificationTestUtils.getEventPollingDTO()); - - Assert.assertEquals(aggregatedPollingResponse.getStatus(), EventNotificationConstants.NOT_FOUND); - } - - @Test - public void testPollNotifications() throws OBEventNotificationException { - - mockedAggregatedPollingDAO = Mockito.mock(AggregatedPollingDAO.class); - mockedEventNotificationGenerator = Mockito.mock(EventNotificationGenerator.class); - - PowerMockito.mockStatic(EventPollingStoreInitializer.class); - PowerMockito.when(EventPollingStoreInitializer.getAggregatedPollingDAO()).thenReturn( - mockedAggregatedPollingDAO); - PowerMockito.mockStatic(EventNotificationServiceUtil.class); - PowerMockito.when(EventNotificationServiceUtil.getEventNotificationGenerator()).thenReturn( - mockedEventNotificationGenerator); - PowerMockito.when(mockedAggregatedPollingDAO.getNotificationsByClientIdAndStatus(Mockito.anyString(), - Mockito.anyString(), Mockito.anyInt())).thenReturn( - EventNotificationTestUtils.getSampleSavedTestNotification()); - PowerMockito.when(mockedAggregatedPollingDAO.getNotificationStatus(Mockito.anyString())).thenReturn(true); - - EventPollingService eventPollingService = new EventPollingService(); - - AggregatedPollingResponse aggregatedPollingResponse = eventPollingService.pollEvents( - EventNotificationTestUtils.getEventPollingDTO()); - - Assert.assertEquals(aggregatedPollingResponse.getStatus(), EventNotificationConstants.OK); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventSubscriptionServiceTests.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventSubscriptionServiceTests.java deleted file mode 100644 index d8f906df..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/service/EventSubscriptionServiceTests.java +++ /dev/null @@ -1,171 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.service; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.util.DatabaseUtil; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationTestConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dao.EventSubscriptionDAO; -import com.wso2.openbanking.accelerator.event.notifications.service.exceptions.OBEventNotificationException; -import com.wso2.openbanking.accelerator.event.notifications.service.model.EventSubscription; -import com.wso2.openbanking.accelerator.event.notifications.service.persistence.EventSubscriptionStoreInitializer; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import com.wso2.openbanking.accelerator.event.notifications.service.utils.EventNotificationTestUtils; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.powermock.modules.testng.PowerMockTestCase; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.Test; - -import java.sql.Connection; -import java.util.List; - -/** - * This is to test the Event Subscription Service. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({DatabaseUtil.class, EventSubscriptionStoreInitializer.class, EventNotificationServiceUtil.class, - OpenBankingConfigParser.class}) -public class EventSubscriptionServiceTests extends PowerMockTestCase { - private static Connection mockedConnection; - private static EventSubscriptionDAO mockedEventSubscriptionDAO; - - @BeforeClass - public void initTest() { - - mockedConnection = Mockito.mock(Connection.class); - } - - @BeforeMethod - public void mock() throws ConsentManagementException, OBEventNotificationException { - - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()).thenReturn(mockedConnection); - EventNotificationTestUtils.mockConfigParser(); - - mockedEventSubscriptionDAO = Mockito.mock(EventSubscriptionDAO.class); - - PowerMockito.mockStatic(EventSubscriptionStoreInitializer.class); - PowerMockito.when(EventSubscriptionStoreInitializer.getEventSubscriptionDao()). - thenReturn(mockedEventSubscriptionDAO); - } - - @Test - public void testCreateEventSubscription() throws OBEventNotificationException { - PowerMockito.when(mockedEventSubscriptionDAO.storeEventSubscription(Mockito.anyObject(), Mockito.any())). - thenReturn(EventNotificationTestUtils.getSampleStoredEventSubscription()); - PowerMockito.when(mockedEventSubscriptionDAO.storeSubscribedEventTypes(Mockito.anyObject(), Mockito.anyString(), - Mockito.any())).thenReturn(EventNotificationTestUtils.getSampleStoredEventTypes()); - - EventSubscriptionService eventSubscriptionService = new EventSubscriptionService(); - EventSubscription sampleEventSubscription = EventNotificationTestUtils.getSampleStoredEventSubscription(); - - EventSubscription result = eventSubscriptionService.createEventSubscription( - EventNotificationTestUtils.getSampleEventSubscription()); - - Assert.assertNotNull(sampleEventSubscription.getSubscriptionId()); // Check that subscriptionId is not null - Assert.assertNotNull(sampleEventSubscription.getTimeStamp()); // Check that timeStamp is not null - Assert.assertEquals(EventNotificationConstants.CREATED, result.getStatus()); - Assert.assertNotNull(result.getEventTypes()); - Assert.assertTrue(result.getEventTypes().contains(EventNotificationTestConstants. - SAMPLE_NOTIFICATION_EVENT_TYPE_1)); - Assert.assertTrue(result.getEventTypes().contains(EventNotificationTestConstants. - SAMPLE_NOTIFICATION_EVENT_TYPE_2)); - - } - - @Test - public void testGetEventSubscriptionBySubscriptionId() throws OBEventNotificationException { - PowerMockito.when(mockedEventSubscriptionDAO.getEventSubscriptionBySubscriptionId(Mockito.anyObject(), - Mockito.anyString())).thenReturn(EventNotificationTestUtils.getSampleStoredEventSubscription()); - - EventSubscriptionService eventSubscriptionService = new EventSubscriptionService(); - - EventSubscription result = eventSubscriptionService. - getEventSubscriptionBySubscriptionId(EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - - Assert.assertNotNull(result); - } - - @Test - public void testGetEventSubscriptionsByClientId() throws OBEventNotificationException { - PowerMockito.when(mockedEventSubscriptionDAO.getEventSubscriptionsByClientId(Mockito.anyObject(), - Mockito.anyString())).thenReturn(EventNotificationTestUtils.getSampleStoredEventSubscriptions()); - - EventSubscriptionService eventSubscriptionService = new EventSubscriptionService(); - - List result = eventSubscriptionService. - getEventSubscriptionsByClientId(EventNotificationTestConstants.SAMPLE_CLIENT_ID); - - Assert.assertEquals(result.size(), 2); - } - - @Test - public void testGetEventSubscriptionsByClientIdAndEventType() throws OBEventNotificationException { - PowerMockito.when(mockedEventSubscriptionDAO.getEventSubscriptionsByEventType(Mockito.anyObject(), - Mockito.anyString())). - thenReturn(EventNotificationTestUtils.getSampleStoredEventSubscriptions()); - - EventSubscriptionService eventSubscriptionService = new EventSubscriptionService(); - - List result = eventSubscriptionService. - getEventSubscriptionsByClientIdAndEventType( - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1); - - Assert.assertEquals(2, result.size()); - } - - @Test - public void testUpdateEventSubscription() throws OBEventNotificationException { - PowerMockito.when(mockedEventSubscriptionDAO.getEventSubscriptionBySubscriptionId(Mockito.anyObject(), - Mockito.anyString())).thenReturn(EventNotificationTestUtils.getSampleStoredEventSubscription()); - PowerMockito.when(mockedEventSubscriptionDAO.updateEventSubscription(Mockito.anyObject(), Mockito.any())). - thenReturn(true); - PowerMockito.when(mockedEventSubscriptionDAO.deleteSubscribedEventTypes(Mockito.anyObject(), - Mockito.anyString())).thenReturn(true); - PowerMockito.when(mockedEventSubscriptionDAO.storeSubscribedEventTypes(Mockito.anyObject(), Mockito.anyString(), - Mockito.any())).thenReturn(EventNotificationTestUtils.getSampleStoredEventTypes()); - - EventSubscriptionService eventSubscriptionService = new EventSubscriptionService(); - - Boolean result = eventSubscriptionService. - updateEventSubscription(EventNotificationTestUtils.getSampleEventSubscriptionToBeUpdated()); - - Assert.assertTrue(result); - } - - @Test - public void testDeleteEventSubscription() throws OBEventNotificationException { - PowerMockito.when(mockedEventSubscriptionDAO.deleteEventSubscription(Mockito.anyObject(), Mockito.anyString())). - thenReturn(true); - - EventSubscriptionService eventSubscriptionService = new EventSubscriptionService(); - - Boolean result = eventSubscriptionService. - deleteEventSubscription(EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - - Assert.assertTrue(result); - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/utils/EventNotificationTestUtils.java b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/utils/EventNotificationTestUtils.java deleted file mode 100644 index 2878f1a8..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/java/com/wso2/openbanking/accelerator/event/notifications/service/utils/EventNotificationTestUtils.java +++ /dev/null @@ -1,275 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.service.utils; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationTestConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.EventPollingDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.EventSubscriptionDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationCreationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.model.AggregatedPollingResponse; -import com.wso2.openbanking.accelerator.event.notifications.service.model.EventSubscription; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationError; -import com.wso2.openbanking.accelerator.event.notifications.service.model.NotificationEvent; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.ParseException; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -/** - * EventNotification Test Utils class. - */ -public class EventNotificationTestUtils { - - public static List getSampleSavedTestNotification() { - - List notificationList = new ArrayList<>(); - notificationList.add(getSampleNotificationDTO()); - - return notificationList; - } - public static NotificationDTO getSampleNotificationDTO() { - - NotificationDTO notificationDTO = new NotificationDTO(); - notificationDTO.setResourceId(EventNotificationTestConstants.SAMPLE_RESOURCE_ID); - notificationDTO.setNotificationId(EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - notificationDTO.setStatus(EventNotificationConstants.OPEN); - notificationDTO.setUpdatedTimeStamp(EventNotificationTestConstants.UPDATED_TIME); - - return notificationDTO; - } - public static List getSampleNotificationsList() throws ParseException { - List eventsList = new ArrayList(); - NotificationEvent notificationEvent = new NotificationEvent(); - notificationEvent.setEventType(EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1); - notificationEvent.setEventInformation(getSampleEventInformation()); - eventsList.add(notificationEvent); - - return eventsList; - } - - public static JSONObject getEventRequest() { - - JSONObject sampleEventPollingRequest = new JSONObject(); - sampleEventPollingRequest.put(EventNotificationConstants.X_WSO2_CLIENT_ID, - EventNotificationTestConstants.SAMPLE_CLIENT_ID); - sampleEventPollingRequest.put(EventNotificationConstants.RETURN_IMMEDIATELY, true); - sampleEventPollingRequest.put(EventNotificationConstants.MAX_EVENTS, 5); - sampleEventPollingRequest.put(EventNotificationConstants.ACK, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - return sampleEventPollingRequest; - } - - public static JSONObject getSampleEventInformation() { - - JSONObject jsonObject = new JSONObject(); - jsonObject.put("key1", "value1"); - jsonObject.put("key2", "value2"); - jsonObject.put("key3", "value3"); - - return jsonObject; - } - - public static void mockConfigParser() { - - OpenBankingConfigParser openBankingConfigParserMock = Mockito.mock(OpenBankingConfigParser.class); - Map configuration = new HashMap<>(); - configuration.put(OpenBankingConstants.TOKEN_ISSUER, "www.wso2.com"); - configuration.put(EventNotificationConstants.MAX_EVENTS, 5); - Mockito.when(openBankingConfigParserMock.getConfiguration()).thenReturn(configuration); - PowerMockito.mockStatic(OpenBankingConfigParser.class); - PowerMockito.when(OpenBankingConfigParser.getInstance()).thenReturn(openBankingConfigParserMock); - - } - - public static AggregatedPollingResponse getAggregatedPollingResponse() { - - Map sets = new HashMap<>(); - sets.put("4f312007-4d3f-40e4-a525-0f6ee8bb54d9", EventNotificationTestConstants.SAMPLE_SET); - AggregatedPollingResponse aggregatedPollingResponse = new AggregatedPollingResponse(); - aggregatedPollingResponse.setCount(0); - aggregatedPollingResponse.setStatus("OK"); - aggregatedPollingResponse.setSets(sets); - return aggregatedPollingResponse; - } - - public static JSONObject getPollingError() { - - JSONObject errorObj = new JSONObject(); - errorObj.put("65ac7453-13b0-4d2f-9946-dff6e6089a4f", errorInfo()); - errorObj.put("78ac7453-13b0-4d2f-9946-dff6e608345f", errorInfo()); - return errorObj; - } - - public static JSONObject errorInfo() { - - JSONObject errorInfo = new JSONObject(); - errorInfo.put("err", "authentication_failed"); - errorInfo.put("description", "The SET could not be authenticated"); - return errorInfo; - } - - public static NotificationCreationDTO getNotificationCreationDTO() { - - NotificationCreationDTO notificationCreationDTO = new NotificationCreationDTO(); - notificationCreationDTO.setClientId(EventNotificationTestConstants.SAMPLE_CLIENT_ID); - notificationCreationDTO.setEventPayload(EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1, - getSampleEventInformation()); - notificationCreationDTO.setResourceId(EventNotificationTestConstants.SAMPLE_RESOURCE_ID); - - return notificationCreationDTO; - } - - public static NotificationError getNotificationError() { - - NotificationError notificationError = new NotificationError(); - notificationError.setNotificationId("d3fcb77a-274d-4851-b392-a2c0af312fd8"); - notificationError.setErrorCode(EventNotificationTestConstants.ERROR_CODE); - notificationError.setErrorDescription(EventNotificationTestConstants.ERROR_DESCRIPTION); - - return notificationError; - } - - public static EventPollingDTO getEventPollingDTO() { - - EventPollingDTO eventPollingDTO = new EventPollingDTO(); - eventPollingDTO.setMaxEvents(3); - eventPollingDTO.setClientId(EventNotificationTestConstants.SAMPLE_CLIENT_ID); - eventPollingDTO.setErrors("d3fcb77a-274d-4851-b392-a2c0af312fd8", getNotificationError()); - eventPollingDTO.setAck(EventNotificationTestConstants.SAMPLE_NOTIFICATION_ID); - - return eventPollingDTO; - } - - public static ArrayList getSampleEventList() throws ParseException { - ArrayList eventsList = new ArrayList(); - NotificationEvent notificationEvent = new NotificationEvent(); - notificationEvent.setEventType(EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPE_1); - notificationEvent.setEventInformation(getSampleEventInformation()); - eventsList.add(notificationEvent); - - return eventsList; - } - - public static EventSubscription getSampleEventSubscription() { - - EventSubscription eventSubscription = new EventSubscription(); - eventSubscription.setClientId(EventNotificationTestConstants.SAMPLE_CLIENT_ID); - eventSubscription.setCallbackUrl(EventNotificationTestConstants.SAMPLE_CALLBACK_URL); - eventSubscription.setSpecVersion(EventNotificationTestConstants.SAMPLE_SPEC_VERSION); - eventSubscription.setEventTypes(EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPES); - return eventSubscription; - } - - public static EventSubscription getSampleStoredEventSubscription() { - EventSubscription eventSubscription = new EventSubscription(); - eventSubscription.setSubscriptionId(EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - eventSubscription.setClientId(EventNotificationTestConstants.SAMPLE_CLIENT_ID); - eventSubscription.setCallbackUrl(EventNotificationTestConstants.SAMPLE_CALLBACK_URL); - eventSubscription.setSpecVersion(EventNotificationTestConstants.SAMPLE_SPEC_VERSION); - eventSubscription.setTimeStamp(1626480000L); - eventSubscription.setEventTypes(EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPES); - eventSubscription.setStatus("CREATED"); - - JSONObject requestData = new JSONObject(); - requestData.put(EventNotificationConstants.SUBSCRIPTION_ID_PARAM, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - requestData.put("clientId", EventNotificationTestConstants.SAMPLE_CLIENT_ID); - requestData.put(EventNotificationConstants.CALLBACK_URL_PARAM, - EventNotificationTestConstants.SAMPLE_CALLBACK_URL); - requestData.put(EventNotificationConstants.VERSION_PARAM, EventNotificationTestConstants.SAMPLE_SPEC_VERSION); - requestData.put(EventNotificationConstants.EVENT_TYPE_PARAM, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPES); - requestData.put("timeStamp", 1626480000L); - requestData.put("status", "CREATED"); - eventSubscription.setRequestData(requestData.toString()); - return eventSubscription; - } - - public static EventSubscription getSampleStoredEventSubscription2() { - EventSubscription eventSubscription = new EventSubscription(); - eventSubscription.setSubscriptionId(EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_2); - eventSubscription.setClientId(EventNotificationTestConstants.SAMPLE_CLIENT_ID); - eventSubscription.setCallbackUrl(EventNotificationTestConstants.SAMPLE_CALLBACK_URL); - eventSubscription.setSpecVersion(EventNotificationTestConstants.SAMPLE_SPEC_VERSION); - eventSubscription.setEventTypes(EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPES); - eventSubscription.setStatus("CREATED"); - return eventSubscription; - } - - public static List getSampleStoredEventSubscriptions() { - List eventSubscriptions = new ArrayList<>(); - eventSubscriptions.add(getSampleStoredEventSubscription()); - eventSubscriptions.add(getSampleStoredEventSubscription2()); - return eventSubscriptions; - } - - public static EventSubscription getSampleEventSubscriptionToBeUpdated() { - EventSubscription eventSubscription = new EventSubscription(); - eventSubscription.setSubscriptionId(EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - eventSubscription.setCallbackUrl("test.com"); - eventSubscription.setEventTypes(EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPES); - - JSONObject requestData = new JSONObject(); - requestData.put(EventNotificationConstants.SUBSCRIPTION_ID_PARAM, - EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - requestData.put(EventNotificationConstants.CALLBACK_URL_PARAM, "test.com"); - requestData.put(EventNotificationConstants.EVENT_TYPE_PARAM, - EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPES); - eventSubscription.setRequestData(requestData.toString()); - return eventSubscription; - } - - public static List getSampleStoredEventTypes() { - return EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPES; - } - - public static EventSubscriptionDTO getSampleEventSubscriptionDTO() { - EventSubscriptionDTO eventSubscriptionDTO = new EventSubscriptionDTO(); - JSONObject request = new JSONObject(); - request.put("callbackUrl", EventNotificationTestConstants.SAMPLE_CALLBACK_URL); - request.put("eventTypes", EventNotificationTestConstants.SAMPLE_NOTIFICATION_EVENT_TYPES); - request.put("version", EventNotificationTestConstants.SAMPLE_SPEC_VERSION); - eventSubscriptionDTO.setClientId(EventNotificationTestConstants.SAMPLE_CLIENT_ID); - eventSubscriptionDTO.setSubscriptionId(EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - eventSubscriptionDTO.setRequestData(request); - return eventSubscriptionDTO; - } - - public static EventSubscriptionDTO getSampleEventSubscriptionUpdateDTO() { - EventSubscriptionDTO eventSubscriptionDTO = new EventSubscriptionDTO(); - List eventTypes = Arrays.asList("event 1", "event 2"); - JSONObject request = new JSONObject(); - request.put("callbackUrl", "updated url"); - request.put("eventTypes", eventTypes); - eventSubscriptionDTO.setClientId(EventNotificationTestConstants.SAMPLE_CLIENT_ID); - eventSubscriptionDTO.setSubscriptionId(EventNotificationTestConstants.SAMPLE_SUBSCRIPTION_ID_1); - eventSubscriptionDTO.setRequestData(request); - return eventSubscriptionDTO; - } -} diff --git a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/resources/testng.xml b/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/resources/testng.xml deleted file mode 100644 index 783c7d6a..00000000 --- a/open-banking-accelerator/components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service/src/test/resources/testng.xml +++ /dev/null @@ -1,42 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/pom.xml b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/pom.xml deleted file mode 100644 index 8d95e164..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/pom.xml +++ /dev/null @@ -1,189 +0,0 @@ - - - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../../pom.xml - - 4.0.0 - - com.wso2.openbanking.accelerator.throttler.dao - WSO2 Open Banking - Throttler DAO - WSO2 Open Banking - Throttler DAO Module - jar - - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - provided - - - org.testng - testng - - - com.h2database - h2 - test - - - org.jacoco - org.jacoco.agent - runtime - test - - - org.mockito - mockito-all - test - - - org.powermock - powermock-api-mockito - test - - - org.powermock - powermock-module-testng - test - - - - - - org.apache.maven.plugins - maven-surefire-plugin - - - src/test/resources/testng.xml - - - target/jacoco.exec - - true - - - - - org.jacoco - jacoco-maven-plugin - ${jacoco.version} - - - - **/*Exception.class - **/*Constants.class - **/ThrottleDataModel.class - - **/DataStoreInitializerFactory.class - **/DataStoreInitializer.class - **/OBThrottlerSQLStatements.class - - - - - default-prepare-agent - - prepare-agent - - - - default-prepare-agent-integration - - prepare-agent-integration - - - - default-report - - report - - - - default-report-integration - - report-integration - - - - default-check - - check - - - - - BUNDLE - - - INSTRUCTION - COVEREDRATIO - 0.80 - - - - - - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - false - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-exclude.xml - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - org.apache.maven.plugins - maven-compiler-plugin - - 1.8 - 1.8 - UTF-8 - - - - - diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/OBThrottlerDAO.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/OBThrottlerDAO.java deleted file mode 100644 index e0fb95ed..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/OBThrottlerDAO.java +++ /dev/null @@ -1,99 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.dao; - -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataDeletionException; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataInsertionException; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataRetrievalException; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataUpdationException; -import com.wso2.openbanking.accelerator.throttler.dao.model.ThrottleDataModel; - -import java.sql.Connection; -import java.sql.Timestamp; - -/** - * DAO class for throttle data. - */ -public interface OBThrottlerDAO { - - /** - * Store throttle data. - * - * @param connection connection object - * @param throttleGroup - throttle group - * @param throttleParam - throttle parameter - * @param currentTimestamp - current timestamp - * @param unlockTimestamp - unlock timestamp - * @return - ThrottleDataModel - * @throws OBThrottlerDataInsertionException - OBThrottlerDataInsertionException - */ - ThrottleDataModel storeThrottleData(Connection connection, String throttleGroup, String throttleParam, - Timestamp currentTimestamp, Timestamp unlockTimestamp) - throws OBThrottlerDataInsertionException; - - /** - * Update throttle data. - * - * @param connection connection object - * @param throttleGroup - throttle group - * @param throttleParam - throttle parameter - * @param currentTimestamp - current timestamp - * @param unlockTimestamp - unlock timestamp - * @param occurrences - number of occurrences of the parameter - * @return - ThrottleDataModel - * @throws OBThrottlerDataUpdationException - OBThrottlerDataUpdationException - */ - ThrottleDataModel updateThrottleData(Connection connection, String throttleGroup, String throttleParam, - Timestamp currentTimestamp, Timestamp unlockTimestamp, int occurrences) - throws OBThrottlerDataUpdationException; - - /** - * Retrieve throttle data. - * - * @param connection connection object - * @param throttleGroup - throttle group - * @param throttleParam - throttle parameter - * @return - ThrottleDataModel - * @throws OBThrottlerDataRetrievalException - OBThrottlerDataRetrievalException - */ - ThrottleDataModel getThrottleData(Connection connection, String throttleGroup, String throttleParam) - throws OBThrottlerDataRetrievalException; - - /** - * Remove throttle data from database. - * - * @param connection connection object - * @param throttleGroup - throttle group - * @param throttleParam - throttle parameter - * @throws OBThrottlerDataDeletionException - OBThrottlerDataDeletionException - */ - void deleteThrottleData(Connection connection, String throttleGroup, String throttleParam) - throws OBThrottlerDataDeletionException; - - /** - * Check if throttle data exists in the database. - * - * @param connection connection object - * @param throttleGroup - throttle group - * @param throttleParam - throttle parameter - * @return - boolean - * @throws OBThrottlerDataRetrievalException - OBThrottlerDataRetrievalException - */ - boolean isThrottleDataExists(Connection connection, String throttleGroup, String throttleParam) - throws OBThrottlerDataRetrievalException; -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/constants/OBThrottlerDAOConstants.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/constants/OBThrottlerDAOConstants.java deleted file mode 100644 index f57d3383..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/constants/OBThrottlerDAOConstants.java +++ /dev/null @@ -1,41 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.dao.constants; - -/** - * This class contains all the constants needed for the ob throttler DAO layer. - */ -public class OBThrottlerDAOConstants { - - public static final int FIRST_OCCURRENCE = 1; - public static final String LAST_UPDATE_TIMESTAMP = "LAST_UPDATE_TIMESTAMP"; - public static final String UNLOCK_TIMESTAMP = "UNLOCK_TIMESTAMP"; - public static final String OCCURRENCES = "OCCURRENCES"; - - public static final String THROTTLE_DATA_STORE_ERROR_MSG = "Error occurred while persisting throttle data in the " + - "database"; - public static final String THROTTLE_DATA_UPDATE_ERROR_MSG = "Error occurred while updating throttle data in the " + - "database"; - public static final String THROTTLE_DATA_RETRIEVE_ERROR_MSG = "Error occurred while retrieving throttle data " + - "from the database"; - public static final String THROTTLE_DATA_RESULT_SET_RETRIEVE_ERROR_MSG = "Error occurred while processing the " + - "throttle data result set retrieval"; - public static final String THROTTLE_DATA_DELETE_ERROR_MSG = "Error occurred while deleting throttle data " + - "from the database"; - public static final String NO_RECORDS_FOUND_ERROR_MSG = "No records are found for the given inputs"; -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/exception/OBThrottlerDataDeletionException.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/exception/OBThrottlerDataDeletionException.java deleted file mode 100644 index 7963f35e..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/exception/OBThrottlerDataDeletionException.java +++ /dev/null @@ -1,36 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.dao.exception; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; - -/** - * OB Throttler data deletion exception class. - */ -public class OBThrottlerDataDeletionException extends OpenBankingException { - - public OBThrottlerDataDeletionException(String message) { - - super(message); - } - - public OBThrottlerDataDeletionException(String message, Throwable e) { - - super(message, e); - } -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/exception/OBThrottlerDataInsertionException.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/exception/OBThrottlerDataInsertionException.java deleted file mode 100644 index 5c15f6b0..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/exception/OBThrottlerDataInsertionException.java +++ /dev/null @@ -1,36 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.dao.exception; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; - -/** - * OB Throttler data insertion exception class. - */ -public class OBThrottlerDataInsertionException extends OpenBankingException { - - public OBThrottlerDataInsertionException(String message) { - - super(message); - } - - public OBThrottlerDataInsertionException(String message, Throwable e) { - - super(message, e); - } -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/exception/OBThrottlerDataRetrievalException.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/exception/OBThrottlerDataRetrievalException.java deleted file mode 100644 index d4f253d2..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/exception/OBThrottlerDataRetrievalException.java +++ /dev/null @@ -1,36 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.dao.exception; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; - -/** - * OB Throttler data retrieval exception class. - */ -public class OBThrottlerDataRetrievalException extends OpenBankingException { - - public OBThrottlerDataRetrievalException(String message) { - - super(message); - } - - public OBThrottlerDataRetrievalException(String message, Throwable e) { - - super(message, e); - } -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/exception/OBThrottlerDataUpdationException.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/exception/OBThrottlerDataUpdationException.java deleted file mode 100644 index a75a41e6..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/exception/OBThrottlerDataUpdationException.java +++ /dev/null @@ -1,36 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.dao.exception; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; - -/** - * OB Throttler data updation exception class. - */ -public class OBThrottlerDataUpdationException extends OpenBankingException { - - public OBThrottlerDataUpdationException(String message) { - - super(message); - } - - public OBThrottlerDataUpdationException(String message, Throwable e) { - - super(message, e); - } -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/impl/OBThrottlerDAOImpl.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/impl/OBThrottlerDAOImpl.java deleted file mode 100644 index f6782efe..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/impl/OBThrottlerDAOImpl.java +++ /dev/null @@ -1,232 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.dao.impl; - -import com.wso2.openbanking.accelerator.throttler.dao.OBThrottlerDAO; -import com.wso2.openbanking.accelerator.throttler.dao.constants.OBThrottlerDAOConstants; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataDeletionException; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataInsertionException; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataRetrievalException; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataUpdationException; -import com.wso2.openbanking.accelerator.throttler.dao.model.ThrottleDataModel; -import com.wso2.openbanking.accelerator.throttler.dao.queries.OBThrottlerSQLStatements; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.sql.Timestamp; - -/** - * Implementation of OBThrottlerDAO. - */ -public class OBThrottlerDAOImpl implements OBThrottlerDAO { - - private static final Log log = LogFactory.getLog(OBThrottlerDAOImpl.class); - private OBThrottlerSQLStatements sqlStatements; - - /** - * Load sql statements specific to database type. - * - * @param sqlStatements -sqlStatements specific to db type - */ - public OBThrottlerDAOImpl(OBThrottlerSQLStatements sqlStatements) { - - this.sqlStatements = sqlStatements; - } - - /** - * {@inheritDoc} - */ - @Override - public ThrottleDataModel storeThrottleData(Connection connection, String throttleGroup, String throttleParam, - Timestamp currentTimestamp, Timestamp unlockTimestamp) - throws OBThrottlerDataInsertionException { - - String storeThrottleDataSql = sqlStatements.storeThrottleData(); - int rowCount; - - //store data - try (PreparedStatement storePreparedStatement = connection.prepareStatement(storeThrottleDataSql)) { - //Set prepared statement parameters - storePreparedStatement.setString(1, throttleGroup); - storePreparedStatement.setString(2, throttleParam); - storePreparedStatement.setTimestamp(3, currentTimestamp); - storePreparedStatement.setTimestamp(4, unlockTimestamp); - storePreparedStatement.setInt(5, OBThrottlerDAOConstants.FIRST_OCCURRENCE); - - rowCount = storePreparedStatement.executeUpdate(); - } catch (SQLException e) { - log.error(OBThrottlerDAOConstants.THROTTLE_DATA_STORE_ERROR_MSG); - throw new OBThrottlerDataInsertionException(OBThrottlerDAOConstants.THROTTLE_DATA_STORE_ERROR_MSG, e); - } - - if (rowCount > 0) { - if (log.isDebugEnabled()) { - log.debug(String.format( - "Stored ThrottleGroup: '%s', ThrottleParam: '%s', CurrentTimestamp: '%s', " + - "UnlockTimestamp: '%s', Occurrences: 1", throttleGroup, throttleParam, - currentTimestamp, unlockTimestamp).replaceAll("[\r\n]", "")); - } - return new ThrottleDataModel(throttleGroup, throttleParam, currentTimestamp, unlockTimestamp, 1); - } else { - throw new OBThrottlerDataInsertionException("Failed to properly persist throttle data in database."); - } - } - - /** - * {@inheritDoc} - */ - @Override - public ThrottleDataModel updateThrottleData(Connection connection, String throttleGroup, String throttleParam, - Timestamp currentTimestamp, Timestamp unlockTimestamp, int occurrences) - throws OBThrottlerDataUpdationException { - - String updateThrottleDataSql = sqlStatements.updateThrottleData(); - int rowCount; - - //update database - try (PreparedStatement updatePreparedStatement = connection.prepareStatement(updateThrottleDataSql)) { - //Set prepared statement parameters - updatePreparedStatement.setTimestamp(1, currentTimestamp); - updatePreparedStatement.setTimestamp(2, unlockTimestamp); - updatePreparedStatement.setInt(3, occurrences); - updatePreparedStatement.setString(4, throttleGroup); - updatePreparedStatement.setString(5, throttleParam); - rowCount = updatePreparedStatement.executeUpdate(); - } catch (SQLException e) { - log.error(OBThrottlerDAOConstants.THROTTLE_DATA_UPDATE_ERROR_MSG); - throw new OBThrottlerDataUpdationException(OBThrottlerDAOConstants.THROTTLE_DATA_UPDATE_ERROR_MSG, e); - } - - if (rowCount > 0) { - if (log.isDebugEnabled()) { - log.debug(String.format( - "Updated ThrottleGroup: '%s', ThrottleParam: '%s', CurrentTimestamp: '%s', " + - "UnlockTimestamp: '%s', Occurrences: %d", throttleGroup, throttleParam, - currentTimestamp, unlockTimestamp, occurrences).replaceAll("[\r\n]", "")); - } - return new ThrottleDataModel(throttleGroup, throttleParam, currentTimestamp, unlockTimestamp, occurrences); - } else { - throw new OBThrottlerDataUpdationException("Failed to properly update throttle data in database."); - } - } - - /** - * {@inheritDoc} - */ - @Override - public ThrottleDataModel getThrottleData(Connection connection, String throttleGroup, String throttleParam) - throws OBThrottlerDataRetrievalException { - - ThrottleDataModel throttleDataModel; - String sql = sqlStatements.retrieveThrottleData(); - - //retrieve data from database - try (PreparedStatement preparedStatement = connection.prepareStatement(sql)) { - preparedStatement.setString(1, throttleGroup); - preparedStatement.setString(2, throttleParam); - try (ResultSet resultSet = preparedStatement.executeQuery()) { - if (resultSet.next()) { - Timestamp lastUpdateTimestamp = resultSet. - getTimestamp(OBThrottlerDAOConstants.LAST_UPDATE_TIMESTAMP); - Timestamp unlockTimestamp = resultSet.getTimestamp(OBThrottlerDAOConstants.UNLOCK_TIMESTAMP); - int occurrences = resultSet.getInt(OBThrottlerDAOConstants.OCCURRENCES); - throttleDataModel = new ThrottleDataModel(throttleGroup, throttleParam, - lastUpdateTimestamp, unlockTimestamp, occurrences); - } else { - log.error(OBThrottlerDAOConstants.NO_RECORDS_FOUND_ERROR_MSG); - throw new OBThrottlerDataRetrievalException(OBThrottlerDAOConstants.NO_RECORDS_FOUND_ERROR_MSG); - } - } catch (SQLException e) { - throw new OBThrottlerDataRetrievalException( - OBThrottlerDAOConstants.THROTTLE_DATA_RESULT_SET_RETRIEVE_ERROR_MSG, e); - } - } catch (SQLException e) { - log.error(OBThrottlerDAOConstants.THROTTLE_DATA_RETRIEVE_ERROR_MSG); - throw new OBThrottlerDataRetrievalException(OBThrottlerDAOConstants.THROTTLE_DATA_RETRIEVE_ERROR_MSG, e); - } - return throttleDataModel; - } - - /** - * {@inheritDoc} - */ - @Override - public void deleteThrottleData(Connection connection, String throttleGroup, String throttleParam) - throws OBThrottlerDataDeletionException { - - String removeThrottleDataSql = sqlStatements.removeThrottleData(); - int rowCount; - - //remove data from database - try (PreparedStatement removePreparedStatement = connection.prepareStatement(removeThrottleDataSql)) { - //Set prepared statement parameters - removePreparedStatement.setString(1, throttleGroup); - removePreparedStatement.setString(2, throttleParam); - rowCount = removePreparedStatement.executeUpdate(); - } catch (SQLException e) { - log.error(OBThrottlerDAOConstants.THROTTLE_DATA_DELETE_ERROR_MSG); - throw new OBThrottlerDataDeletionException(OBThrottlerDAOConstants.THROTTLE_DATA_DELETE_ERROR_MSG, e); - } - - if (rowCount > 0) { - if (log.isDebugEnabled()) { - log.debug(String.format("Removed ThrottleGroup: '%s', ThrottleParam: '%s'", - throttleGroup, throttleParam).replaceAll("[\r\n]", "")); - } - } else { - throw new OBThrottlerDataDeletionException(String.format("Throttle data for %s:%s does not exist in " + - "the database", throttleGroup, throttleParam)); - } - } - - /** - * {@inheritDoc} - */ - @Override - public boolean isThrottleDataExists(Connection connection, String throttleGroup, String throttleParam) - throws OBThrottlerDataRetrievalException { - - boolean throttleDataExists = false; - String sql = sqlStatements.isThrottleDataExists(); - - //retrieve data from database - try (PreparedStatement preparedStatement = connection.prepareStatement(sql)) { - preparedStatement.setString(1, throttleGroup); - preparedStatement.setString(2, throttleParam); - try (ResultSet resultSet = preparedStatement.executeQuery()) { - if (resultSet.next()) { - throttleDataExists = resultSet.getInt(1) > 0; - } else { - log.error(OBThrottlerDAOConstants.NO_RECORDS_FOUND_ERROR_MSG); - throw new OBThrottlerDataRetrievalException(OBThrottlerDAOConstants.NO_RECORDS_FOUND_ERROR_MSG); - } - } catch (SQLException e) { - throw new OBThrottlerDataRetrievalException( - OBThrottlerDAOConstants.THROTTLE_DATA_RESULT_SET_RETRIEVE_ERROR_MSG, e); - } - } catch (SQLException e) { - log.error(OBThrottlerDAOConstants.THROTTLE_DATA_RETRIEVE_ERROR_MSG); - throw new OBThrottlerDataRetrievalException(OBThrottlerDAOConstants.THROTTLE_DATA_RETRIEVE_ERROR_MSG, e); - } - return throttleDataExists; - } -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/model/ThrottleDataModel.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/model/ThrottleDataModel.java deleted file mode 100644 index bc5b810f..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/model/ThrottleDataModel.java +++ /dev/null @@ -1,100 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.dao.model; - -import java.sql.Timestamp; - -/** - * DTO class for throttle data. - */ -public class ThrottleDataModel { - - private String throttleGroup; - private String throttleParam; - private Timestamp lastUpdateTimestamp; - private Timestamp unlockTimestamp; - private int occurrences; - - /** - * Constructor. - * - * @param throttleGroup - Throttle group - * @param throttleParam - Throttle parameter - * @param lastUpdateTimestamp - Updated timestamp - * @param unlockTimestamp - Parameter unlocking timestamp - * @param occurrences - number of occurrences of the parameter - */ - public ThrottleDataModel(String throttleGroup, String throttleParam, Timestamp lastUpdateTimestamp, - Timestamp unlockTimestamp, int occurrences) { - - this.throttleGroup = throttleGroup; - this.throttleParam = throttleParam; - this.lastUpdateTimestamp = new Timestamp(lastUpdateTimestamp.getTime()); - this.unlockTimestamp = new Timestamp(unlockTimestamp.getTime()); - this.occurrences = occurrences; - } - - public String getThrottleGroup() { - - return throttleGroup; - } - - public void setThrottleGroup(String throttleGroup) { - - this.throttleGroup = throttleGroup; - } - - public String getThrottleParam() { - return throttleParam; - } - - public void setThrottleParam(String throttleParam) { - - this.throttleParam = throttleParam; - } - - public Timestamp getLastUpdateTimestamp() { - - return new Timestamp(lastUpdateTimestamp.getTime()); - } - - public void setLastUpdateTimestamp(Timestamp lastUpdateTimestamp) { - - this.lastUpdateTimestamp = new Timestamp(lastUpdateTimestamp.getTime()); - } - - public Timestamp getUnlockTimestamp() { - - return new Timestamp(unlockTimestamp.getTime()); - } - - public void setUnlockTimestamp(Timestamp unlockTimestamp) { - - this.unlockTimestamp = new Timestamp(unlockTimestamp.getTime()); - } - - public int getOccurrences() { - - return occurrences; - } - - public void setOccurrences(int occurrences) { - - this.occurrences = occurrences; - } -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/persistence/DataStoreInitializer.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/persistence/DataStoreInitializer.java deleted file mode 100644 index ce4c2610..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/persistence/DataStoreInitializer.java +++ /dev/null @@ -1,44 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.dao.persistence; - -import com.wso2.openbanking.accelerator.common.exception.OBThrottlerException; -import com.wso2.openbanking.accelerator.throttler.dao.OBThrottlerDAO; - -/** - * This class handles throttler DAO layer initiation with the relevant SQL statements per database type. - */ -public class DataStoreInitializer { - - private static OBThrottlerDAO obThrottlerDAO = null; - - /** - * Initialize the DAO according to the database connection. - * - * @return OBThrottlerDAO - * @throws OBThrottlerException - OBThrottlerException - */ - public static synchronized OBThrottlerDAO initializeOBThrottlerDAO() throws OBThrottlerException { - - if (obThrottlerDAO == null) { - DataStoreInitializerFactory dataStoreInitializerFactory = new DataStoreInitializerFactory(); - obThrottlerDAO = dataStoreInitializerFactory.initializeDataStore(); - } - return obThrottlerDAO; - } -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/persistence/DataStoreInitializerFactory.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/persistence/DataStoreInitializerFactory.java deleted file mode 100644 index b60702c3..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/persistence/DataStoreInitializerFactory.java +++ /dev/null @@ -1,54 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.dao.persistence; - -import com.wso2.openbanking.accelerator.common.exception.OBThrottlerException; -import com.wso2.openbanking.accelerator.common.persistence.JDBCPersistenceManager; -import com.wso2.openbanking.accelerator.throttler.dao.OBThrottlerDAO; -import com.wso2.openbanking.accelerator.throttler.dao.impl.OBThrottlerDAOImpl; -import com.wso2.openbanking.accelerator.throttler.dao.queries.OBThrottlerSQLStatements; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.SQLException; - -/** - * Data Store Initializer Factory class. - */ -public class DataStoreInitializerFactory { - - private static final Log log = LogFactory.getLog(DataStoreInitializerFactory.class); - - public OBThrottlerDAO initializeDataStore() throws OBThrottlerException { - - try (Connection connection = JDBCPersistenceManager.getInstance().getDBConnection()) { - String driverName = connection.getMetaData().getDriverName(); - - if (log.isDebugEnabled()) { - log.debug("Initiated OBThrottlerDAO with " + driverName.replaceAll("[\r\n]", "")); - } - // returning default queries for all database types - return new OBThrottlerDAOImpl(new OBThrottlerSQLStatements()); - - } catch (SQLException e) { - log.error(String.format("Error while getting the database connection. %s", e).replaceAll("[\r\n]", "")); - throw new OBThrottlerException("Error while getting the database connection : ", e); - } - } -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/queries/OBThrottlerSQLStatements.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/queries/OBThrottlerSQLStatements.java deleted file mode 100644 index 778c50cd..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/java/com/wso2/openbanking/accelerator/throttler/dao/queries/OBThrottlerSQLStatements.java +++ /dev/null @@ -1,51 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.dao.queries; - -/** - * SQL Statements required for OB Throttler. - */ -public class OBThrottlerSQLStatements { - - public String storeThrottleData() { - - return "INSERT INTO OB_THROTTLE_DATA (THROTTLE_GROUP, THROTTLE_PARAM, LAST_UPDATE_TIMESTAMP, " + - "UNLOCK_TIMESTAMP, OCCURRENCES) VALUES (?, ?, ?, ?, ?)"; - } - - public String updateThrottleData() { - - return "UPDATE OB_THROTTLE_DATA SET LAST_UPDATE_TIMESTAMP = ?, UNLOCK_TIMESTAMP = ?, OCCURRENCES = ? " + - "WHERE THROTTLE_GROUP = ? AND THROTTLE_PARAM = ?"; - } - - public String retrieveThrottleData() { - - return "SELECT * FROM OB_THROTTLE_DATA WHERE THROTTLE_GROUP = ? AND THROTTLE_PARAM = ?"; - } - - public String removeThrottleData() { - - return "DELETE FROM OB_THROTTLE_DATA WHERE THROTTLE_GROUP = ? AND THROTTLE_PARAM = ?"; - } - - public String isThrottleDataExists() { - - return "SELECT COUNT(1) FROM OB_THROTTLE_DATA WHERE THROTTLE_GROUP = ? AND THROTTLE_PARAM = ?"; - } -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/resources/findbugs-exclude.xml b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/resources/findbugs-exclude.xml deleted file mode 100644 index c4f8e532..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/resources/findbugs-exclude.xml +++ /dev/null @@ -1,21 +0,0 @@ - - - - - diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/resources/findbugs-include.xml b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/resources/findbugs-include.xml deleted file mode 100644 index 8932a22e..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/java/com/wso2/openbanking/accelerator/throttler/dao/impl/OBThrottlerDAOTests.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/java/com/wso2/openbanking/accelerator/throttler/dao/impl/OBThrottlerDAOTests.java deleted file mode 100644 index 802996d6..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/java/com/wso2/openbanking/accelerator/throttler/dao/impl/OBThrottlerDAOTests.java +++ /dev/null @@ -1,275 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.dao.impl; - -import com.wso2.openbanking.accelerator.throttler.dao.OBThrottlerDAO; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataDeletionException; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataInsertionException; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataRetrievalException; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataUpdationException; -import com.wso2.openbanking.accelerator.throttler.dao.model.ThrottleDataModel; -import com.wso2.openbanking.accelerator.throttler.dao.queries.OBThrottlerSQLStatements; -import com.wso2.openbanking.accelerator.throttler.dao.util.OBThrottlerDAOTestData; -import com.wso2.openbanking.accelerator.throttler.dao.util.OBThrottlerDAOUtils; -import org.mockito.Mockito; -import org.testng.Assert; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.DataProvider; -import org.testng.annotations.Test; - -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.sql.Timestamp; - -/** - * Test for Open Banking throttler DAO. - */ -public class OBThrottlerDAOTests { - - private static final String DB_NAME = "OB_THROTTLE_DB"; - - private OBThrottlerDAO obThrottlerDAO; - private Connection mockedConnection; - private PreparedStatement mockedPreparedStatement; - private ResultSet mockedResultSet; - private ThrottleDataModel storedThrottleDataModel; - - @BeforeClass - public void initTest() throws Exception { - - OBThrottlerDAOUtils.initializeDataSource(DB_NAME, OBThrottlerDAOUtils.getFilePath("dbScripts/h2.sql")); - obThrottlerDAO = new OBThrottlerDAOImpl(new OBThrottlerSQLStatements()); - mockedConnection = Mockito.mock(Connection.class); - mockedPreparedStatement = Mockito.mock(PreparedStatement.class); - mockedResultSet = Mockito.mock(ResultSet.class); - } - - @DataProvider(name = "sampleOBThrottleDataProvider") - public Object[][] provideOBThrottleData() { - - /* - * throttleGroup - * throttleParam - * currentTimestamp - * unlockTimestamp - * occurrences - */ - return OBThrottlerDAOTestData.DataProviders.OB_THROTTLER_DATA_HOLDER; - } - - // data insertion tests - @Test(dataProvider = "sampleOBThrottleDataProvider") - public void testStoreThrottleData(String throttleGroup, String throttleParam, Timestamp currentTimestamp, - Timestamp unlockTimestamp, int occurrences) throws Exception { - - try (Connection connection = OBThrottlerDAOUtils.getConnection(DB_NAME)) { - - storedThrottleDataModel = obThrottlerDAO.storeThrottleData(connection, throttleGroup, throttleParam, - currentTimestamp, unlockTimestamp); - } - Assert.assertNotNull(storedThrottleDataModel); - Assert.assertNotNull(storedThrottleDataModel.getThrottleGroup()); - Assert.assertNotNull(storedThrottleDataModel.getThrottleParam()); - Assert.assertNotNull(storedThrottleDataModel.getLastUpdateTimestamp()); - Assert.assertNotNull(storedThrottleDataModel.getUnlockTimestamp()); - } - - @Test(expectedExceptions = OBThrottlerDataInsertionException.class) - public void testStoreThrottleDataInsertionError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(0).when(mockedPreparedStatement).executeUpdate(); - - obThrottlerDAO.storeThrottleData(mockedConnection, OBThrottlerDAOTestData.THROTTLE_GROUP, - OBThrottlerDAOTestData.THROTTLE_PARAM, OBThrottlerDAOTestData.CURRENT_TIMESTAMP, - OBThrottlerDAOTestData.UNLOCK_TIMESTAMP); - } - - @Test(expectedExceptions = OBThrottlerDataInsertionException.class) - public void testStoreThrottleDataSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - obThrottlerDAO.storeThrottleData(mockedConnection, OBThrottlerDAOTestData.THROTTLE_GROUP, - OBThrottlerDAOTestData.THROTTLE_PARAM, OBThrottlerDAOTestData.CURRENT_TIMESTAMP, - OBThrottlerDAOTestData.UNLOCK_TIMESTAMP); - } - - //data updation tests - @Test(dataProvider = "sampleOBThrottleDataProvider", dependsOnMethods = "testStoreThrottleData") - public void testUpdateThrottleData(String throttleGroup, String throttleParam, Timestamp currentTimestamp, - Timestamp unlockTimestamp, int occurrences) throws Exception { - - ThrottleDataModel updatedThrottleDataModel; - - try (Connection connection = OBThrottlerDAOUtils.getConnection(DB_NAME)) { - - updatedThrottleDataModel = obThrottlerDAO.updateThrottleData(connection, - storedThrottleDataModel.getThrottleGroup(), storedThrottleDataModel.getThrottleParam(), - storedThrottleDataModel.getLastUpdateTimestamp(), storedThrottleDataModel.getUnlockTimestamp(), - occurrences + 1); - } - Assert.assertEquals(updatedThrottleDataModel.getOccurrences(), 2); - } - - @Test(expectedExceptions = OBThrottlerDataUpdationException.class) - public void testUpdateThrottleDataError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(0).when(mockedPreparedStatement).executeUpdate(); - - obThrottlerDAO.updateThrottleData(mockedConnection, OBThrottlerDAOTestData.THROTTLE_GROUP, - OBThrottlerDAOTestData.THROTTLE_PARAM, OBThrottlerDAOTestData.CURRENT_TIMESTAMP, - OBThrottlerDAOTestData.UNLOCK_TIMESTAMP, 1); - } - - @Test(expectedExceptions = OBThrottlerDataUpdationException.class) - public void testUpdateThrottleDataSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - obThrottlerDAO.updateThrottleData(mockedConnection, OBThrottlerDAOTestData.THROTTLE_GROUP, - OBThrottlerDAOTestData.THROTTLE_PARAM, OBThrottlerDAOTestData.CURRENT_TIMESTAMP, - OBThrottlerDAOTestData.UNLOCK_TIMESTAMP, 1); - } - - //data retrieval tests - @Test(dependsOnMethods = "testUpdateThrottleData") - public void testRetrieveThrottleData() throws Exception { - - ThrottleDataModel retrievedThrottleDataModel; - - try (Connection connection = OBThrottlerDAOUtils.getConnection(DB_NAME)) { - retrievedThrottleDataModel = obThrottlerDAO.getThrottleData(connection, - storedThrottleDataModel.getThrottleGroup(), storedThrottleDataModel.getThrottleParam()); - } - - Assert.assertNotNull(retrievedThrottleDataModel); - Assert.assertEquals(retrievedThrottleDataModel.getThrottleGroup(), storedThrottleDataModel.getThrottleGroup()); - Assert.assertNotNull(retrievedThrottleDataModel.getThrottleParam()); - Assert.assertEquals(retrievedThrottleDataModel.getThrottleParam(), storedThrottleDataModel.getThrottleParam()); - Assert.assertNotNull(retrievedThrottleDataModel.getUnlockTimestamp()); - Assert.assertNotNull(retrievedThrottleDataModel.getLastUpdateTimestamp()); - Assert.assertNotNull(retrievedThrottleDataModel.getOccurrences()); - } - - @Test(dependsOnMethods = "testUpdateThrottleData") - public void testRetrieveThrottleDataExists() throws Exception { - - Boolean isRetrieved; - - try (Connection connection = OBThrottlerDAOUtils.getConnection(DB_NAME)) { - isRetrieved = obThrottlerDAO.isThrottleDataExists(connection, - storedThrottleDataModel.getThrottleGroup(), storedThrottleDataModel.getThrottleParam()); - } - - Assert.assertTrue(isRetrieved); - } - - @Test(expectedExceptions = OBThrottlerDataRetrievalException.class) - public void testRetrieveThrottleDataError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(0).when(mockedPreparedStatement).executeUpdate(); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - obThrottlerDAO.getThrottleData(mockedConnection, OBThrottlerDAOTestData.THROTTLE_GROUP, - OBThrottlerDAOTestData.THROTTLE_PARAM); - } - - @Test (expectedExceptions = OBThrottlerDataRetrievalException.class) - public void testRetrieveThrottleDataResultSetError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(mockedResultSet).when(mockedPreparedStatement).executeQuery(); - Mockito.doReturn(false).when(mockedResultSet).next(); - obThrottlerDAO.getThrottleData(mockedConnection, OBThrottlerDAOTestData.THROTTLE_GROUP, - OBThrottlerDAOTestData.THROTTLE_PARAM); - } - - @Test(expectedExceptions = OBThrottlerDataRetrievalException.class) - public void testRetrieveThrottleDataExistsError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(0).when(mockedPreparedStatement).executeUpdate(); - Mockito.doThrow(SQLException.class).when(mockedPreparedStatement).executeQuery(); - obThrottlerDAO.isThrottleDataExists(mockedConnection, OBThrottlerDAOTestData.THROTTLE_GROUP, - OBThrottlerDAOTestData.THROTTLE_PARAM); - } - - @Test (expectedExceptions = OBThrottlerDataRetrievalException.class) - public void testRetrieveThrottleDataExistsResultSetError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(mockedResultSet).when(mockedPreparedStatement).executeQuery(); - Mockito.doReturn(false).when(mockedResultSet).next(); - obThrottlerDAO.isThrottleDataExists(mockedConnection, OBThrottlerDAOTestData.THROTTLE_GROUP, - OBThrottlerDAOTestData.THROTTLE_PARAM); - } - - @Test(expectedExceptions = OBThrottlerDataRetrievalException.class) - public void testRetrieveThrottleDataSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - obThrottlerDAO.getThrottleData(mockedConnection, OBThrottlerDAOTestData.THROTTLE_GROUP, - OBThrottlerDAOTestData.THROTTLE_PARAM); - } - - @Test(expectedExceptions = OBThrottlerDataRetrievalException.class) - public void testRetrieveThrottleDataExistsSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - obThrottlerDAO.isThrottleDataExists(mockedConnection, OBThrottlerDAOTestData.THROTTLE_GROUP, - OBThrottlerDAOTestData.THROTTLE_PARAM); - } - - //data deletion tests - @Test(dependsOnMethods = "testRetrieveThrottleData") - public void testDeleteConsentAttribute() throws Exception { - - try (Connection connection = OBThrottlerDAOUtils.getConnection(DB_NAME)) { - - obThrottlerDAO.deleteThrottleData(connection, storedThrottleDataModel.getThrottleGroup(), - storedThrottleDataModel.getThrottleParam()); - } - } - - @Test(expectedExceptions = OBThrottlerDataDeletionException.class) - public void testDeleteThrottleDataError() throws Exception { - - Mockito.doReturn(mockedPreparedStatement).when(mockedConnection) - .prepareStatement(Mockito.anyString()); - Mockito.doReturn(0).when(mockedPreparedStatement).executeUpdate(); - - obThrottlerDAO.deleteThrottleData(mockedConnection, OBThrottlerDAOTestData.THROTTLE_GROUP, - OBThrottlerDAOTestData.THROTTLE_PARAM); - } - - @Test(expectedExceptions = OBThrottlerDataDeletionException.class) - public void testDeleteThrottleDataSQLError() throws Exception { - - Mockito.doThrow(SQLException.class).when(mockedConnection).prepareStatement(Mockito.anyString()); - obThrottlerDAO.deleteThrottleData(mockedConnection, OBThrottlerDAOTestData.THROTTLE_GROUP, - OBThrottlerDAOTestData.THROTTLE_PARAM); - } -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/java/com/wso2/openbanking/accelerator/throttler/dao/util/OBThrottlerDAOTestData.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/java/com/wso2/openbanking/accelerator/throttler/dao/util/OBThrottlerDAOTestData.java deleted file mode 100644 index ae425b30..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/java/com/wso2/openbanking/accelerator/throttler/dao/util/OBThrottlerDAOTestData.java +++ /dev/null @@ -1,61 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.dao.util; - -import java.sql.Timestamp; -import java.util.Date; - -/** - * Test data for Open Banking throttler DAO. - */ -public class OBThrottlerDAOTestData { - - public static final String THROTTLE_GROUP = "OBIdentifierAuthenticator"; - - public static final String THROTTLE_PARAM = "user-ip-192.168.1.1"; - - public static final Timestamp CURRENT_TIMESTAMP = new Timestamp(new Date().getTime()); - - public static final Timestamp UNLOCK_TIMESTAMP = new Timestamp(CURRENT_TIMESTAMP.getTime() + (1000L * 180)); - - public static final int OCCURRENCES = 1; - - /** - * Data provider. - */ - public static final class DataProviders { - - /* - * throttleGroup - * throttleParam - * currentTimestamp - * unlockTimestamp - * occurrences - */ - public static final Object[][] OB_THROTTLER_DATA_HOLDER = new Object[][]{ - - { - THROTTLE_GROUP, - THROTTLE_PARAM, - CURRENT_TIMESTAMP, - UNLOCK_TIMESTAMP, - OCCURRENCES - } - }; - } -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/java/com/wso2/openbanking/accelerator/throttler/dao/util/OBThrottlerDAOUtils.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/java/com/wso2/openbanking/accelerator/throttler/dao/util/OBThrottlerDAOUtils.java deleted file mode 100644 index 63a3e914..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/java/com/wso2/openbanking/accelerator/throttler/dao/util/OBThrottlerDAOUtils.java +++ /dev/null @@ -1,63 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.dao.util; - -import org.apache.commons.dbcp.BasicDataSource; -import org.apache.commons.lang3.StringUtils; - -import java.nio.file.Paths; -import java.sql.Connection; -import java.sql.SQLException; -import java.util.HashMap; -import java.util.Map; - -/** - * Test for Open Banking throttler DAO utils. - */ -public class OBThrottlerDAOUtils { - - private static Map dataSourceMap = new HashMap<>(); - - public static void initializeDataSource(String databaseName, String scriptPath) throws Exception { - BasicDataSource dataSource = new BasicDataSource(); - dataSource.setDriverClassName("org.h2.Driver"); - dataSource.setUsername("username"); - dataSource.setPassword("password"); - dataSource.setUrl("jdbc:h2:mem:" + databaseName); - - try (Connection connection = dataSource.getConnection()) { - connection.createStatement().executeUpdate("RUNSCRIPT FROM '" + scriptPath + "'"); - } - dataSourceMap.put(databaseName, dataSource); - } - - public static Connection getConnection(String database) throws SQLException { - if (dataSourceMap.get(database) != null) { - return dataSourceMap.get(database).getConnection(); - } - throw new RuntimeException("Invalid datasource."); - } - - public static String getFilePath(String fileName) { - if (StringUtils.isNotBlank(fileName)) { - return Paths.get(System.getProperty("user.dir"), "src", "test", "resources", fileName) - .toString(); - } - return null; - } -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/resources/dbScripts/h2.sql b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/resources/dbScripts/h2.sql deleted file mode 100644 index 3bcb12cf..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/resources/dbScripts/h2.sql +++ /dev/null @@ -1,8 +0,0 @@ -CREATE TABLE IF NOT EXISTS OB_THROTTLE_DATA ( - THROTTLE_GROUP VARCHAR(100) NOT NULL, - THROTTLE_PARAM VARCHAR(100) NOT NULL, - LAST_UPDATE_TIMESTAMP DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, - UNLOCK_TIMESTAMP DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, - OCCURRENCES INTEGER NOT NULL, - PRIMARY KEY (THROTTLE_GROUP,THROTTLE_PARAM) -); diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/resources/testng.xml b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/resources/testng.xml deleted file mode 100644 index 7d5a867e..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao/src/test/resources/testng.xml +++ /dev/null @@ -1,27 +0,0 @@ - - - - - - - - - - - diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/pom.xml b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/pom.xml deleted file mode 100644 index 4dce7220..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/pom.xml +++ /dev/null @@ -1,213 +0,0 @@ - - - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../../pom.xml - - 4.0.0 - - com.wso2.openbanking.accelerator.throttler.service - WSO2 Open Banking - Throttler Service - WSO2 Open Banking - Throttler Service Module - bundle - - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.throttler.dao - provided - - - org.testng - testng - - - com.h2database - h2 - test - - - org.jacoco - org.jacoco.agent - runtime - test - - - org.mockito - mockito-all - test - - - org.powermock - powermock-api-mockito - test - - - org.powermock - powermock-module-testng - test - - - - - - - org.apache.maven.plugins - maven-surefire-plugin - - - src/test/resources/testng.xml - - - - - - org.jacoco - jacoco-maven-plugin - ${jacoco.version} - - - - **/*Constants.class - **/*Component.class - **/*DataHolder.class - - - - - default-prepare-agent - - prepare-agent - - - - default-prepare-agent-integration - - prepare-agent-integration - - - - default-report - - report - - - - default-report-integration - - report-integration - - - - default-check - - check - - - - - BUNDLE - - - INSTRUCTION - COVEREDRATIO - 0.80 - - - - - - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - true - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - org.apache.maven.plugins - maven-compiler-plugin - - 1.8 - 1.8 - UTF-8 - - - - org.apache.felix - maven-bundle-plugin - true - - - - ${project.artifactId} - - - com.wso2.openbanking.accelerator.throttler.service.internal - - - org.osgi.framework;version="${osgi.framework.imp.pkg.version.range}", - org.osgi.service.component;version="${osgi.service.component.imp.pkg.version.range}", - com.wso2.openbanking.accelerator.common.*;version="${project.version}", - org.apache.commons.lang3;version="${commons-lang.version}" - - - !com.wso2.openbanking.accelerator.throttler.service.internal, - com.wso2.openbanking.accelerator.throttler.service.*;version="${project.version}", - - * - <_dsannotations>* - - - - - - diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/java/com/wso2/openbanking/accelerator/throttler/service/OBThrottleService.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/java/com/wso2/openbanking/accelerator/throttler/service/OBThrottleService.java deleted file mode 100644 index 03641744..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/java/com/wso2/openbanking/accelerator/throttler/service/OBThrottleService.java +++ /dev/null @@ -1,245 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.service; - -import com.wso2.openbanking.accelerator.common.exception.OBThrottlerException; -import com.wso2.openbanking.accelerator.common.util.DatabaseUtil; -import com.wso2.openbanking.accelerator.throttler.dao.OBThrottlerDAO; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataDeletionException; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataInsertionException; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataRetrievalException; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataUpdationException; -import com.wso2.openbanking.accelerator.throttler.dao.model.ThrottleDataModel; -import com.wso2.openbanking.accelerator.throttler.dao.persistence.DataStoreInitializer; -import com.wso2.openbanking.accelerator.throttler.service.constants.OBThrottlerServiceConstants; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.Timestamp; -import java.util.Date; -import java.util.HashMap; -import java.util.Map; - -/** - * Service class for OB Throttler. - *

- * Contains methods required to throttle the occurrence of a given parameter. - * The parameters can be separated into groups by 'throttleGroup' attribute, which will - * allow throttling same parameter values in different groups. - */ -public class OBThrottleService { - - private static Log log = LogFactory.getLog(OBThrottleService.class); - protected Map> throttleDataMap = new HashMap<>(); - private static OBThrottleService instance = null; - - // private constructor - private OBThrottleService() { - } - - /** - * @return OBThrottleService instance - */ - public static synchronized OBThrottleService getInstance() { - - if (instance == null) { - instance = new OBThrottleService(); - } - return instance; - } - - /** - * Update throttle database and throttleDataMap. - * - * @param throttleGroup - throttle group - * @param throttleParam - throttle parameter - * @param throttleLimit - allowed number of occurrences - * @param throttleTimePeriod - time period that the parameter is throttled (seconds) - * @throws OBThrottlerException - OBThrottlerException - */ - public void updateThrottleData(String throttleGroup, String throttleParam, int throttleLimit, - int throttleTimePeriod) throws OBThrottlerException { - - ThrottleDataModel throttleDataModel; - ThrottleDataModel existingThrottleDataModel; - OBThrottlerDAO obThrottlerDAO = DataStoreInitializer.initializeOBThrottlerDAO(); - Timestamp currentTimestamp = new Timestamp(new Date().getTime()); - Timestamp unlockTimestamp = new Timestamp(currentTimestamp.getTime() + (1000L * throttleTimePeriod)); - - Connection connection = DatabaseUtil.getDBConnection(); - - try { - //remove expired data from database by checking throttle status - getThrottleStatus(connection, throttleGroup, throttleParam, obThrottlerDAO); - //check if throttle group and parameter exists. Add new record if not. - - if (obThrottlerDAO.isThrottleDataExists(connection, throttleGroup, throttleParam)) { - existingThrottleDataModel = obThrottlerDAO.getThrottleData(connection, throttleGroup, throttleParam); - //increment Occurrences - int updatedOccurrences = existingThrottleDataModel.getOccurrences() + 1; - throttleDataModel = obThrottlerDAO.updateThrottleData(connection, throttleGroup, throttleParam, - currentTimestamp, unlockTimestamp, updatedOccurrences); - } else { - throttleDataModel = obThrottlerDAO.storeThrottleData(connection, throttleGroup, throttleParam, - currentTimestamp, unlockTimestamp); - } - DatabaseUtil.commitTransaction(connection); - log.debug(OBThrottlerServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - if (throttleDataModel.getOccurrences() > throttleLimit) { - updateThrottleDataMap(throttleGroup, throttleParam, throttleDataModel.getUnlockTimestamp()); - } - } catch (OBThrottlerDataInsertionException e) { - log.error(OBThrottlerServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new OBThrottlerException(OBThrottlerServiceConstants.DATA_INSERTION_ROLLBACK_ERROR_MSG, e); - } catch (OBThrottlerDataUpdationException e) { - log.error(OBThrottlerServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new OBThrottlerException(OBThrottlerServiceConstants.DATA_UPDATE_ROLLBACK_ERROR_MSG, e); - } catch (OBThrottlerDataRetrievalException e) { - log.error(OBThrottlerServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new OBThrottlerException(OBThrottlerServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } catch (OBThrottlerDataDeletionException e) { - log.error(OBThrottlerServiceConstants.DATA_DELETE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new OBThrottlerException(OBThrottlerServiceConstants.DATA_DELETE_ROLLBACK_ERROR_MSG, e); - } finally { - log.debug(OBThrottlerServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - /** - * Check if the given parameter is throttled. - * - * @param throttleGroup - throttle group - * @param throttleParam - throttle parameter - * @return - boolean - * @throws OBThrottlerException - OBThrottlerDataDeletionException - */ - public boolean isThrottled(String throttleGroup, String throttleParam) throws OBThrottlerException { - - Connection connection = DatabaseUtil.getDBConnection(); - OBThrottlerDAO obThrottlerDAO = DataStoreInitializer.initializeOBThrottlerDAO(); - - try { - boolean throttleStatus = getThrottleStatus(connection, throttleGroup, throttleParam, obThrottlerDAO); - DatabaseUtil.commitTransaction(connection); - log.debug(OBThrottlerServiceConstants.TRANSACTION_COMMITTED_LOG_MSG); - return throttleStatus; - } catch (OBThrottlerDataDeletionException e) { - log.error(OBThrottlerServiceConstants.DATA_DELETE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new OBThrottlerException(OBThrottlerServiceConstants.DATA_DELETE_ROLLBACK_ERROR_MSG, e); - } finally { - log.debug(OBThrottlerServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - /** - * Delete the throttle data record from DB on a successful attempt. - * - * @param throttleGroup - throttle group - * @param throttleParam - throttle parameter - * @throws OBThrottlerException - OBThrottlerDataDeletionException, OBThrottlerDataRetrievalException - */ - public void deleteRecordOnSuccessAttempt(String throttleGroup, String throttleParam) throws OBThrottlerException { - - Connection connection = DatabaseUtil.getDBConnection(); - OBThrottlerDAO obThrottlerDAO = DataStoreInitializer.initializeOBThrottlerDAO(); - - try { - if (obThrottlerDAO.isThrottleDataExists(connection, throttleGroup, throttleParam)) { - obThrottlerDAO.deleteThrottleData(connection, throttleGroup, throttleParam); - DatabaseUtil.commitTransaction(connection); - } - } catch (OBThrottlerDataRetrievalException e) { - log.error(OBThrottlerServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new OBThrottlerException(OBThrottlerServiceConstants.DATA_RETRIEVE_ERROR_MSG, e); - } catch (OBThrottlerDataDeletionException e) { - log.error(OBThrottlerServiceConstants.DATA_DELETE_ROLLBACK_ERROR_MSG, e); - DatabaseUtil.rollbackTransaction(connection); - throw new OBThrottlerException(OBThrottlerServiceConstants.DATA_DELETE_ROLLBACK_ERROR_MSG, e); - } finally { - log.debug(OBThrottlerServiceConstants.DATABASE_CONNECTION_CLOSE_LOG_MSG); - DatabaseUtil.closeConnection(connection); - } - } - - /** - * Check if the given parameter is throttled. This method is overloaded. - * - * @param connection connection object - * @param throttleGroup - throttle group - * @param throttleParam - throttle parameter - * @return - boolean - * @throws OBThrottlerDataDeletionException - OBThrottlerDataDeletionException - */ - private boolean getThrottleStatus(Connection connection, String throttleGroup, String throttleParam, - OBThrottlerDAO obThrottlerDAO) throws OBThrottlerDataDeletionException { - - Map throttleParamMap; - if (throttleDataMap.containsKey(throttleGroup)) { - throttleParamMap = throttleDataMap.get(throttleGroup); - if (throttleParamMap.containsKey(throttleParam)) { - //check if the parameter is still locked - Timestamp currentTimestamp = new Timestamp(new Date().getTime()); - Timestamp unlockTimestamp = throttleParamMap.get(throttleParam); - if (unlockTimestamp.after(currentTimestamp)) { - return true; - } else { - //remove throttle parameter from throttle data map if expired - throttleParamMap.remove(throttleParam); - throttleDataMap.put(throttleGroup, throttleParamMap); - //remove from database - obThrottlerDAO.deleteThrottleData(connection, throttleGroup, throttleParam); - return false; - } - } else { - return false; - } - } else { - return false; - } - } - - /** - * Update throttle data map. - * - * @param throttleGroup - throttle group - * @param throttleParam - throttle parameter - * @param unlockTimestamp - timestamp that the parameter will be unlocked. - */ - protected void updateThrottleDataMap(String throttleGroup, String throttleParam, Timestamp unlockTimestamp) { - - Map throttleParamMap; - //check if throttle group already exists - if (throttleDataMap.containsKey(throttleGroup)) { - throttleParamMap = throttleDataMap.get(throttleGroup); - } else { - throttleParamMap = new HashMap<>(); - } - //put parameter and unlockTimestamp to the throttle data map - throttleParamMap.put(throttleParam, unlockTimestamp); - throttleDataMap.put(throttleGroup, throttleParamMap); - } -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/java/com/wso2/openbanking/accelerator/throttler/service/constants/OBThrottlerServiceConstants.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/java/com/wso2/openbanking/accelerator/throttler/service/constants/OBThrottlerServiceConstants.java deleted file mode 100644 index 7cfae883..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/java/com/wso2/openbanking/accelerator/throttler/service/constants/OBThrottlerServiceConstants.java +++ /dev/null @@ -1,35 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.service.constants; - -/** - * OB Throttler Service Constants. - */ -public class OBThrottlerServiceConstants { - - public static final String TRANSACTION_COMMITTED_LOG_MSG = "Transaction committed"; - public static final String DATABASE_CONNECTION_CLOSE_LOG_MSG = "Closing database connection"; - - public static final String DATA_INSERTION_ROLLBACK_ERROR_MSG = "Error occurred while inserting data. Rolling " + - "back the transaction"; - public static final String DATA_UPDATE_ROLLBACK_ERROR_MSG = "Error occurred while updating data. Rolling " + - "back the transaction"; - public static final String DATA_RETRIEVE_ERROR_MSG = "Error occurred while retrieving data"; - public static final String DATA_DELETE_ROLLBACK_ERROR_MSG = "Error occurred while deleting data. Rolling " + - "back the transaction"; -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/java/com/wso2/openbanking/accelerator/throttler/service/internal/OBThrottlerDataHolder.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/java/com/wso2/openbanking/accelerator/throttler/service/internal/OBThrottlerDataHolder.java deleted file mode 100644 index e8f49f8c..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/java/com/wso2/openbanking/accelerator/throttler/service/internal/OBThrottlerDataHolder.java +++ /dev/null @@ -1,52 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.service.internal; - -import org.wso2.carbon.user.core.service.RealmService; - -/** - * OBThrottler Data Holder. - */ -public class OBThrottlerDataHolder { - - private static OBThrottlerDataHolder instance = new OBThrottlerDataHolder(); - - private RealmService realmService; - - private OBThrottlerDataHolder() { - - } - - public static OBThrottlerDataHolder getInstance() { - - return instance; - } - - public RealmService getRealmService() { - - if (realmService == null) { - throw new RuntimeException("Realm Service is not available. Component did not start correctly."); - } - return realmService; - } - - void setRealmService(RealmService realmService) { - - this.realmService = realmService; - } -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/java/com/wso2/openbanking/accelerator/throttler/service/internal/OBThrottlerServiceComponent.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/java/com/wso2/openbanking/accelerator/throttler/service/internal/OBThrottlerServiceComponent.java deleted file mode 100644 index 052acacf..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/java/com/wso2/openbanking/accelerator/throttler/service/internal/OBThrottlerServiceComponent.java +++ /dev/null @@ -1,87 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.service.internal; - -import com.wso2.openbanking.accelerator.throttler.service.OBThrottleService; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.osgi.service.component.ComponentContext; -import org.osgi.service.component.annotations.Activate; -import org.osgi.service.component.annotations.Component; -import org.osgi.service.component.annotations.Deactivate; -import org.osgi.service.component.annotations.Reference; -import org.osgi.service.component.annotations.ReferenceCardinality; -import org.osgi.service.component.annotations.ReferencePolicy; -import org.wso2.carbon.context.PrivilegedCarbonContext; -import org.wso2.carbon.user.core.service.RealmService; - -/** - * OBThrottler component. - */ -@Component( - name = "open.banking.throttler.component", - immediate = true -) -public class OBThrottlerServiceComponent { - - private static final Log log = LogFactory.getLog(OBThrottlerServiceComponent.class); - - public static RealmService getRealmService() { - return (RealmService) PrivilegedCarbonContext.getThreadLocalCarbonContext() - .getOSGiService(RealmService.class); - } - - @Reference( - name = "realm.service", - service = RealmService.class, - cardinality = ReferenceCardinality.MANDATORY, - policy = ReferencePolicy.DYNAMIC, - unbind = "unsetRealmService" - ) - protected void setRealmService(RealmService realmService) { - - log.debug("Setting the Realm Service"); - OBThrottlerDataHolder.getInstance().setRealmService(realmService); - } - - @Activate - protected void activate(ComponentContext ctxt) { - - try { - OBThrottleService obThrottleService = OBThrottleService.getInstance(); - ctxt.getBundleContext().registerService(OBThrottleService.class.getName(), - obThrottleService, null); - log.debug("OBThrottleService bundle is activated"); - - } catch (Throwable e) { - log.error("OBThrottleService bundle activation Failed", e); - } - } - - @Deactivate - protected void deactivate(ComponentContext ctxt) { - - log.debug("OBThrottleService bundle is deactivated"); - } - - protected void unsetRealmService(RealmService realmService) { - - log.debug("UnSetting the Realm Service"); - OBThrottlerDataHolder.getInstance().setRealmService(null); - } -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/resources/findbugs-include.xml b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/resources/findbugs-include.xml deleted file mode 100644 index 8932a22e..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/test/java/com/wso2/openbanking/accelerator/throttler/service/OBThrottleServiceTests.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/test/java/com/wso2/openbanking/accelerator/throttler/service/OBThrottleServiceTests.java deleted file mode 100644 index a6b23b7e..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/test/java/com/wso2/openbanking/accelerator/throttler/service/OBThrottleServiceTests.java +++ /dev/null @@ -1,303 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.service; - -import com.wso2.openbanking.accelerator.common.exception.OBThrottlerException; -import com.wso2.openbanking.accelerator.common.util.DatabaseUtil; -import com.wso2.openbanking.accelerator.throttler.dao.OBThrottlerDAO; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataDeletionException; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataInsertionException; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataRetrievalException; -import com.wso2.openbanking.accelerator.throttler.dao.exception.OBThrottlerDataUpdationException; -import com.wso2.openbanking.accelerator.throttler.dao.model.ThrottleDataModel; -import com.wso2.openbanking.accelerator.throttler.dao.persistence.DataStoreInitializer; -import com.wso2.openbanking.accelerator.throttler.service.util.OBThrottleServiceTestData; -import org.mockito.Mockito; -import org.powermock.api.mockito.PowerMockito; -import org.powermock.core.classloader.annotations.PowerMockIgnore; -import org.powermock.core.classloader.annotations.PrepareForTest; -import org.testng.Assert; -import org.testng.IObjectFactory; -import org.testng.annotations.BeforeClass; -import org.testng.annotations.BeforeMethod; -import org.testng.annotations.ObjectFactory; -import org.testng.annotations.Test; - -import java.sql.Connection; -import java.sql.Timestamp; -import java.util.HashMap; - -/** - * Test for Open banking throttle service. - */ -@PowerMockIgnore("jdk.internal.reflect.*") -@PrepareForTest({DatabaseUtil.class, DataStoreInitializer.class}) -public class OBThrottleServiceTests { - - private OBThrottleService obThrottleService; - private OBThrottlerDAO mockedOBThrottlerDAO; - private Connection mockedConnection; - private ThrottleDataModel throttleDataModel; - - @BeforeClass - public void initTest() { - - obThrottleService = OBThrottleService.getInstance(); - mockedOBThrottlerDAO = Mockito.mock(OBThrottlerDAO.class); - mockedConnection = Mockito.mock(Connection.class); - throttleDataModel = Mockito.mock(ThrottleDataModel.class); - } - - @BeforeMethod - public void mock() throws OBThrottlerException { - - mockStaticClasses(); - } - - @ObjectFactory - public IObjectFactory getObjectFactory() { - - return new org.powermock.modules.testng.PowerMockObjectFactory(); - } - - @Test - public void testThrottledOutScenario() throws Exception { - - Mockito.doNothing().when(mockedOBThrottlerDAO).deleteThrottleData(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString()); - - obThrottleService.throttleDataMap.put(OBThrottleServiceTestData.THROTTLE_GROUP, - new HashMap() { - { - put(OBThrottleServiceTestData.THROTTLE_PARAM, - OBThrottleServiceTestData.UNLOCK_TIMESTAMP_GREATER_THAN_CURRENT_TIMESTAMP); - } - }); - - Boolean isThrottled = obThrottleService.isThrottled(OBThrottleServiceTestData.THROTTLE_GROUP, - OBThrottleServiceTestData.THROTTLE_PARAM); - - Assert.assertTrue(isThrottled); - } - - @Test - public void testNotThrottledOutScenario() throws Exception { - - Mockito.doNothing().when(mockedOBThrottlerDAO).deleteThrottleData(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString()); - - obThrottleService.throttleDataMap.put(OBThrottleServiceTestData.THROTTLE_SECOND_GROUP, - new HashMap() { - { - put(OBThrottleServiceTestData.THROTTLE_PARAM, - OBThrottleServiceTestData.UNLOCK_TIMESTAMP_LESS_THAN_CURRENT_TIMESTAMP); - } - }); - - Boolean isThrottled = obThrottleService.isThrottled( - OBThrottleServiceTestData.THROTTLE_SECOND_GROUP, OBThrottleServiceTestData.THROTTLE_PARAM); - - Assert.assertFalse(isThrottled); - } - - @Test(priority = 1) - public void testThrottleGroupNotInThrottleDataMap() throws Exception { - - Mockito.doNothing().when(mockedOBThrottlerDAO).deleteThrottleData(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString()); - - Boolean isThrottled = obThrottleService.isThrottled( - OBThrottleServiceTestData.THROTTLE_GROUP_BASIC_AUTH, OBThrottleServiceTestData.THROTTLE_PARAM); - - Assert.assertFalse(isThrottled); - } - - @Test(priority = 1) - public void testThrottleParamNotInThrottleDataMap() throws Exception { - - Mockito.doNothing().when(mockedOBThrottlerDAO).deleteThrottleData(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString()); - - Boolean isThrottled = obThrottleService.isThrottled( - OBThrottleServiceTestData.THROTTLE_SECOND_GROUP, OBThrottleServiceTestData.THROTTLE_SECOND_PARAM); - - Assert.assertFalse(isThrottled); - } - - @Test(priority = 2) - public void testThrottleGroupInUpdateThrottleDataMap() { - - obThrottleService.updateThrottleDataMap(OBThrottleServiceTestData.THROTTLE_SECOND_GROUP, - OBThrottleServiceTestData.THROTTLE_PARAM, - OBThrottleServiceTestData.UNLOCK_TIMESTAMP_GREATER_THAN_CURRENT_TIMESTAMP); - - Assert.assertTrue(obThrottleService.throttleDataMap - .containsKey(OBThrottleServiceTestData.THROTTLE_SECOND_GROUP)); - } - - @Test(priority = 2) - public void testThrottleGroupNotInUpdateThrottleDataMap() { - - obThrottleService.updateThrottleDataMap(OBThrottleServiceTestData.THROTTLE_GROUP_BASIC_AUTH, - OBThrottleServiceTestData.THROTTLE_SECOND_PARAM, - OBThrottleServiceTestData.UNLOCK_TIMESTAMP_GREATER_THAN_CURRENT_TIMESTAMP); - - Assert.assertTrue(obThrottleService.throttleDataMap - .containsKey(OBThrottleServiceTestData.THROTTLE_GROUP_BASIC_AUTH)); - } - - @Test - public void testUpdateThrottleData() throws Exception { - - Mockito.doReturn(true).when(mockedOBThrottlerDAO).isThrottleDataExists(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(OBThrottleServiceTestData.getSampleTestThrottleData()).when(mockedOBThrottlerDAO) - .getThrottleData(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(OBThrottleServiceTestData.getSampleUpdateTestThrottleData()).when(mockedOBThrottlerDAO) - .updateThrottleData(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString(), Mockito.anyObject(), - Mockito.anyObject(), Mockito.anyInt()); - Mockito.doReturn(OBThrottleServiceTestData.getSampleUpdateTestThrottleData().getOccurrences()) - .when(throttleDataModel).getOccurrences(); - - obThrottleService.updateThrottleData(OBThrottleServiceTestData.THROTTLE_GROUP, - OBThrottleServiceTestData.THROTTLE_SECOND_PARAM, 3, 180); - - } - - @Test - public void testStoreThrottleData() throws Exception { - - Mockito.doReturn(false).when(mockedOBThrottlerDAO).isThrottleDataExists(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(OBThrottleServiceTestData.getSampleUpdateTestThrottleData()).when(mockedOBThrottlerDAO) - .storeThrottleData(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString(), Mockito.anyObject(), - Mockito.anyObject()); - Mockito.doReturn(OBThrottleServiceTestData.getSampleUpdateTestThrottleData().getOccurrences()) - .when(throttleDataModel).getOccurrences(); - - obThrottleService.updateThrottleData(OBThrottleServiceTestData.THROTTLE_GROUP, - OBThrottleServiceTestData.THROTTLE_SECOND_PARAM, 3, 180); - - } - - @Test(expectedExceptions = OBThrottlerException.class) - public void testStoreThrottleDataError() throws Exception { - - Mockito.doReturn(false).when(mockedOBThrottlerDAO).isThrottleDataExists(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString()); - Mockito.doThrow(OBThrottlerDataInsertionException.class).when(mockedOBThrottlerDAO) - .storeThrottleData(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString(), Mockito.anyObject(), - Mockito.anyObject()); - Mockito.doReturn(OBThrottleServiceTestData.getSampleUpdateTestThrottleData().getOccurrences()) - .when(throttleDataModel).getOccurrences(); - - obThrottleService.updateThrottleData(OBThrottleServiceTestData.THROTTLE_GROUP, - OBThrottleServiceTestData.THROTTLE_SECOND_PARAM, 3, 180); - - } - - @Test(expectedExceptions = OBThrottlerException.class) - public void testUpdateThrottleDataError() throws Exception { - - Mockito.doReturn(true).when(mockedOBThrottlerDAO).isThrottleDataExists(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(OBThrottleServiceTestData.getSampleTestThrottleData()).when(mockedOBThrottlerDAO) - .getThrottleData(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString()); - Mockito.doThrow(OBThrottlerDataUpdationException.class).when(mockedOBThrottlerDAO) - .updateThrottleData(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString(), Mockito.anyObject(), - Mockito.anyObject(), Mockito.anyInt()); - Mockito.doReturn(OBThrottleServiceTestData.getSampleUpdateTestThrottleData().getOccurrences()) - .when(throttleDataModel).getOccurrences(); - - obThrottleService.updateThrottleData(OBThrottleServiceTestData.THROTTLE_GROUP, - OBThrottleServiceTestData.THROTTLE_SECOND_PARAM, 3, 180); - - } - - @Test(expectedExceptions = OBThrottlerException.class) - public void testRetrievalThrottleDataError() throws Exception { - - Mockito.doReturn(true).when(mockedOBThrottlerDAO).isThrottleDataExists(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString()); - Mockito.doThrow(OBThrottlerDataRetrievalException.class).when(mockedOBThrottlerDAO) - .getThrottleData(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString()); - Mockito.doReturn(OBThrottleServiceTestData.getSampleUpdateTestThrottleData()).when(mockedOBThrottlerDAO) - .updateThrottleData(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString(), Mockito.anyObject(), - Mockito.anyObject(), Mockito.anyInt()); - Mockito.doReturn(OBThrottleServiceTestData.getSampleUpdateTestThrottleData().getOccurrences()) - .when(throttleDataModel).getOccurrences(); - - obThrottleService.updateThrottleData(OBThrottleServiceTestData.THROTTLE_GROUP, - OBThrottleServiceTestData.THROTTLE_SECOND_PARAM, 3, 180); - - } - - @Test - public void testDeleteRecordOnSuccessAttempt() throws Exception { - - Mockito.doReturn(true).when(mockedOBThrottlerDAO).isThrottleDataExists(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString()); - - obThrottleService.deleteRecordOnSuccessAttempt(OBThrottleServiceTestData.THROTTLE_GROUP, - OBThrottleServiceTestData.THROTTLE_SECOND_PARAM); - - } - - @Test(expectedExceptions = OBThrottlerException.class) - public void testRetrievalThrottleDataErrorWhenDeleteRecordOnSuccessAttempt() throws Exception { - - Mockito.doThrow(OBThrottlerDataRetrievalException.class).when(mockedOBThrottlerDAO) - .isThrottleDataExists(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString()); - - obThrottleService.deleteRecordOnSuccessAttempt(OBThrottleServiceTestData.THROTTLE_GROUP, - OBThrottleServiceTestData.THROTTLE_SECOND_PARAM); - - } - - @Test(expectedExceptions = OBThrottlerException.class) - public void testDeleteThrottleDataErrorWhenDeleteRecordOnSuccessAttempt() throws Exception { - - Mockito.doReturn(true).when(mockedOBThrottlerDAO).isThrottleDataExists(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString()); - Mockito.doThrow(OBThrottlerDataDeletionException.class).when(mockedOBThrottlerDAO) - .deleteThrottleData(Mockito.anyObject(), - Mockito.anyString(), Mockito.anyString()); - - obThrottleService.deleteRecordOnSuccessAttempt(OBThrottleServiceTestData.THROTTLE_GROUP, - OBThrottleServiceTestData.THROTTLE_SECOND_PARAM); - - } - - private void mockStaticClasses() throws OBThrottlerException { - - PowerMockito.mockStatic(DatabaseUtil.class); - PowerMockito.when(DatabaseUtil.getDBConnection()).thenReturn(Mockito.mock(Connection.class)); - - PowerMockito.mockStatic(DataStoreInitializer.class); - PowerMockito.when(DataStoreInitializer.initializeOBThrottlerDAO()).thenReturn(mockedOBThrottlerDAO); - } -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/test/java/com/wso2/openbanking/accelerator/throttler/service/util/OBThrottleServiceTestData.java b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/test/java/com/wso2/openbanking/accelerator/throttler/service/util/OBThrottleServiceTestData.java deleted file mode 100644 index 93e2eaa9..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/test/java/com/wso2/openbanking/accelerator/throttler/service/util/OBThrottleServiceTestData.java +++ /dev/null @@ -1,63 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.throttler.service.util; - -import com.wso2.openbanking.accelerator.throttler.dao.model.ThrottleDataModel; - -import java.sql.Timestamp; -import java.util.Date; - -/** - * Test data for Open Banking throttle service. - */ -public class OBThrottleServiceTestData { - - public static final Timestamp CURRENT_TIMESTAMP = new Timestamp(new Date().getTime()); - - public static final Timestamp UNLOCK_TIMESTAMP_GREATER_THAN_CURRENT_TIMESTAMP = - new Timestamp(CURRENT_TIMESTAMP.getTime() + (1000L * 180)); - - public static final Timestamp UNLOCK_TIMESTAMP_LESS_THAN_CURRENT_TIMESTAMP = - new Timestamp(CURRENT_TIMESTAMP.getTime() - (1000L * 180)); - - public static final String THROTTLE_GROUP = "OBIdentifierAuthenticator"; - - public static final String THROTTLE_SECOND_GROUP = "OBIdentifierAuthenticator-1"; - - public static final String THROTTLE_GROUP_BASIC_AUTH = "BasicAuth"; - - public static final String THROTTLE_PARAM = "user-ip-192.168.1.1"; - - public static final String THROTTLE_SECOND_PARAM = "user-ip-192.168.1.1"; - - public static ThrottleDataModel getSampleTestThrottleData() { - - ThrottleDataModel throttleDataModel = new ThrottleDataModel(THROTTLE_GROUP, THROTTLE_PARAM, CURRENT_TIMESTAMP, - UNLOCK_TIMESTAMP_GREATER_THAN_CURRENT_TIMESTAMP, 1); - return throttleDataModel; - } - - public static ThrottleDataModel getSampleUpdateTestThrottleData() { - - ThrottleDataModel throttleDataModel = new ThrottleDataModel(THROTTLE_GROUP, THROTTLE_PARAM, CURRENT_TIMESTAMP, - UNLOCK_TIMESTAMP_GREATER_THAN_CURRENT_TIMESTAMP, 5); - - return throttleDataModel; - } - -} diff --git a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/test/resources/testng.xml b/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/test/resources/testng.xml deleted file mode 100644 index d8692438..00000000 --- a/open-banking-accelerator/components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service/src/test/resources/testng.xml +++ /dev/null @@ -1,27 +0,0 @@ - - - - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/.openapi-generator-ignore b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/.openapi-generator-ignore deleted file mode 100755 index 4886bc8b..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/.openapi-generator-ignore +++ /dev/null @@ -1,25 +0,0 @@ -# OpenAPI Generator Ignore -# Generated by openapi-generator https://github.com/openapitools/openapi-generator - -# Use this file to prevent files from being overwritten by the generator. -# The patterns follow closely to .gitignore or .dockerignore. - -# As an example, the C# client generator defines ApiClient.cs. -# You can make changes and tell OpenAPI Generator to ignore just this file by uncommenting the following line: -#ApiClient.cs - -# You can match any string of characters against a directory, file or extension with a single asterisk (*): -#foo/*/qux -# The above matches foo/bar/qux and foo/baz/qux, but not foo/bar/baz/qux - -# You can recursively match patterns against a directory, file or extension with a double asterisk (**): -#foo/**/qux -# This matches foo/bar/qux, foo/baz/qux, and foo/bar/baz/qux - -# You can also negate patterns with an exclamation (!). -# For example, you can ignore all files in a docs folder with the file extension .md: -#docs/*.md -# Then explicitly reverse the ignore rule for a single file: -#!docs/README.md - -**/impl/* \ No newline at end of file diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/findbugs-exclude.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/findbugs-exclude.xml deleted file mode 100755 index 665c19b4..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/findbugs-exclude.xml +++ /dev/null @@ -1,24 +0,0 @@ - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/pom.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/pom.xml deleted file mode 100755 index 760f2e03..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/pom.xml +++ /dev/null @@ -1,166 +0,0 @@ - - - - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../../pom.xml - - 4.0.0 - - openbanking-application-info-endpoint - war - WSO2 Open Banking - Application Info Endpoint - - - - io.swagger - swagger-jaxrs - provided - - - org.springframework - spring-web - provided - - - org.apache.cxf - cxf-bundle-jaxrs - provided - - - com.fasterxml.jackson.core - jackson-databind - provided - - - org.wso2.carbon.identity.framework - org.wso2.carbon.identity.core - provided - - - org.wso2.carbon.identity.framework - org.wso2.carbon.identity.application.authentication.framework - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.identity - provided - - - org.wso2.carbon.identity.framework - org.wso2.carbon.identity.application.mgt - provided - - - org.apache.commons - commons-lang3 - provided - - - org.testng - testng - test - - - - - - - org.openapitools - openapi-generator-maven-plugin - ${openapi.generator.plugin.version} - - - - generate - - - - true - ${project.basedir}/src/main/resources/application-info-130.yaml - jaxrs-cxf - - src/gen/java/ - true - - false - com.wso2.open.banking.application.info.endpoint.model - com.wso2.open.banking.application.info.endpoint.api - impl - DTO - ${project.basedir} - - - - - - maven-war-plugin - ${maven-war-plugin.version} - - - - - src/main/webapp - - - api#openbanking#application - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - false - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-exclude.xml - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/api/ApplicationInformationApi.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/api/ApplicationInformationApi.java deleted file mode 100755 index b1418b99..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/api/ApplicationInformationApi.java +++ /dev/null @@ -1,93 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.open.banking.application.info.endpoint.api; - -import com.wso2.open.banking.application.info.endpoint.model.ApplicationBulkMetadataSuccessDTO; -import com.wso2.open.banking.application.info.endpoint.model.ApplicationInfoErrorDTO; -import com.wso2.open.banking.application.info.endpoint.model.ApplicationSingleMetadataSuccessDTO; -import io.swagger.annotations.Api; -import io.swagger.annotations.ApiOperation; -import io.swagger.annotations.ApiResponse; -import io.swagger.annotations.ApiResponses; - -import javax.validation.constraints.NotNull; -import javax.ws.rs.GET; -import javax.ws.rs.Path; -import javax.ws.rs.PathParam; -import javax.ws.rs.Produces; -import javax.ws.rs.QueryParam; -import javax.ws.rs.core.Response; -import java.util.List; - -/** - * ApplicationInfoAPI - * - *

This specifies a RESTful API for retriving OAuth Application Information - * - */ -@Path("/") -@Api(value = "/", description = "") -public interface ApplicationInformationApi { - - /** - * Retrieve Bulk Application Metadata - * - */ - @GET - @Path("/metadata/") - @Produces({ "application/json" }) - @ApiOperation(value = "Retrieve Bulk Application Metadata", tags={ "Application Information", }) - @ApiResponses(value = { - @ApiResponse(code = 200, message = "OK", response = ApplicationBulkMetadataSuccessDTO.class), - @ApiResponse(code = 404, message = "Service Provider Data Not Found"), - @ApiResponse(code = 400, message = "Bad Request", response = ApplicationInfoErrorDTO.class), - @ApiResponse(code = 500, message = "Internal Server Error", response = ApplicationInfoErrorDTO.class) }) - public Response getBulkApplicationMetadata(@QueryParam("clientIds") @NotNull List clientIds); - - /** - * Retrieve All Application Metadata - * - */ - @GET - @Path("/all/metadata") - @Produces({ "application/json" }) - @ApiOperation(value = "Retrieve Bulk Application Metadata", tags={ "Application Information", }) - @ApiResponses(value = { - @ApiResponse(code = 200, message = "OK", response = ApplicationBulkMetadataSuccessDTO.class), - @ApiResponse(code = 404, message = "Service Provider Data Not Found"), - @ApiResponse(code = 400, message = "Bad Request", response = ApplicationInfoErrorDTO.class), - @ApiResponse(code = 500, message = "Internal Server Error", response = ApplicationInfoErrorDTO.class) }) - public Response getAllApplicationMetadata(); - - /** - * Retrieve Single Application Metadata - * - */ - @GET - @Path("/metadata/{id}") - @Produces({ "application/json" }) - @ApiOperation(value = "Retrieve Single Application Metadata", tags={ "Application Information" }) - @ApiResponses(value = { - @ApiResponse(code = 200, message = "OK", response = ApplicationSingleMetadataSuccessDTO.class), - @ApiResponse(code = 404, message = "Service Provider Data Not Found"), - @ApiResponse(code = 400, message = "Bad Request", response = ApplicationInfoErrorDTO.class), - @ApiResponse(code = 500, message = "Internal Server Error", response = ApplicationInfoErrorDTO.class) }) - public Response getSingleApplicationMetadata(@PathParam("id") String id); -} - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/model/ApplicationBulkMetadataSuccessDTO.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/model/ApplicationBulkMetadataSuccessDTO.java deleted file mode 100755 index 4d08f344..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/model/ApplicationBulkMetadataSuccessDTO.java +++ /dev/null @@ -1,88 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.open.banking.application.info.endpoint.model; - -import com.fasterxml.jackson.annotation.JsonProperty; -import io.swagger.annotations.ApiModel; -import io.swagger.annotations.ApiModelProperty; - -import javax.validation.Valid; -import javax.validation.constraints.NotNull; -import java.util.HashMap; -import java.util.Map; - -/** - * defines metadata for requested applications - **/ -@ApiModel(description="defines metadata for requested applications") -public class ApplicationBulkMetadataSuccessDTO { - - @ApiModelProperty(required = true, value = "Key value pairs of clientId and attributes") - @Valid - /** - * Key value pairs of clientId and attributes - **/ - private Map data = new HashMap(); - /** - * Key value pairs of clientId and attributes - * @return data - **/ - @JsonProperty("data") - @NotNull - public Map getData() { - return data; - } - - public void setData(Map data) { - this.data = data; - } - - public ApplicationBulkMetadataSuccessDTO data(Map data) { - this.data = data; - return this; - } - - public ApplicationBulkMetadataSuccessDTO putDataItem(String key, ApplicationMetadataResourceDTO dataItem) { - this.data.put(key, dataItem); - return this; - } - - - @Override - public String toString() { - StringBuilder sb = new StringBuilder(); - sb.append("class ApplicationBulkMetadataSuccessDTO {\n"); - - sb.append(" data: ").append(toIndentedString(data)).append("\n"); - sb.append("}"); - return sb.toString(); - } - - /** - * Convert the given object to string with each line indented by 4 spaces - * (except the first line). - */ - private static String toIndentedString(java.lang.Object o) { - if (o == null) { - return "null"; - } - return o.toString().replace("\n", "\n "); - } -} - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/model/ApplicationInfoErrorDTO.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/model/ApplicationInfoErrorDTO.java deleted file mode 100755 index a51c7f89..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/model/ApplicationInfoErrorDTO.java +++ /dev/null @@ -1,126 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.open.banking.application.info.endpoint.model; - -import com.fasterxml.jackson.annotation.JsonProperty; -import io.swagger.annotations.ApiModel; -import io.swagger.annotations.ApiModelProperty; - -/** - * defines error object for application infromation api - **/ -@ApiModel(description="defines error object for application infromation api") -public class ApplicationInfoErrorDTO { - - @ApiModelProperty(value = "HTTP error code as string") - /** - * HTTP error code as string - **/ - private String status; - - @ApiModelProperty(value = "error summmary") - /** - * error summmary - **/ - private String title; - - @ApiModelProperty(value = "human readable error description") - /** - * human readable error description - **/ - private String description; - /** - * HTTP error code as string - * @return status - **/ - @JsonProperty("status") - public String getStatus() { - return status; - } - - public void setStatus(String status) { - this.status = status; - } - - public ApplicationInfoErrorDTO status(String status) { - this.status = status; - return this; - } - - /** - * error summmary - * @return title - **/ - @JsonProperty("title") - public String getTitle() { - return title; - } - - public void setTitle(String title) { - this.title = title; - } - - public ApplicationInfoErrorDTO title(String title) { - this.title = title; - return this; - } - - /** - * human readable error description - * @return description - **/ - @JsonProperty("description") - public String getDescription() { - return description; - } - - public void setDescription(String description) { - this.description = description; - } - - public ApplicationInfoErrorDTO description(String description) { - this.description = description; - return this; - } - - - @Override - public String toString() { - StringBuilder sb = new StringBuilder(); - sb.append("class ApplicationInfoErrorDTO {\n"); - - sb.append(" status: ").append(toIndentedString(status)).append("\n"); - sb.append(" title: ").append(toIndentedString(title)).append("\n"); - sb.append(" description: ").append(toIndentedString(description)).append("\n"); - sb.append("}"); - return sb.toString(); - } - - /** - * Convert the given object to string with each line indented by 4 spaces - * (except the first line). - */ - private static String toIndentedString(java.lang.Object o) { - if (o == null) { - return "null"; - } - return o.toString().replace("\n", "\n "); - } -} - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/model/ApplicationMetadataResourceDTO.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/model/ApplicationMetadataResourceDTO.java deleted file mode 100755 index a4d13617..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/model/ApplicationMetadataResourceDTO.java +++ /dev/null @@ -1,133 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.open.banking.application.info.endpoint.model; - -import com.fasterxml.jackson.annotation.JsonProperty; -import io.swagger.annotations.ApiModel; -import io.swagger.annotations.ApiModelProperty; - -import java.util.Map; - -/** - * defines resource object for application - **/ -@ApiModel(description="defines resource object for application") -public class ApplicationMetadataResourceDTO { - - @ApiModelProperty(value = "type of object") - /** - * type of object - **/ - private String type; - - @ApiModelProperty(value = "OAuth Client id of the application") - /** - * OAuth Client id of the application - **/ - private String id; - - @ApiModelProperty(value = "Key-Value pairs of application metadata") - /** - * Key-Value pairs of application metadata - **/ - private Map metadata = null; - /** - * type of object - * @return type - **/ - @JsonProperty("type") - public String getType() { - return type; - } - - public void setType(String type) { - this.type = type; - } - - public ApplicationMetadataResourceDTO type(String type) { - this.type = type; - return this; - } - - /** - * OAuth Client id of the application - * @return id - **/ - @JsonProperty("Id") - public String getId() { - return id; - } - - public void setId(String id) { - this.id = id; - } - - public ApplicationMetadataResourceDTO id(String id) { - this.id = id; - return this; - } - - /** - * Key-Value pairs of application metadata - * @return metadata - **/ - @JsonProperty("metadata") - public Map getMetadata() { - return metadata; - } - - public void setMetadata(Map metadata) { - this.metadata = metadata; - } - - public ApplicationMetadataResourceDTO metadata(Map metadata) { - this.metadata = metadata; - return this; - } - - public ApplicationMetadataResourceDTO putMetadataItem(String key, String metadataItem) { - this.metadata.put(key, metadataItem); - return this; - } - - - @Override - public String toString() { - StringBuilder sb = new StringBuilder(); - sb.append("class ApplicationMetadataResourceDTO {\n"); - - sb.append(" type: ").append(toIndentedString(type)).append("\n"); - sb.append(" id: ").append(toIndentedString(id)).append("\n"); - sb.append(" metadata: ").append(toIndentedString(metadata)).append("\n"); - sb.append("}"); - return sb.toString(); - } - - /** - * Convert the given object to string with each line indented by 4 spaces - * (except the first line). - */ - private static String toIndentedString(java.lang.Object o) { - if (o == null) { - return "null"; - } - return o.toString().replace("\n", "\n "); - } -} - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/model/ApplicationSingleMetadataSuccessDTO.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/model/ApplicationSingleMetadataSuccessDTO.java deleted file mode 100755 index c10a7c99..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/gen/java/com/wso2/open/banking/application/info/endpoint/model/ApplicationSingleMetadataSuccessDTO.java +++ /dev/null @@ -1,78 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.open.banking.application.info.endpoint.model; - -import com.fasterxml.jackson.annotation.JsonProperty; -import io.swagger.annotations.ApiModel; -import io.swagger.annotations.ApiModelProperty; - -import javax.validation.Valid; -import javax.validation.constraints.NotNull; - -/** - * defines metadata for requested applications - **/ -@ApiModel(description="defines metadata for requested applications") -public class ApplicationSingleMetadataSuccessDTO { - - @ApiModelProperty(required = true, value = "") - @Valid - private ApplicationMetadataResourceDTO data = null; - /** - * Get data - * @return data - **/ - @JsonProperty("data") - @NotNull - public ApplicationMetadataResourceDTO getData() { - return data; - } - - public void setData(ApplicationMetadataResourceDTO data) { - this.data = data; - } - - public ApplicationSingleMetadataSuccessDTO data(ApplicationMetadataResourceDTO data) { - this.data = data; - return this; - } - - - @Override - public String toString() { - StringBuilder sb = new StringBuilder(); - sb.append("class ApplicationSingleMetadataSuccessDTO {\n"); - - sb.append(" data: ").append(toIndentedString(data)).append("\n"); - sb.append("}"); - return sb.toString(); - } - - /** - * Convert the given object to string with each line indented by 4 spaces - * (except the first line). - */ - private static String toIndentedString(java.lang.Object o) { - if (o == null) { - return "null"; - } - return o.toString().replace("\n", "\n "); - } -} - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/java/com/wso2/open/banking/application/info/endpoint/api/constants/MetaDataSQLStatements.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/java/com/wso2/open/banking/application/info/endpoint/api/constants/MetaDataSQLStatements.java deleted file mode 100644 index e22e37a6..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/java/com/wso2/open/banking/application/info/endpoint/api/constants/MetaDataSQLStatements.java +++ /dev/null @@ -1,38 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.open.banking.application.info.endpoint.api.constants; - -/** - * MetaDataSQLStatements. - * - *

This specifies SQL Statements for retrieving all consent id's for consent manager application - * - */ -public class MetaDataSQLStatements { - - /** - * SQL query to retrieve list of clientIds. - * @return - */ - public String getAllClientIds() { - - return "SELECT DISTINCT CLIENT_ID FROM OB_CONSENT"; - } - -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/java/com/wso2/open/banking/application/info/endpoint/api/data/MetaDataDAOImpl.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/java/com/wso2/open/banking/application/info/endpoint/api/data/MetaDataDAOImpl.java deleted file mode 100644 index c5d2c1d0..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/java/com/wso2/open/banking/application/info/endpoint/api/data/MetaDataDAOImpl.java +++ /dev/null @@ -1,84 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.open.banking.application.info.endpoint.api.data; - -import com.google.common.collect.ImmutableMap; -import com.wso2.open.banking.application.info.endpoint.api.constants.MetaDataSQLStatements; -import com.wso2.openbanking.accelerator.common.persistence.JDBCPersistenceManager; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.List; -import java.util.Map; - -import javax.ws.rs.InternalServerErrorException; -import javax.ws.rs.core.Response; - -/** - * MetaDataDAOImpl. - * - *

This specifies a DAO Impl for retrieving all client id's for the consent manager application - */ -public class MetaDataDAOImpl { - - private static final Log log = LogFactory.getLog(MetaDataDAOImpl.class); - - /** - * Returns the List of distinct clientIds from the consent table. - * - * @return - */ - public List getAllDistinctClientIds() { - - MetaDataSQLStatements sqlStatements = new MetaDataSQLStatements(); - final String initialConsentRequest = sqlStatements.getAllClientIds(); - List clientIdList = new ArrayList<>(); - - try (Connection connection = JDBCPersistenceManager.getInstance().getDBConnection(); - PreparedStatement preparedStatement = connection.prepareStatement(initialConsentRequest)) { - - try (ResultSet rs = preparedStatement.executeQuery()) { - while (rs.next()) { - clientIdList.add(rs.getString(1)); - } - } - if (log.isDebugEnabled()) { - log.debug(String.format("ClientIds %s provided for bulk retrieval", - Arrays.toString(clientIdList.toArray()))); - } - - return clientIdList; - } catch (SQLException e) { - log.error("Error occurred while retrieving ClientIds.", e); - Map error = ImmutableMap.of( - "error", "Error occurred while retrieving ClientIds"); - - throw new InternalServerErrorException(Response.status - (Response.Status.INTERNAL_SERVER_ERROR).entity(error).build()); - } - - } - -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/java/com/wso2/open/banking/application/info/endpoint/api/impl/ApplicationInformationApiServiceImpl.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/java/com/wso2/open/banking/application/info/endpoint/api/impl/ApplicationInformationApiServiceImpl.java deleted file mode 100644 index 1cdb0806..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/java/com/wso2/open/banking/application/info/endpoint/api/impl/ApplicationInformationApiServiceImpl.java +++ /dev/null @@ -1,254 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.open.banking.application.info.endpoint.api.impl; - -import com.wso2.open.banking.application.info.endpoint.api.ApplicationInformationApi; -import com.wso2.open.banking.application.info.endpoint.api.data.MetaDataDAOImpl; -import com.wso2.open.banking.application.info.endpoint.api.utils.MappingUtil; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.context.PrivilegedCarbonContext; -import org.wso2.carbon.identity.application.common.IdentityApplicationManagementException; -import org.wso2.carbon.identity.application.common.model.InboundAuthenticationConfig; -import org.wso2.carbon.identity.application.common.model.InboundAuthenticationRequestConfig; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.application.common.model.ServiceProviderProperty; -import org.wso2.carbon.identity.application.common.util.IdentityApplicationConstants; -import org.wso2.carbon.identity.application.mgt.ApplicationManagementService; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.List; -import java.util.Optional; -import java.util.stream.Collectors; - -import javax.ws.rs.WebApplicationException; -import javax.ws.rs.core.Response; - -/** - * ApplicationInfoAPI. - * - *

This specifies a RESTful API for retriving OAuth Application Information - */ -public class ApplicationInformationApiServiceImpl implements ApplicationInformationApi { - - private static final String QUERY_PARAM_BULK_DELIMITER = ","; - private static final String ERROR_INVALID_REQUEST = "invalid clientIds given in request"; - private static final String ERROR_FETCHING_SP = "Unable to retrieve information," + - " please contact system administrator"; - private static final String APPLICATION_NOT_EXISTS = "Unavailable Application"; - - private static final Log log = LogFactory.getLog(ApplicationInformationApiServiceImpl.class); - - /** - * Retrieve Bulk Application Metadata. - * - * @param clientIds client ID sequences for retrieval. - * @return client response. - */ - public Response getBulkApplicationMetadata(List clientIds) { - - // Take List of query params for clientIds and split by delimiter and remove duplicates - List splitClientIds = clientIds.stream() - .map(str -> str.split(QUERY_PARAM_BULK_DELIMITER)) - .flatMap(Arrays::stream) - .distinct() - .filter(StringUtils::isNotEmpty) - .collect(Collectors.toList()); - - // Check if any IDs are set if not send error message - if (splitClientIds.isEmpty()) { - - return Response.status(Response.Status.BAD_REQUEST) - .entity(MappingUtil.buildErrorDTO( - String.valueOf(Response.Status.BAD_REQUEST.getStatusCode()), - Response.Status.BAD_REQUEST.getReasonPhrase(), ERROR_INVALID_REQUEST)) - .build(); - } - - if (log.isDebugEnabled()) { - log.debug(String.format("ClientIds %s provided for bulk retrieval", - Arrays.toString(splitClientIds.toArray()))); - } - - // Retrieve Service Providers from list - List serviceProviderList = splitClientIds - .stream() - .map(this::getOAuthServiceProvider) - .collect(Collectors.toList()); - - return Response.ok() - .entity(MappingUtil.mapBulkMetadataResponseDTO(serviceProviderList)) - .build(); - } - - /** - * Retrieve All Bulk Application Metadata. - * - * @return client response. - */ - public Response getAllApplicationMetadata() { - - MetaDataDAOImpl metaDataDAOImpl = new MetaDataDAOImpl(); - - List clientIdList = metaDataDAOImpl.getAllDistinctClientIds(); - - return getBulkApplicationMetadata(clientIdList); - } - - - /** - * Retrieve Single Application Metadata. - * - * @param id clientId of application. - * @return client response. - */ - public Response getSingleApplicationMetadata(String id) { - - ServiceProvider selectedServiceProvider = getOAuthServiceProvider(id); - - // If Service provider is present map value and return - return Response.ok() - .entity(MappingUtil.mapSingleMetadataResponseDTO(selectedServiceProvider)) - .build(); - } - - /** - * Get Service provider from clientId. - * - * @param clientId of application. - * @return Service Provider. - * @throws WebApplicationException client error. - */ - private ServiceProvider getOAuthServiceProvider(String clientId) throws WebApplicationException { - - ApplicationManagementService managementService = this.getApplicationManagementService(); - Optional serviceProvider; - try { - serviceProvider = Optional.ofNullable(managementService.getServiceProviderByClientId(clientId, - IdentityApplicationConstants.OAuth2.NAME, getTenantDomain())); - } catch (IdentityApplicationManagementException e) { - - log.error(String.format("Unable to retrieve service provider information for clientId %s", clientId), e); - - // Throw Web Application exception - throw new WebApplicationException(Response.status(Response.Status.INTERNAL_SERVER_ERROR) - .entity(MappingUtil.buildErrorDTO( - String.valueOf(Response.Status.INTERNAL_SERVER_ERROR.getStatusCode()), - Response.Status.INTERNAL_SERVER_ERROR.getReasonPhrase(), - ERROR_FETCHING_SP)) - .build()); - } - - // Handle empty or default service provider. - if (!serviceProvider.isPresent() || - serviceProvider.get().getApplicationName().equals(IdentityApplicationConstants.DEFAULT_SP_CONFIG)) { - - final String errorMessage = String.format("Unable to find application for clientId %s", clientId); - - if (log.isDebugEnabled()) { - log.debug(errorMessage); - } - - return handleSPForDefaultOrNull(clientId, serviceProvider); - } - - return serviceProvider.get(); - } - - /** - * Populate serviceProvider with APPLICATION_NOT_EXISTS based on default or empty. - * @param clientId - * @param serviceProvider - * @return - */ - private ServiceProvider handleSPForDefaultOrNull(String clientId, Optional serviceProvider) { - - ServiceProvider serviceProviderForErrorScenarios; - - // set new SP and inboundAuthenticationConfig for cases where serviceProvider is not present - if (!serviceProvider.isPresent()) { - serviceProviderForErrorScenarios = new ServiceProvider(); - - InboundAuthenticationConfig inboundAuthenticationConfig = new InboundAuthenticationConfig(); - InboundAuthenticationRequestConfig[] configs = new InboundAuthenticationRequestConfig[1]; - configs[0] = new InboundAuthenticationRequestConfig(); - configs[0].setInboundAuthKey(clientId); - configs[0].setInboundAuthType(IdentityApplicationConstants.OAuth2.NAME); - inboundAuthenticationConfig.setInboundAuthenticationRequestConfigs(configs); - serviceProviderForErrorScenarios.setInboundAuthenticationConfig(inboundAuthenticationConfig); - serviceProvider = Optional.of(serviceProviderForErrorScenarios); - } - - // continue populating SP with properties - serviceProviderForErrorScenarios = serviceProvider.get(); - List serviceProviderPropertyList = new ArrayList<>(); - serviceProviderForErrorScenarios.setApplicationName(APPLICATION_NOT_EXISTS); - - ServiceProviderProperty displayName = new ServiceProviderProperty(); - displayName.setName("software_id"); - displayName.setValue(APPLICATION_NOT_EXISTS); - - ServiceProviderProperty clientName = new ServiceProviderProperty(); - clientName.setName("client_name"); - clientName.setValue(APPLICATION_NOT_EXISTS); - - serviceProviderPropertyList.add(displayName); - serviceProviderPropertyList.add(clientName); - ServiceProviderProperty[] serviceProviderProperties = - new ServiceProviderProperty[serviceProviderPropertyList.size()]; - - serviceProviderPropertyList.toArray(serviceProviderProperties); - serviceProviderForErrorScenarios.setSpProperties(serviceProviderProperties); - - InboundAuthenticationRequestConfig[] inboundAuthenticationRequestConfigs = serviceProviderForErrorScenarios - .getInboundAuthenticationConfig().getInboundAuthenticationRequestConfigs(); - if (inboundAuthenticationRequestConfigs.length != 0) { - inboundAuthenticationRequestConfigs[0].setInboundAuthKey(clientId); - inboundAuthenticationRequestConfigs[0].setInboundAuthType(IdentityApplicationConstants.OAuth2.NAME); - } - return serviceProviderForErrorScenarios; - } - - /** - * Get WSO2 IS Application Mgt Service from threadlocal carbon context. - * - * @return Application Management Service Implementation. - */ - private ApplicationManagementService getApplicationManagementService() { - - return (ApplicationManagementService) PrivilegedCarbonContext - .getThreadLocalCarbonContext() - .getOSGiService(ApplicationManagementService.class, null); - - } - - /** - * Get Tenant Domain String from carbon context. - * - * @return tenant domain of current context. - */ - private String getTenantDomain() { - - return PrivilegedCarbonContext.getThreadLocalCarbonContext().getTenantDomain(true); - } - -} - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/java/com/wso2/open/banking/application/info/endpoint/api/utils/MappingUtil.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/java/com/wso2/open/banking/application/info/endpoint/api/utils/MappingUtil.java deleted file mode 100755 index b7eb2247..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/java/com/wso2/open/banking/application/info/endpoint/api/utils/MappingUtil.java +++ /dev/null @@ -1,150 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.open.banking.application.info.endpoint.api.utils; - -import com.wso2.open.banking.application.info.endpoint.model.ApplicationBulkMetadataSuccessDTO; -import com.wso2.open.banking.application.info.endpoint.model.ApplicationInfoErrorDTO; -import com.wso2.open.banking.application.info.endpoint.model.ApplicationMetadataResourceDTO; -import com.wso2.open.banking.application.info.endpoint.model.ApplicationSingleMetadataSuccessDTO; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.util.OpenBankingUtils; -import com.wso2.openbanking.accelerator.identity.sp.metadata.extension.SPMetadataFilter; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.application.common.model.InboundAuthenticationRequestConfig; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.application.common.model.ServiceProviderProperty; -import org.wso2.carbon.identity.application.common.util.IdentityApplicationConstants; - -import java.util.Arrays; -import java.util.List; -import java.util.Map; -import java.util.Optional; -import java.util.stream.Collectors; - -/** - * Map internal models to external API DTOs. - */ -public class MappingUtil { - - private static final Log log = LogFactory.getLog(MappingUtil.class); - private static final SPMetadataFilter metadataFilter = (SPMetadataFilter) OpenBankingUtils.getClassInstanceFromFQN( - OpenBankingConfigParser.getInstance().getSPMetadataFilterExtension()); - private static final String SOFTWARE_ID = "software_id"; - - /** - * Map Single Metadata API Response DTO from service provider. - * - * @param serviceProvider service provider to populate response with. - * @return - */ - public static ApplicationSingleMetadataSuccessDTO mapSingleMetadataResponseDTO(ServiceProvider serviceProvider) { - - // Create Target DTO - ApplicationSingleMetadataSuccessDTO successDTO = new ApplicationSingleMetadataSuccessDTO(); - - // Map single metadata resource to DTO - successDTO.setData(mapApplicationMetadataResourceDTO(serviceProvider)); - - return successDTO; - } - - /** - * Map bulk Metadata API Response DTO from list of service providers. - * - * @param serviceProviderList service provider list to populate response with. - * @return - */ - public static ApplicationBulkMetadataSuccessDTO mapBulkMetadataResponseDTO(List - serviceProviderList) { - - ApplicationBulkMetadataSuccessDTO successDTO = new ApplicationBulkMetadataSuccessDTO(); - - // Stream list of service providers and map to resource DTOs - successDTO.setData(serviceProviderList.stream() - .map(MappingUtil::mapApplicationMetadataResourceDTO) - .collect(Collectors.toMap(ApplicationMetadataResourceDTO::getId, obj -> obj))); - - return successDTO; - } - - /** - * Map Single Application Metadata Resource from service provider. - * - * @param serviceProvider service provider to populate response with. - * @return - */ - public static ApplicationMetadataResourceDTO mapApplicationMetadataResourceDTO(ServiceProvider serviceProvider) { - - ApplicationMetadataResourceDTO resourceDTO = new ApplicationMetadataResourceDTO(); - - // Filter out OAuth2 from array of InboundAuthenticationRequestConfig and get clientId - Optional clientId = Arrays.stream(serviceProvider.getInboundAuthenticationConfig() - .getInboundAuthenticationRequestConfigs()) - .filter(conf -> conf.getInboundAuthType().equals(IdentityApplicationConstants.OAuth2.NAME)) - .findFirst().map(InboundAuthenticationRequestConfig::getInboundAuthKey); - - // Set type of resource - resourceDTO.setType(IdentityApplicationConstants.OAuth2.NAME); - - // If clientId is present set to target DTO - clientId.ifPresent(resourceDTO::setId); - - // Stream through ServiceProvider property array and map to target attributes - Map metadata = Arrays.stream(serviceProvider.getSpProperties()) - .collect(Collectors.toMap(ServiceProviderProperty::getName, ServiceProviderProperty::getValue)); - - // Return default application name for software_id - if (StringUtils.isEmpty(metadata.get(SOFTWARE_ID))) { - metadata.put(SOFTWARE_ID, serviceProvider.getApplicationName()); - } - - // filter metadata map using the configured metadata filter logic. (default: DefaultSPMetadataFilter) - metadata = metadataFilter.filter(metadata); - - if (log.isDebugEnabled()) { - log.debug(String.format("Application metadata list for client_id %s : %s", - clientId.orElse(""), metadata)); - } - resourceDTO.setMetadata(metadata); - - return resourceDTO; - - } - - /** - * Build Client Error Reponse. - * - * @param status status code of the error. - * @param title summary of the error. - * @param description human readable descriptive error. - * @return Reponse object. - */ - public static ApplicationInfoErrorDTO buildErrorDTO(String status, String title, String description) { - - ApplicationInfoErrorDTO errorDTO = new ApplicationInfoErrorDTO(); - - errorDTO.setStatus(status); - errorDTO.setTitle(title); - errorDTO.setDescription(description); - return errorDTO; - } - -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/resources/application-info-300.yaml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/resources/application-info-300.yaml deleted file mode 100755 index 1c72700a..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/resources/application-info-300.yaml +++ /dev/null @@ -1,147 +0,0 @@ -openapi: 3.0.0 -info: - version: v3.0.0 - title: ApplicationInfoAPI - description: This specifies a RESTful API for retrieving OAuth Application Information - contact: - name: WSO2 - url: http://wso2.com/solutions/financial/open-banking/ - email: openbankingdemo@wso2.com - license: - name: WSO2 Commercial License - url: https://wso2.com -servers: - - url: https://{ob_km_host}:{ob_km_port}/api/openbanking/application/ - variables: - ob_km_host: - default: localhost - description: Host of the Open Banking Key Manager - ob_km_port: - default: "9446" - description: Port of the Open Banking Key Manager -paths: - /metadata/: - get: - summary: Retrieve Bulk Application Metadata - operationId: getBulkApplicationMetadata - tags: - - Application Information - parameters: - - in: query - name: clientIds - required: true - schema: - type: array - items: - type: string - style: form - explode: false - responses: - "200": - description: OK - content: - application/json: - schema: - $ref: "#/components/schemas/ApplicationBulkMetadataSuccess" - "404": - description: Service Provider Data Not Found - "400": - description: Bad Request - content: - application/json: - schema: - $ref: "#/components/schemas/ApplicationInfoError" - "500": - description: Internal Server Error - content: - application/json: - schema: - $ref: "#/components/schemas/ApplicationInfoError" - /metadata/{id}: - get: - summary: Retrieve Single Application Metadata - operationId: getSingleApplicationMetadata - tags: - - Application Information - parameters: - - name: id - in: path - description: The client id of the application - required: true - schema: - type: string - responses: - "200": - description: OK - content: - application/json: - schema: - $ref: "#/components/schemas/ApplicationSingleMetadataSuccess" - "404": - description: Service Provider Data Not Found - "400": - description: Bad Request - content: - application/json: - schema: - $ref: "#/components/schemas/ApplicationInfoError" - "500": - description: Internal Server Error - content: - application/json: - schema: - $ref: "#/components/schemas/ApplicationInfoError" -components: - schemas: - ApplicationBulkMetadataSuccess: - title: Application Bulk Metadata Success Response - description: Defines metadata for the requested applications - type: object - properties: - data: - type: object - description: Key value pairs of client ids and attributes - additionalProperties: - $ref: "#/components/schemas/ApplicationMetadataResource" - - required: - - data - ApplicationSingleMetadataSuccess: - title: Application Single Metadata Success Response - description: Defines metadata for a requested application - type: object - properties: - data: - $ref: "#/components/schemas/ApplicationMetadataResource" - required: - - data - ApplicationMetadataResource: - title: Application Metadata Resource - description: Defines a resource object for an application - type: object - properties: - type: - type: string - description: Object type - Id: - type: string - description: The OAuth client id of the application - metadata: - type: object - description: Key-Value pairs of application metadata - additionalProperties: - type: string - ApplicationInfoError: - title: Error Response - description: Defines an error object for the Application Information API - type: object - properties: - status: - type: string - description: The HTTP error code as a string - title: - type: string - description: Error summary - description: - type: string - description: Human readable error description diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/resources/findbugs-exclude.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/resources/findbugs-exclude.xml deleted file mode 100644 index 6804ed0a..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/resources/findbugs-exclude.xml +++ /dev/null @@ -1,30 +0,0 @@ - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/resources/findbugs-include.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/resources/findbugs-include.xml deleted file mode 100644 index 8932a22e..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/webapp/META-INF/webapp-classloading.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/webapp/META-INF/webapp-classloading.xml deleted file mode 100755 index dfc87b0b..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/webapp/META-INF/webapp-classloading.xml +++ /dev/null @@ -1,34 +0,0 @@ - - - - - - - - false - - - Carbon,CXF3 - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/webapp/WEB-INF/beans.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/webapp/WEB-INF/beans.xml deleted file mode 100755 index 3aed2904..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/webapp/WEB-INF/beans.xml +++ /dev/null @@ -1,34 +0,0 @@ - - - - - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/webapp/WEB-INF/web.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/webapp/WEB-INF/web.xml deleted file mode 100755 index 16c696c8..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint/src/main/webapp/WEB-INF/web.xml +++ /dev/null @@ -1,70 +0,0 @@ - - - - - - WSO2 Open Banking - Application Info - WSO2 Open Banking - Application Info API - - - contextConfigLocation - WEB-INF/beans.xml - - - - HttpHeaderSecurityFilter - org.apache.catalina.filters.HttpHeaderSecurityFilter - - hstsEnabled - false - - - - - HttpHeaderSecurityFilter - * - - - - - org.springframework.web.context.ContextLoaderListener - - - - - CXFServlet - - org.apache.cxf.transport.servlet.CXFServlet - - 1 - - - - CXFServlet - /* - - - - 60 - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/pom.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/pom.xml deleted file mode 100644 index f90da6fb..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/pom.xml +++ /dev/null @@ -1,124 +0,0 @@ - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../../pom.xml - - 4.0.0 - - com.wso2.openbanking.accelerator.ciba.authentication.endpoint - WSO2 Open Banking - CIBA Authentication Endpoint - WSO2 Open Banking - CIBA Authentication Endpoint - war - - - - org.testng - testng - test - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.consent.extensions - provided - - - com.fasterxml.jackson.core - jackson-databind - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - provided - - - org.wso2.carbon.identity.inbound.auth.oauth2 - org.wso2.carbon.identity.oauth.ciba - provided - - - org.wso2.carbon.identity.framework - org.wso2.carbon.identity.application.authentication.framework - provided - - - org.wso2.carbon.identity.outbound.auth.push - org.wso2.carbon.identity.application.authenticator.push.common - provided - - - org.wso2.carbon.identity.outbound.auth.push - org.wso2.carbon.identity.application.authenticator.push.device.handler - provided - - - org.wso2.carbon.identity.outbound.auth.push - org.wso2.carbon.identity.application.authenticator.push - provided - - - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - false - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - maven-war-plugin - ${maven-war-plugin.version} - - - - - src/main/webapp - - - api#openbanking#ciba - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/java/com/wso2/openbanking/accelerator/ciba/authentication/endpoint/impl/api/CIBAAuthenticationEndpoint.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/java/com/wso2/openbanking/accelerator/ciba/authentication/endpoint/impl/api/CIBAAuthenticationEndpoint.java deleted file mode 100644 index f158c269..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/java/com/wso2/openbanking/accelerator/ciba/authentication/endpoint/impl/api/CIBAAuthenticationEndpoint.java +++ /dev/null @@ -1,685 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.ciba.authentication.endpoint.impl.api; - -import com.nimbusds.jwt.JWTClaimsSet; -import com.nimbusds.jwt.JWTParser; -import com.wso2.openbanking.accelerator.ciba.authentication.endpoint.impl.exception.CIBAAuthenticationEndpointException; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.CarbonUtils; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.builder.ConsentStepsBuilder; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentData; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentPersistData; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentPersistStep; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentRetrievalStep; -import com.wso2.openbanking.accelerator.consent.extensions.ciba.model.CIBAAuthenticationEndpointErrorResponse; -import com.wso2.openbanking.accelerator.consent.extensions.ciba.model.CIBAAuthenticationEndpointInterface; -import com.wso2.openbanking.accelerator.consent.extensions.common.AuthErrorCode; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentCache; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionExporter; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionUtils; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.identity.util.HTTPClientUtils; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import org.apache.commons.io.IOUtils; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.HttpResponse; -import org.apache.http.HttpStatus; -import org.apache.http.client.methods.HttpGet; -import org.apache.http.client.methods.HttpUriRequest; -import org.apache.http.impl.client.CloseableHttpClient; -import org.apache.http.util.EntityUtils; -import org.wso2.carbon.identity.application.authentication.framework.context.AuthenticationContext; -import org.wso2.carbon.identity.application.authentication.framework.exception.AuthenticationFailedException; -import org.wso2.carbon.identity.application.authentication.framework.model.AuthenticatedUser; -import org.wso2.carbon.identity.application.authenticator.push.common.PushAuthContextManager; -import org.wso2.carbon.identity.application.authenticator.push.common.PushJWTValidator; -import org.wso2.carbon.identity.application.authenticator.push.common.exception.PushAuthTokenValidationException; -import org.wso2.carbon.identity.application.authenticator.push.common.impl.PushAuthContextManagerImpl; -import org.wso2.carbon.identity.application.authenticator.push.device.handler.DeviceHandler; -import org.wso2.carbon.identity.application.authenticator.push.device.handler.exception.PushDeviceHandlerClientException; -import org.wso2.carbon.identity.application.authenticator.push.device.handler.exception.PushDeviceHandlerServerException; -import org.wso2.carbon.identity.application.authenticator.push.device.handler.impl.DeviceHandlerImpl; -import org.wso2.carbon.identity.application.authenticator.push.dto.AuthDataDTO; -import org.wso2.carbon.identity.oauth.cache.SessionDataCacheEntry; -import org.wso2.carbon.identity.oauth.ciba.common.AuthReqStatus; -import org.wso2.carbon.identity.oauth.ciba.dao.CibaDAOFactory; -import org.wso2.carbon.identity.oauth.ciba.exceptions.CibaCoreException; -import org.wso2.carbon.identity.oauth2.model.OAuth2Parameters; - -import java.io.IOException; -import java.lang.reflect.InvocationTargetException; -import java.net.HttpURLConnection; -import java.text.ParseException; -import java.util.List; -import java.util.Map; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; -import javax.ws.rs.Consumes; -import javax.ws.rs.GET; -import javax.ws.rs.POST; -import javax.ws.rs.Path; -import javax.ws.rs.Produces; -import javax.ws.rs.core.Context; -import javax.ws.rs.core.HttpHeaders; -import javax.ws.rs.core.Response; -import javax.ws.rs.core.UriInfo; - -import static com.wso2.openbanking.accelerator.consent.extensions.ciba.authenticator.CIBAPushAuthenticator.createErrorResponse; - -/** - * Implementation class for the CIBA authentication endpoint API. - */ -@Path("/") -public class CIBAAuthenticationEndpoint { - - private static final Log log = LogFactory.getLog(CIBAAuthenticationEndpoint.class); - private static CIBAAuthenticationEndpointInterface cibaAuthenticationEndpointInterfaceTK; - private static List consentPersistSteps = null; - private static List consentRetrievalSteps = null; - - public CIBAAuthenticationEndpoint() { - - initializeConsentSteps(); - } - - @SuppressFBWarnings("JAXRS_ENDPOINT") - // Suppressed content - Endpoint - // Suppression reason - False Positive : This endpoint is secured with access control lists in the configuration - // Suppressed warning count - 1 - @POST - @Path("/push-auth/authenticate") - @Consumes({"application/json; charset=utf-8"}) - @Produces({"application/json; charset=utf-8"}) - public Response handleCIBAAuthenticationRequest(@Context HttpServletRequest request, - @Context HttpServletResponse response, @Context UriInfo uriInfo) { - - try { - log.info("CIBA authentication call received"); - handleMobileResponse(request, response); - } catch (CIBAAuthenticationEndpointException e) { - // create error response - CIBAAuthenticationEndpointErrorResponse errorResponse = createErrorResponse(e.getHttpStatusCode(), - e.getErrorCode(), e.getErrorDescription()); - return Response.status(errorResponse.getHttpStatusCode() != 0 ? - errorResponse.getHttpStatusCode() : e.getHttpStatusCode()) - .entity(errorResponse.getPayload()).build(); - } - - return Response.status(HttpStatus.SC_ACCEPTED).build(); - } - - @SuppressFBWarnings("JAXRS_ENDPOINT") - // Suppressed content - Endpoint - // Suppression reason - False Positive : This endpoint is secured with access control lists in the configuration - // Suppressed warning count - 1 - @GET - @Path("/push-auth/discovery-data") - @Produces({"application/json; charset=utf-8"}) - public Response handleDiscoveryRequest(@Context HttpServletRequest request, - @Context HttpServletResponse response, - @Context HttpHeaders headers) { - - try { - log.info("CIBA discovery call received"); - JSONObject deviceRegistrationData = handleDiscovery(request, response, headers); - return Response.status(HttpStatus.SC_ACCEPTED) - .entity(deviceRegistrationData).build(); - - } catch (CIBAAuthenticationEndpointException e) { - // create error response - CIBAAuthenticationEndpointErrorResponse errorResponse = createErrorResponse(e.getHttpStatusCode(), - e.getErrorCode(), e.getErrorDescription()); - return Response.status(errorResponse.getHttpStatusCode() != 0 ? - errorResponse.getHttpStatusCode() : e.getHttpStatusCode()) - .entity(errorResponse.getPayload()).build(); - } - } - - @SuppressFBWarnings("HTTP_PARAMETER_POLLUTION") - // Suppressed content - CIBAAuthenticationEndpointConstants.DEVICE_REGISTRATION_URL - // Suppression reason - False Positive : This is a hard coded, trusted path. It is not a user input - // Suppressed warning count - 1 - private JSONObject handleDiscovery(HttpServletRequest request, HttpServletResponse response, HttpHeaders headers) - throws CIBAAuthenticationEndpointException { - - List authHeaders = headers.getRequestHeader(HttpHeaders.AUTHORIZATION); - String userToken = null; - // Make the API call with user's access token - if (authHeaders.size() != 0) { - userToken = authHeaders.get(0); - } - String registrationUrl = CarbonUtils.getCarbonServerUrl() + - CIBAAuthenticationEndpointConstants.DEVICE_REGISTRATION_URL; - HttpUriRequest deviceRegistrationRequest = new HttpGet(registrationUrl); - deviceRegistrationRequest.setHeader(CIBAAuthenticationEndpointConstants.AUTH_HEADER_NAME, userToken); - JSONObject deviceRegistrationData = sendRequest(deviceRegistrationRequest); - // Change authentication endpoint to OB CIBA webapp as it handles the CIBA authenticate call - deviceRegistrationData.put(CIBAAuthenticationEndpointConstants.AUTHENTICATION_ENDPOINT, - CIBAAuthenticationEndpointConstants.AUTHENTICATION_ENDPOINT_URL_PREFIX - + deviceRegistrationData.getAsString( - CIBAAuthenticationEndpointConstants.AUTHENTICATION_ENDPOINT)); - return deviceRegistrationData; - } - - public JSONObject sendRequest(HttpUriRequest request) - throws CIBAAuthenticationEndpointException { - - String responseStr = null; - try { - CloseableHttpClient client = HTTPClientUtils.getHttpsClient(); - HttpResponse response = client.execute(request); - responseStr = EntityUtils.toString(response.getEntity()); - - if ((response.getStatusLine().getStatusCode() / 100) != 2) { - if (response.getStatusLine().getStatusCode() == HttpURLConnection.HTTP_UNAUTHORIZED) { - log.debug("Received unauthorized(401) response. body: " + responseStr); - throw new CIBAAuthenticationEndpointException(HttpStatus.SC_UNAUTHORIZED, - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_UNAUTHORIZED.getMessage(), - "Received unauthorized Response: " + responseStr); - } - } else { - // received success (200 range) response - Object responseJSON; - try { - responseJSON = new JSONParser(JSONParser.MODE_PERMISSIVE).parse(responseStr); - if (!(responseJSON instanceof JSONObject)) { - log.error("Discovery call response is not a JSON object"); - throw new CIBAAuthenticationEndpointException(HttpStatus.SC_BAD_REQUEST, - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_BAD_REQUEST.getMessage(), - "Discovery call response is not a JSON object"); - } - } catch (net.minidev.json.parser.ParseException e) { - throw new CIBAAuthenticationEndpointException(HttpStatus.SC_INTERNAL_SERVER_ERROR, - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_SERVER_ERROR.getMessage(), - "Unable to parse the response", e); - } - - JSONObject responseData = (JSONObject) responseJSON; - return responseData; - } - - } catch (IOException e) { - log.error("Exception occurred while reading request. Caused by, ", e); - } catch (OpenBankingException e) { - log.error("Exception occurred while generating http client. Caused by, ", e); - } - throw new CIBAAuthenticationEndpointException(HttpStatus.SC_INTERNAL_SERVER_ERROR, - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_SERVER_ERROR.getMessage(), - "Unexpected response received for the request. path: " + - request.getURI() + " response:" + responseStr); - } - - /** - * Initialize consent builder. - */ - private static synchronized void initializeConsentSteps() { - - if (consentRetrievalSteps == null || consentPersistSteps == null) { - ConsentStepsBuilder consentStepsBuilder = ConsentExtensionExporter.getConsentStepsBuilder(); - - if (consentStepsBuilder != null) { - consentRetrievalSteps = consentStepsBuilder.getConsentRetrievalSteps(); - consentPersistSteps = consentStepsBuilder.getConsentPersistSteps(); - } - - if (consentRetrievalSteps != null && !consentRetrievalSteps.isEmpty()) { - log.info("Consent retrieval steps are not null or empty"); - } else { - log.warn("Consent retrieval steps are null or empty"); - } - if (consentPersistSteps != null && !consentPersistSteps.isEmpty()) { - log.info("Consent persist steps are not null or empty"); - } else { - log.warn("Consent persist steps are null or empty"); - } - } else { - log.debug("Retrieval and persist steps are available"); - } - } - - /** - * Persist user consent data. - * - * @param request HTTP request - * @param response HTTP response - * @param sessionDataKey Session Data Key - * @param payload Json payload - * @throws ConsentException - */ - private static void persistConsent(HttpServletRequest request, HttpServletResponse response, - String sessionDataKey, JSONObject payload) throws ConsentException { - - ConsentData consentData = ConsentCache.getConsentDataFromCache(sessionDataKey); - if (consentData == null) { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Unable to get consent data"); - } - - if (payload == null) { - throw new ConsentException(consentData.getRedirectURI(), AuthErrorCode.SERVER_ERROR, - "Payload unavailable", consentData.getState()); - } - - boolean approval; - if (payload.containsKey(CIBAAuthenticationEndpointConstants.APPROVAL)) { - try { - if (payload.get(CIBAAuthenticationEndpointConstants.APPROVAL) instanceof Boolean) { - approval = (Boolean) payload.get(CIBAAuthenticationEndpointConstants.APPROVAL); - } else { - approval = Boolean.parseBoolean((String) payload.get(CIBAAuthenticationEndpointConstants.APPROVAL)); - } - } catch (ClassCastException e) { - log.error("Error while processing consent persistence authorize", e); - throw new ConsentException(ResponseStatus.BAD_REQUEST, - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_PERSIST_INVALID_AUTHORIZE.getMessage()); - } - } else { - throw new ConsentException(consentData.getRedirectURI(), AuthErrorCode.SERVER_ERROR, - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_PERSIST_APPROVAL_MANDATORY.getMessage(), - consentData.getState()); - } - - Map headers = ConsentExtensionUtils.getHeaders(request); - ConsentPersistData consentPersistData = new ConsentPersistData(payload, headers, approval, consentData); - - executePersistence(consentPersistData); - - if (!approval) { - throw new ConsentException(consentData.getRedirectURI(), AuthErrorCode.ACCESS_DENIED, - "User denied the consent", consentData.getState()); - } - - } - - /** - * Execute consent persistence. - * - * @param consentPersistData Consent Persistence data - * @throws ConsentException - */ - private static void executePersistence(ConsentPersistData consentPersistData) throws ConsentException { - - for (ConsentPersistStep step : consentPersistSteps) { - if (log.isDebugEnabled()) { - log.debug("Executing persistence step " + step.getClass().toString()); - } - step.execute(consentPersistData); - } - } - - /** - * Handles authentication request received from mobile app. - * - * @param request HTTP request - * @param response HTTP response - * @throws CIBAAuthenticationEndpointException - */ - public static void handleMobileResponse(HttpServletRequest request, HttpServletResponse response) - throws CIBAAuthenticationEndpointException { - - setCIBAExtension(); - - String responseJsonString; - try { - responseJsonString = IOUtils.toString(request.getInputStream()); - } catch (IOException e) { - throw new CIBAAuthenticationEndpointException(HttpStatus.SC_BAD_REQUEST, - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_BAD_REQUEST.getMessage(), - "Error in reading the request", e); - } - - if (log.isDebugEnabled()) { - log.debug("CIBA authenticate call from mobile received: " + responseJsonString); - } - - Object responseDataJSON; - try { - responseDataJSON = new JSONParser(JSONParser.MODE_PERMISSIVE).parse(responseJsonString); - if (!(responseDataJSON instanceof JSONObject)) { - log.error("response is not a JSON object"); - throw new CIBAAuthenticationEndpointException(HttpStatus.SC_BAD_REQUEST, - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_BAD_REQUEST.getMessage(), - "response is not a JSON object"); - } - } catch (net.minidev.json.parser.ParseException e) { - throw new CIBAAuthenticationEndpointException(HttpStatus.SC_INTERNAL_SERVER_ERROR, - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_SERVER_ERROR.getMessage(), - "Unable to parse the response", e); - } - - JSONObject responseData = (JSONObject) responseDataJSON; - String token = responseData.getAsString(CIBAAuthenticationEndpointConstants.AUTH_RESPONSE); - - if (StringUtils.isEmpty(token)) { - if (log.isDebugEnabled()) { - log.debug(CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_AUTH_RESPONSE_TOKEN_NOT_FOUND); - } - throw new CIBAAuthenticationEndpointException(HttpStatus.SC_BAD_REQUEST, - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_AUTH_RESPONSE_TOKEN_NOT_FOUND - .getCode(), - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_AUTH_RESPONSE_TOKEN_NOT_FOUND - .getMessage()); - } else { - String deviceId = getDeviceIdFromToken(token); - String sessionDataKey = getSessionDataKeyFromToken(token, deviceId); - - if (StringUtils.isEmpty(sessionDataKey)) { - String errorMessage = CIBAAuthenticationEndpointConstants.ErrorMessages - .ERROR_CODE_SESSION_DATA_KEY_NOT_FOUND + deviceId; - if (log.isDebugEnabled()) { - log.debug(errorMessage); - } - - throw new CIBAAuthenticationEndpointException(HttpStatus.SC_BAD_REQUEST, - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_SESSION_DATA_KEY_NOT_FOUND - .getCode(), - errorMessage); - } else { - addToContext(sessionDataKey, token); - - try { - processAuthenticationRequest(request, response, sessionDataKey); - } catch (AuthenticationFailedException e) { - throw new CIBAAuthenticationEndpointException(HttpStatus.SC_BAD_REQUEST, - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_BAD_REQUEST.getMessage(), - "Authentication Failed", e); - } - - response.setStatus(HttpServletResponse.SC_ACCEPTED); - - log.info("Completed processing authentication request from mobile app for session data key " - + sessionDataKey); - - } - } - } - - /** - * Retrieve the config for CIBA consent persistence toolkit extension class for. - */ - private static void setCIBAExtension() { - - try { - cibaAuthenticationEndpointInterfaceTK = (CIBAAuthenticationEndpointInterface) - Class.forName(OpenBankingConfigParser.getInstance() - .getCibaServletExtension()).getDeclaredConstructor().newInstance(); - } catch (InstantiationException | IllegalAccessException | - InvocationTargetException | NoSuchMethodException | ClassNotFoundException e) { - log.error("CIBA Webapp extension not found", e); - } - } - - /** - * Process authentication request received from mobile app. - * - * @param sessionDataKey Session Data Key - * @throws CIBAAuthenticationEndpointException - */ - protected static void processAuthenticationRequest(HttpServletRequest request, - HttpServletResponse response, String sessionDataKey) throws - AuthenticationFailedException, CIBAAuthenticationEndpointException { - - SessionDataCacheEntry cacheEntry = ConsentCache.getCacheEntryFromSessionDataKey(sessionDataKey); - - AuthenticatedUser user = cacheEntry.getLoggedInUser(); - - PushAuthContextManager contextManager = new PushAuthContextManagerImpl(); - AuthenticationContext sessionContext = contextManager.getContext(sessionDataKey); - AuthDataDTO authDataDTO = (AuthDataDTO) sessionContext - .getProperty(CIBAAuthenticationEndpointConstants.CONTEXT_AUTH_DATA); - - String authResponseToken = authDataDTO.getAuthToken(); - String serverChallenge = authDataDTO.getChallenge(); - - String deviceId = getDeviceIdFromToken(authResponseToken); - String publicKey = getPublicKey(deviceId); - - PushJWTValidator validator = new PushJWTValidator(); - JWTClaimsSet claimsSet; - try { - claimsSet = validator.getValidatedClaimSet(authResponseToken, publicKey); - } catch (PushAuthTokenValidationException e) { - String errorMessage = String - .format("Error occurred when trying to validate the JWT signature from device: %s of user: %s.", - deviceId, user.toFullQualifiedUsername()); - throw new AuthenticationFailedException(errorMessage, e); - } - if (claimsSet != null) { - if (validator.validateChallenge(claimsSet, serverChallenge, deviceId)) { - String authStatus; - String metadataJsonString; - JSONArray accountIds; - try { - authStatus = - validator.getClaimFromClaimSet(claimsSet, - CIBAAuthenticationEndpointConstants.TOKEN_RESPONSE, deviceId); - metadataJsonString = (validator.getClaimFromClaimSet(claimsSet, - CIBAAuthenticationEndpointConstants.METADATA, deviceId)); - - Object metadataJSON = new JSONParser(JSONParser.MODE_PERMISSIVE).parse(metadataJsonString); - if (!(metadataJSON instanceof JSONObject)) { - log.error("metadata is not a JSON object"); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, - "metadata is not a JSON object"); - } - JSONObject metadata = (JSONObject) metadataJSON; - - accountIds = - (JSONArray) metadata.get(CIBAAuthenticationEndpointConstants.METADATA_ACCOUNT_IDS); - } catch (PushAuthTokenValidationException | net.minidev.json.parser.ParseException e) { - String errorMessage = "Error in getting claims from the auth response token received from device: " - + deviceId; - throw new AuthenticationFailedException(errorMessage, e); - } - - boolean approval; - if (authStatus.equals(CIBAAuthenticationEndpointConstants.AUTH_REQUEST_STATUS_SUCCESS)) { - approval = true; - } else if (authStatus.equals(CIBAAuthenticationEndpointConstants.AUTH_REQUEST_STATUS_DENIED)) { - approval = false; - } else { - log.error("Invalid authorization status :" + authStatus); - String errorMessage = "Authentication failed! Incorrect auth status " + authStatus + " for user " + - user.toFullQualifiedUsername(); - throw new AuthenticationFailedException(errorMessage); - } - - JSONObject payload = new JSONObject(); - payload.put(CIBAAuthenticationEndpointConstants.APPROVAL, approval); - // Authorize call is skipped in consent persist call in CIBA - payload.put(CIBAAuthenticationEndpointConstants.AUTHORIZE, false); - payload.put(CIBAAuthenticationEndpointConstants.ACCOUNT_IDS, accountIds); - - // add TK data - if (cibaAuthenticationEndpointInterfaceTK != null) { - payload = cibaAuthenticationEndpointInterfaceTK - .updateConsentData(payload); - } - - persistConsent(request, response, sessionDataKey, payload); - persistAuthorization(sessionDataKey, authStatus); - } else { - String errorMessage = String - .format("Authentication failed! JWT challenge validation for device: %s of user: %s.", - deviceId, user); - throw new AuthenticationFailedException(errorMessage); - } - - } else { - String errorMessage = String - .format("Authentication failed! JWT signature is not valid for device: %s of user: %s.", - deviceId, user); - throw new AuthenticationFailedException(errorMessage); - } - - try { - contextManager.clearContext(validator.getClaimFromClaimSet(claimsSet, - CIBAAuthenticationEndpointConstants.TOKEN_SESSION_DATA_KEY, deviceId)); - } catch (PushAuthTokenValidationException e) { - String errorMessage = "Error in getting claim " + - CIBAAuthenticationEndpointConstants.TOKEN_SESSION_DATA_KEY + " from the auth response token " + - "received from device: " + deviceId; - throw new AuthenticationFailedException(errorMessage, e); - } - } - - /** - * Persist authorization response. - * - * @param sessionDataKey Session Data Key - * @param authStatus User action for the authorization request - * @throws CIBAAuthenticationEndpointException - */ - public static void persistAuthorization(String sessionDataKey, String authStatus) - throws CIBAAuthenticationEndpointException { - - SessionDataCacheEntry cacheEntry = ConsentCache.getCacheEntryFromSessionDataKey(sessionDataKey); - - if (cacheEntry != null) { - AuthenticatedUser user = cacheEntry.getLoggedInUser(); - OAuth2Parameters oAuth2Parameters = cacheEntry.getoAuth2Parameters(); - String nonce = oAuth2Parameters.getNonce(); - - try { - if (CIBAAuthenticationEndpointConstants.AUTH_REQUEST_STATUS_SUCCESS.equals(authStatus)) { - String authCodeKey = CibaDAOFactory.getInstance().getCibaAuthMgtDAO().getCibaAuthCodeKey(nonce); - - // Update successful authentication. - CibaDAOFactory.getInstance().getCibaAuthMgtDAO() - .persistAuthenticationSuccess(authCodeKey, user); - } else if (CIBAAuthenticationEndpointConstants.AUTH_REQUEST_STATUS_DENIED.equals(authStatus)) { - String authCodeKey = CibaDAOFactory.getInstance().getCibaAuthMgtDAO().getCibaAuthCodeKey(nonce); - CibaDAOFactory.getInstance().getCibaAuthMgtDAO().updateStatus(authCodeKey, AuthReqStatus.FAILED); - } else { - String errorMessage = "Invalid authorization status: " + authStatus; - throw new CIBAAuthenticationEndpointException(HttpStatus.SC_BAD_REQUEST, - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_BAD_REQUEST.getMessage(), - errorMessage); - } - } catch (CibaCoreException e) { - String errorMessage = "Error while persisting CIBA auth status for session data key " + sessionDataKey; - throw new CIBAAuthenticationEndpointException(HttpStatus.SC_INTERNAL_SERVER_ERROR, - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_SERVER_ERROR.getMessage(), - errorMessage, e); - } - } - } - - /** - * Derive the Device ID from the auth response token header. - * - * @param token Auth response token - * @return Device ID - * @throws CIBAAuthenticationEndpointException if the token string fails to parse to JWT - */ - protected static String getDeviceIdFromToken(String token) throws CIBAAuthenticationEndpointException { - - try { - return String.valueOf(JWTParser.parse(token).getHeader().getCustomParam( - CIBAAuthenticationEndpointConstants.TOKEN_DEVICE_ID)); - } catch (ParseException e) { - throw new CIBAAuthenticationEndpointException(HttpStatus.SC_INTERNAL_SERVER_ERROR, - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_GET_DEVICE_ID_FAILED.getCode(), - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_GET_DEVICE_ID_FAILED.getMessage(), - e); - } - } - - /** - * Derive the SessionDataKey from the auth response token. - * - * @param token Auth response token - * @param deviceId Unique ID of the device trying to authenticate - * @return SessionDataKey - * @throws CIBAAuthenticationEndpointException if the auth response token fails to parse to JWT or the public key - * for the device is not retrieved or if the token is not valid - */ - private static String getSessionDataKeyFromToken(String token, String deviceId) throws - CIBAAuthenticationEndpointException { - - DeviceHandler deviceHandler = new DeviceHandlerImpl(); - PushJWTValidator validator = new PushJWTValidator(); - - try { - String publicKey = deviceHandler.getPublicKey(deviceId); - JWTClaimsSet claimsSet = validator.getValidatedClaimSet(token, publicKey); - return claimsSet.getStringClaim(CIBAAuthenticationEndpointConstants.TOKEN_SESSION_DATA_KEY); - } catch (PushDeviceHandlerServerException | PushDeviceHandlerClientException e) { - String errorMessage = CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_GET_PUBLIC_KEY_FAILED - .toString() + deviceId; - throw new CIBAAuthenticationEndpointException(HttpStatus.SC_INTERNAL_SERVER_ERROR, - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_GET_PUBLIC_KEY_FAILED.getCode(), - errorMessage, e); - } catch (PushAuthTokenValidationException e) { - String errorMessage = CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_TOKEN_VALIDATION_FAILED - .toString() + deviceId; - throw new CIBAAuthenticationEndpointException(HttpStatus.SC_INTERNAL_SERVER_ERROR, - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_TOKEN_VALIDATION_FAILED.getCode(), - errorMessage, e); - } catch (ParseException e) { - throw new CIBAAuthenticationEndpointException(HttpStatus.SC_INTERNAL_SERVER_ERROR, - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_PARSE_JWT_FAILED.getCode(), - CIBAAuthenticationEndpointConstants.ErrorMessages.ERROR_CODE_PARSE_JWT_FAILED.getMessage(), e); - } - } - - /** - * Add the received auth response token to the authentication context. - * - * @param sessionDataKey Unique key to identify the session - * @param token Auth response token - */ - private static void addToContext(String sessionDataKey, String token) { - - PushAuthContextManager contextManager = new PushAuthContextManagerImpl(); - AuthenticationContext context = contextManager.getContext(sessionDataKey); - - AuthDataDTO authDataDTO = (AuthDataDTO) context - .getProperty(CIBAAuthenticationEndpointConstants.CONTEXT_AUTH_DATA); - authDataDTO.setAuthToken(token); - context.setProperty(CIBAAuthenticationEndpointConstants.CONTEXT_AUTH_DATA, authDataDTO); - contextManager.storeContext(sessionDataKey, context); - } - - /** - * Get the public key for the device by the device ID. - * - * @param deviceId Unique ID for the device - * @return Public key string - * @throws AuthenticationFailedException if an error occurs while getting the public key - */ - protected static String getPublicKey(String deviceId) throws AuthenticationFailedException { - - DeviceHandler deviceHandler = new DeviceHandlerImpl(); - try { - return deviceHandler.getPublicKey(deviceId); - } catch (PushDeviceHandlerServerException | PushDeviceHandlerClientException e) { - throw new AuthenticationFailedException("Error occurred when trying to get the public key for device: " - + deviceId + "."); - } - } - -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/java/com/wso2/openbanking/accelerator/ciba/authentication/endpoint/impl/api/CIBAAuthenticationEndpointConstants.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/java/com/wso2/openbanking/accelerator/ciba/authentication/endpoint/impl/api/CIBAAuthenticationEndpointConstants.java deleted file mode 100644 index ee363306..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/java/com/wso2/openbanking/accelerator/ciba/authentication/endpoint/impl/api/CIBAAuthenticationEndpointConstants.java +++ /dev/null @@ -1,118 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.ciba.authentication.endpoint.impl.api; - -/** - * Constants for CIBA authentication endpoint. - */ -public class CIBAAuthenticationEndpointConstants { - - // request related constants - public static final String AUTH_RESPONSE = "authResponse"; - public static final String TOKEN_DEVICE_ID = "did"; - public static final String TOKEN_SESSION_DATA_KEY = "sid"; - public static final String CONTEXT_AUTH_DATA = "authData"; - public static final String TOKEN_RESPONSE = "res"; - public static final String METADATA = "mta"; - public static final String AUTH_REQUEST_STATUS_SUCCESS = "SUCCESSFUL"; - public static final String AUTH_REQUEST_STATUS_DENIED = "DENIED"; - - // device registration related constants - public static final String DEVICE_REGISTRATION_URL = "/api/users/v1/me/push-auth/discovery-data"; - public static final String AUTHENTICATION_ENDPOINT_URL_PREFIX = "/api/openbanking/ciba"; - public static final String AUTHENTICATION_ENDPOINT = "ae"; - public static final String AUTH_HEADER_NAME = "Authorization"; - - // consent related constants - public static final String APPROVAL = "approval"; - public static final String AUTHORIZE = "authorize"; - public static final String ACCOUNT_IDS = "accountIds"; - public static final String METADATA_ACCOUNT_IDS = "approvedAccountIds"; - - /** - * Enum which contains error codes and corresponding error messages. - */ - public enum ErrorMessages { - - ERROR_CODE_AUTH_RESPONSE_TOKEN_NOT_FOUND( - "PBA-15001", - "The request did not contain an authentication response token" - ), - ERROR_CODE_SESSION_DATA_KEY_NOT_FOUND( - "PBA-15002", - "Session data key is not present in the authentication response token received from device: " - ), - ERROR_CODE_GET_DEVICE_ID_FAILED( - "PBA-15003", - "Error occurred when extracting the auth response token." - ), - ERROR_CODE_GET_PUBLIC_KEY_FAILED( - "PBA-15004", - "Error occurred when trying to get the public key from device: " - ), - ERROR_CODE_TOKEN_VALIDATION_FAILED( - "PBA-15005", - "Error occurred when validating auth response token from device: " - ), - ERROR_CODE_PARSE_JWT_FAILED( - "PBA-15006", - "Error occurred when parsing auth response token to JWT." - ), - ERROR_PERSIST_INVALID_AUTHORIZE( - "400", "Invalid value for authorize. Should be true/false" - ), - ERROR_PERSIST_APPROVAL_MANDATORY( - "400", "Mandatory body parameter approval is unavailable" - ), - ERROR_CODE_SERVER_ERROR( - "500", "internal server error" - ), - ERROR_CODE_BAD_REQUEST( - "400", "Bad Request" - ), - ERROR_CODE_UNAUTHORIZED( - "401", "Unauthorized" - ); - - private final String code; - private final String message; - - ErrorMessages(String code, String message) { - - this.code = code; - this.message = message; - } - - public String getCode() { - - return code; - } - - public String getMessage() { - - return message; - } - - @Override - public String toString() { - - return code + " - " + message; - } - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/java/com/wso2/openbanking/accelerator/ciba/authentication/endpoint/impl/exception/CIBAAuthenticationEndpointException.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/java/com/wso2/openbanking/accelerator/ciba/authentication/endpoint/impl/exception/CIBAAuthenticationEndpointException.java deleted file mode 100644 index ad88ac80..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/java/com/wso2/openbanking/accelerator/ciba/authentication/endpoint/impl/exception/CIBAAuthenticationEndpointException.java +++ /dev/null @@ -1,79 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.ciba.authentication.endpoint.impl.exception; - -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; - -/** - * Exception for CIBA authentication endpoint. - */ -public class CIBAAuthenticationEndpointException extends OpenBankingException { - - private String errorDescription; - private String errorCode; - private int httpStatusCode; - - public int getHttpStatusCode() { - - return httpStatusCode; - } - - public void setHttpStatusCode(int httpStatusCode) { - - this.httpStatusCode = httpStatusCode; - } - - public String getErrorDescription() { - - return errorDescription; - } - - public void setErrorDescription(String errorDescription) { - - this.errorDescription = errorDescription; - } - - public String getErrorCode() { - - return errorCode; - } - - public void setErrorCode(String errorCode) { - - this.errorCode = errorCode; - } - - public CIBAAuthenticationEndpointException(int httpStatusCode, String errorCode, String errorDescription, - Throwable e) { - - super(errorDescription, e); - this.errorDescription = errorDescription; - this.errorCode = errorCode; - this.httpStatusCode = httpStatusCode; - - } - - public CIBAAuthenticationEndpointException(int httpStatusCode, String errorCode, String errorDescription) { - - super(errorDescription); - this.errorDescription = errorDescription; - this.errorCode = errorCode; - this.httpStatusCode = httpStatusCode; - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/resources/findbugs-include.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/resources/findbugs-include.xml deleted file mode 100644 index 8932a22e..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/webapp/META-INF/MANIFEST.mf b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/webapp/META-INF/MANIFEST.mf deleted file mode 100644 index 9d885be5..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/webapp/META-INF/MANIFEST.mf +++ /dev/null @@ -1 +0,0 @@ -Manifest-Version: 1.0 diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/webapp/META-INF/webapp-classloading.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/webapp/META-INF/webapp-classloading.xml deleted file mode 100644 index b212826c..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/webapp/META-INF/webapp-classloading.xml +++ /dev/null @@ -1,35 +0,0 @@ - - - - - - - - - false - - - Carbon,CXF3 - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/webapp/WEB-INF/beans.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/webapp/WEB-INF/beans.xml deleted file mode 100644 index bd83ba12..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/webapp/WEB-INF/beans.xml +++ /dev/null @@ -1,34 +0,0 @@ - - - - - - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/webapp/WEB-INF/web.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/webapp/WEB-INF/web.xml deleted file mode 100644 index 231ed2d8..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint/src/main/webapp/WEB-INF/web.xml +++ /dev/null @@ -1,69 +0,0 @@ - - - - - - WSO2 Open Banking - CIBA Authentication Endpoint - WSO2 Open Banking - CIBA Authentication Endpoint - - - contextConfigLocation - WEB-INF/beans.xml - - - - HttpHeaderSecurityFilter - org.apache.catalina.filters.HttpHeaderSecurityFilter - - hstsEnabled - false - - - - - HttpHeaderSecurityFilter - * - - - - - org.springframework.web.context.ContextLoaderListener - - - - - CXFServlet - - org.apache.cxf.transport.servlet.CXFServlet - - 1 - - - - CXFServlet - /* - - - - 60 - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/pom.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/pom.xml deleted file mode 100644 index 6c08ef40..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/pom.xml +++ /dev/null @@ -1,181 +0,0 @@ - - - - - 4.0.0 - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../../pom.xml - - - openbanking-consent-endpoint - war - WSO2 Open Banking - Consent Endpoint - - - - io.swagger - swagger-jaxrs - - - javax.ws.rs - jsr311-api - - - com.google.guava - guava - - - org.yaml - snakeyaml - - - - - net.minidev - json-smart - provided - - - org.testng - testng - test - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.identity - - - org.slf4j - slf4j-api - - - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.consent.dao - - - org.slf4j - slf4j-api - - - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.consent.service - - - org.slf4j - slf4j-api - - - provided - - - org.wso2.carbon.identity.local.auth.api - org.wso2.carbon.identity.local.auth.api.core - - - org.slf4j - slf4j-api - - - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.consent.extensions - - - org.slf4j - slf4j-api - - - provided - - - com.fasterxml.jackson.core - jackson-databind - provided - - - - - - - maven-war-plugin - ${maven-war-plugin.version} - - - - - src/main/webapp - - - api#openbanking#consent - WEB-INF/lib/slf4j-api-*.jar - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - false - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-exclude.xml - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - org.apache.maven.plugins - maven-compiler-plugin - - 8 - 8 - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/api/ConsentAdminEndpoint.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/api/ConsentAdminEndpoint.java deleted file mode 100644 index fca2ecd5..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/api/ConsentAdminEndpoint.java +++ /dev/null @@ -1,204 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.endpoint.api; - -import com.wso2.openbanking.accelerator.consent.endpoint.util.ConsentUtils; -import com.wso2.openbanking.accelerator.consent.extensions.admin.builder.ConsentAdminBuilder; -import com.wso2.openbanking.accelerator.consent.extensions.admin.model.ConsentAdminData; -import com.wso2.openbanking.accelerator.consent.extensions.admin.model.ConsentAdminHandler; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionExporter; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionUtils; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; -import javax.ws.rs.Consumes; -import javax.ws.rs.DELETE; -import javax.ws.rs.GET; -import javax.ws.rs.Path; -import javax.ws.rs.Produces; -import javax.ws.rs.core.Context; -import javax.ws.rs.core.Response; -import javax.ws.rs.core.UriInfo; - -/** - * ConsentSearchEndpoint. - *

- * This specifies a RESTful API for consent search to be used at consent user and customer service portals. - */ -@SuppressFBWarnings("JAXRS_ENDPOINT") -// Suppressed content - Endpoints -// Suppression reason - False Positive : These endpoints are secured with access control -// as defined in the IS deployment.toml file -// Suppressed warning count - 7 -@Path("/admin") -public class ConsentAdminEndpoint { - - private static final Log log = LogFactory.getLog(ConsentAdminEndpoint.class); - - private static ConsentAdminHandler consentAdminHandler = null; - - public ConsentAdminEndpoint() { - if (consentAdminHandler == null) { - initializeConsentAdminHandler(); - } - } - - private static void initializeConsentAdminHandler() { - ConsentAdminBuilder consentAdminBuilder = ConsentExtensionExporter.getConsentAdminBuilder(); - - if (consentAdminBuilder != null) { - consentAdminHandler = consentAdminBuilder.getConsentAdminHandler(); - } - - if (consentAdminHandler != null) { - log.info("Consent admin handler " + consentAdminHandler.getClass().getName() + "initialized"); - } else { - log.warn("Consent admin handler is null"); - } - } - - /** - * Search consent data. - */ - @GET - @Path("/search") - @Consumes({"application/x-www-form-urlencoded"}) - @Produces({"application/json; charset=utf-8"}) - public Response search(@Context HttpServletRequest request, @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - - ConsentAdminData consentAdminData = new ConsentAdminData(ConsentExtensionUtils.getHeaders(request), - uriInfo.getQueryParameters(), uriInfo.getAbsolutePath().getPath(), request, response); - consentAdminHandler.handleSearch(consentAdminData); - return sendResponse(consentAdminData); - } - - - /** - * Search consent status audit records. - */ - @GET - @Path("/search/consent-status-audit") - @Consumes({"application/x-www-form-urlencoded"}) - @Produces({"application/json; charset=utf-8"}) - public Response searchConsentStatusAudit(@Context HttpServletRequest request, @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - - ConsentAdminData consentAdminData = new ConsentAdminData(ConsentExtensionUtils.getHeaders(request), - uriInfo.getQueryParameters(), uriInfo.getAbsolutePath().getPath(), request, response); - consentAdminHandler.handleConsentStatusAuditSearch(consentAdminData); - return sendResponse(consentAdminData); - } - - /** - * Search consent file. - */ - @GET - @Path("/search/consent-file") - @Consumes({"application/x-www-form-urlencoded"}) - @Produces({"application/json; charset=utf-8"}) - public Response searchConsentFile(@Context HttpServletRequest request, @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - - ConsentAdminData consentAdminData = new ConsentAdminData(ConsentExtensionUtils.getHeaders(request), - uriInfo.getQueryParameters(), uriInfo.getAbsolutePath().getPath(), request, response); - consentAdminHandler.handleConsentFileSearch(consentAdminData); - return sendResponse(consentAdminData); - } - - /** - * Get consent amendment history. - */ - @GET - @Path("/consent-amendment-history") - @Consumes({"application/json; charset=utf-8"}) - @Produces({"application/json; charset=utf-8"}) - public Response getConsentAmendmentHistoryById(@Context HttpServletRequest request, - @Context HttpServletResponse response, @Context UriInfo uriInfo) { - - ConsentAdminData consentAdminData = new ConsentAdminData(ConsentExtensionUtils.getHeaders(request), - uriInfo.getQueryParameters(), uriInfo.getAbsolutePath().getPath(), request, response); - consentAdminHandler.handleConsentAmendmentHistoryRetrieval(consentAdminData); - return sendResponse(consentAdminData); - } - - /** - * Revoke consent data. - */ - @DELETE - @Path("/revoke") - @Consumes({"application/x-www-form-urlencoded"}) - @Produces({"application/json; charset=utf-8"}) - public Response revoke(@Context HttpServletRequest request, @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - - ConsentAdminData consentAdminData = new ConsentAdminData(ConsentExtensionUtils.getHeaders(request), - ConsentUtils.getJSONObjectPayload(request), uriInfo.getQueryParameters(), - uriInfo.getAbsolutePath().getPath(), request, response); - consentAdminHandler.handleRevoke(consentAdminData); - return sendResponse(consentAdminData); - } - - /** - * Invoke consent expiration task. - */ - @GET - @Path("/expire-consents") - @Consumes({"application/json; charset=utf-8"}) - @Produces({"application/json; charset=utf-8"}) - public Response expireConsents(@Context HttpServletRequest request, - @Context HttpServletResponse response, @Context UriInfo uriInfo) { - - ConsentAdminData consentAdminData = new ConsentAdminData(ConsentExtensionUtils.getHeaders(request), - uriInfo.getQueryParameters(), uriInfo.getAbsolutePath().getPath(), request, response); - consentAdminHandler.handleConsentExpiry(consentAdminData); - return sendResponse(consentAdminData); - } - - /** - * Invoke retention data db sync task. - */ - @GET - @Path("/sync-temporary-retention-data") - @Consumes({"application/json; charset=utf-8"}) - @Produces({"application/json; charset=utf-8"}) - public Response syncTemporaryRetentionData(@Context HttpServletRequest request, - @Context HttpServletResponse response, @Context UriInfo uriInfo) { - - ConsentAdminData consentAdminData = new ConsentAdminData(ConsentExtensionUtils.getHeaders(request), - uriInfo.getQueryParameters(), uriInfo.getAbsolutePath().getPath(), request, response); - consentAdminHandler.handleTemporaryRetentionDataSyncing(consentAdminData); - return sendResponse(consentAdminData); - } - - private Response sendResponse(ConsentAdminData consentAdminData) { - if (consentAdminData.getPayload() != null || consentAdminData.getResponseStatus() != null) { - return Response.status(consentAdminData.getResponseStatus().getStatusCode()). - entity(consentAdminData.getResponsePayload()).build(); - } else { - log.debug("Response status or payload unavailable. Throwing exception"); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Response data unavailable"); - } - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/api/ConsentAuthorizeEndpoint.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/api/ConsentAuthorizeEndpoint.java deleted file mode 100644 index 0dc62439..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/api/ConsentAuthorizeEndpoint.java +++ /dev/null @@ -1,361 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.endpoint.api; - -import com.google.gson.Gson; -import com.google.gson.JsonObject; -import com.google.gson.JsonParser; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.consent.endpoint.util.ConsentConstants; -import com.wso2.openbanking.accelerator.consent.endpoint.util.ConsentUtils; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.builder.ConsentStepsBuilder; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentData; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentPersistData; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentPersistStep; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentRetrievalStep; -import com.wso2.openbanking.accelerator.consent.extensions.common.AuthErrorCode; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentCache; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionExporter; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionUtils; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import io.swagger.jaxrs.PATCH; -import net.minidev.json.JSONObject; -import net.minidev.json.JSONValue; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.oauth.cache.SessionDataCacheEntry; -import org.wso2.carbon.identity.oauth2.model.OAuth2Parameters; - -import java.io.Serializable; -import java.net.URI; -import java.net.URISyntaxException; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.Set; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; -import javax.ws.rs.Consumes; -import javax.ws.rs.GET; -import javax.ws.rs.Path; -import javax.ws.rs.PathParam; -import javax.ws.rs.Produces; -import javax.ws.rs.QueryParam; -import javax.ws.rs.core.Context; -import javax.ws.rs.core.MediaType; -import javax.ws.rs.core.Response; - -/** - * ConsentAuthorizeEndpoint. - * This specifies a RESTful API to be used in a code/hybrid flow consent approval. - */ -@SuppressFBWarnings("JAXRS_ENDPOINT") -// Suppressed content - Endpoints -// Suppression reason - False Positive : These endpoints are secured with access control -// as defined in the IS deployment.toml file -// Suppressed warning count - 2 -@Path("/authorize") -public class ConsentAuthorizeEndpoint { - - private static final Log log = LogFactory.getLog(ConsentAuthorizeEndpoint.class); - - private static final String ERROR_PERSIST_INVALID_APPROVAL = "Invalid value for approval. Should be true/false"; - private static final String ERROR_PERSIST_APPROVAL_MANDATORY = "Mandatory body parameter approval is unavailable"; - private static final String ERROR_NO_TYPE_AND_APP_DATA = "Type and application data is unavailable"; - private static final String ERROR_SERVER_ERROR = "Internal server error"; - private static final String ERROR_NO_DATA_IN_SESSION_CACHE = - "Data unavailable in session cache corresponding to the key provided"; - private static final String ERROR_CONSENT_DATA_RETRIEVAL = "Error while retrieving data consent data"; - private static final String ERROR_INVALID_VALUE_FOR_AUTHORIZE_PARAM = "\"authorize\" parameter is not defined " + - "properly or invalid"; - private static final int STATUS_FOUND = 302; - private static final String IS_ERROR = "isError"; - private static final String APPROVAL = "approval"; - private static final String COOKIES = "cookies"; - private static List consentPersistSteps = null; - private static List consentRetrievalSteps = null; - private static ConsentCoreServiceImpl consentCoreService = new ConsentCoreServiceImpl(); - private static final String preserveConsent = (String) OpenBankingConfigParser.getInstance().getConfiguration() - .get(ConsentConstants.PRESERVE_CONSENT); - private static final boolean storeConsent = preserveConsent == null ? false : Boolean.parseBoolean(preserveConsent); - - public ConsentAuthorizeEndpoint() { - initializeConsentSteps(); - } - - private static synchronized void initializeConsentSteps() { - - if (consentRetrievalSteps == null || consentPersistSteps == null) { - ConsentStepsBuilder consentStepsBuilder = ConsentExtensionExporter.getConsentStepsBuilder(); - - if (consentStepsBuilder != null) { - consentRetrievalSteps = consentStepsBuilder.getConsentRetrievalSteps(); - consentPersistSteps = consentStepsBuilder.getConsentPersistSteps(); - } - - if (consentRetrievalSteps != null && !consentRetrievalSteps.isEmpty()) { - log.info("Consent retrieval steps are not null or empty"); - } else { - log.warn("Consent retrieval steps are null or empty"); - } - if (consentPersistSteps != null && !consentPersistSteps.isEmpty()) { - log.info("Consent persist steps are not null or empty"); - } else { - log.warn("Consent persist steps are null or empty"); - } - } else { - log.debug("Retrieval and persist steps are available"); - } - } - - /** - * Retrieve data for consent page. - */ - @GET - @Path("/retrieve/{session-data-key}") - @Consumes({"application/x-www-form-urlencoded"}) - @Produces({"application/json; charset=utf-8"}) - public Response retrieve(@Context HttpServletRequest request, @Context HttpServletResponse response, - @PathParam("session-data-key") String sessionDataKey) throws ConsentException, - ConsentManagementException { - - String loggedInUser; - String app; - String spQueryParams; - String scopeString; - - SessionDataCacheEntry cacheEntry = ConsentCache.getCacheEntryFromSessionDataKey(sessionDataKey); - OAuth2Parameters oAuth2Parameters = cacheEntry.getoAuth2Parameters(); - URI redirectURI; - try { - redirectURI = new URI(oAuth2Parameters.getRedirectURI()); - } catch (URISyntaxException e) { - //Unlikely to happen. In case it happens, error response is sent - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Invalid redirect URI"); - } - //Extracting client ID for regulatory identification and redirect URI for error redirects - String clientId = oAuth2Parameters.getClientId(); - String state = oAuth2Parameters.getState(); - - Map sensitiveDataMap = - ConsentExtensionUtils.getSensitiveDataWithConsentKey(sessionDataKey); - - if ("false".equals(sensitiveDataMap.get(IS_ERROR))) { - loggedInUser = (String) sensitiveDataMap.get("loggedInUser"); - app = (String) sensitiveDataMap.get("application"); - spQueryParams = (String) sensitiveDataMap.get("spQueryParams"); - scopeString = (String) sensitiveDataMap.get("scope"); - if (!scopeString.contains("openid")) { - String[] scopes = cacheEntry.getParamMap().get("scope"); - if (scopes != null && scopes.length != 0 && scopes[0].contains("openid")) { - scopeString = scopes[0]; - } - } - } else { - String isError = (String) sensitiveDataMap.get(IS_ERROR); - //Have to throw standard error because cannot access redirect URI with this error - log.error("Error while getting endpoint parameters. " + isError); - throw new ConsentException(redirectURI, AuthErrorCode.SERVER_ERROR, ERROR_SERVER_ERROR, state); - } - - JSONObject jsonObject = new JSONObject(); - ConsentData consentData = new ConsentData(sessionDataKey, loggedInUser, spQueryParams, scopeString, app, - ConsentExtensionUtils.getHeaders(request)); - consentData.setSensitiveDataMap(sensitiveDataMap); - consentData.setRedirectURI(redirectURI); - - if (clientId == null) { - log.error("Client Id not available"); - //Unlikely error. Included just in case. - throw new ConsentException(redirectURI, AuthErrorCode.SERVER_ERROR, ERROR_SERVER_ERROR, state); - } - consentData.setClientId(clientId); - consentData.setState(state); - - try { - consentData.setRegulatory(IdentityCommonUtil.getRegulatoryFromSPMetaData(clientId)); - } catch (OpenBankingException e) { - log.error("Error while getting regulatory data", e); - throw new ConsentException(redirectURI, AuthErrorCode.SERVER_ERROR, "Error while obtaining regulatory data", - state); - } - - executeRetrieval(consentData, jsonObject); - if (consentData.getType() == null || consentData.getApplication() == null) { - log.error(ERROR_NO_TYPE_AND_APP_DATA); - throw new ConsentException(consentData.getRedirectURI(), AuthErrorCode.SERVER_ERROR, - ERROR_SERVER_ERROR, state); - } - ConsentExtensionUtils.setCommonDataToResponse(consentData, jsonObject); - Gson gson = new Gson(); - String consent = gson.toJson(consentData); - Map authorizeData = new HashMap<>(); - authorizeData.put(consentData.getSessionDataKey(), consent); - ConsentCache.addConsentDataToCache(sessionDataKey, consentData); - if (storeConsent) { - if (consentCoreService.getConsentAttributesByName(sessionDataKey).isEmpty()) { - consentCoreService.storeConsentAttributes(consentData.getConsentId(), authorizeData); - } - } - return Response.ok(jsonObject.toJSONString(), MediaType.APPLICATION_JSON).build(); - } - - /** - * Persist user consent data. - */ - @PATCH - @Path("/persist/{session-data-key}") - @Consumes({"application/json; charset=utf-8"}) - @Produces({"application/json; charset=utf-8"}) - public Response persist(@Context HttpServletRequest request, @Context HttpServletResponse response, - @PathParam("session-data-key") String sessionDataKey, - @QueryParam("authorize") String authorize) - throws ConsentException, ConsentManagementException, URISyntaxException { - - ConsentData consentData = ConsentCache.getConsentDataFromCache(sessionDataKey); - URI location; - try { - if (consentData == null) { - if (storeConsent) { - Map consentDetailsMap = - consentCoreService.getConsentAttributesByName(sessionDataKey); - if (consentDetailsMap.isEmpty()) { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Unable to get consent data"); - } - Set keys = consentDetailsMap.keySet(); - String consentId = new ArrayList<>(keys).get(0); - JsonObject consentDetails = new JsonParser() - .parse(consentDetailsMap.get(consentId)).getAsJsonObject(); - consentData = ConsentUtils.getConsentDataFromAttributes(consentDetails, sessionDataKey); - - if (consentDetailsMap.isEmpty()) { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Unable to get consent data"); - } - } else { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Unable to get consent data"); - } - } - JSONObject payload; - try { - payload = ConsentUtils.getJSONObjectPayload(request); - } catch (ConsentException e) { - throw new ConsentException(consentData.getRedirectURI(), AuthErrorCode.SERVER_ERROR, - ERROR_NO_DATA_IN_SESSION_CACHE, consentData.getState()); - } - Map headers = ConsentExtensionUtils.getHeaders(request); - - if (payload == null) { - throw new ConsentException(consentData.getRedirectURI(), AuthErrorCode.SERVER_ERROR, - "Payload unavailable", consentData.getState()); - } - - boolean approval; - if (payload.containsKey(APPROVAL)) { - try { - if (payload.get(APPROVAL) instanceof Boolean) { - approval = (Boolean) payload.get(APPROVAL); - } else { - approval = Boolean.parseBoolean((String) payload.get(APPROVAL)); - } - } catch (ClassCastException e) { - log.error("Error while processing consent persistence approval", e); - throw new ConsentException(consentData.getRedirectURI(), AuthErrorCode.SERVER_ERROR, - ERROR_PERSIST_INVALID_APPROVAL, consentData.getState()); - } - } else { - throw new ConsentException(consentData.getRedirectURI(), AuthErrorCode.SERVER_ERROR, - ERROR_PERSIST_APPROVAL_MANDATORY, consentData.getState()); - } - - ConsentPersistData consentPersistData = new ConsentPersistData(payload, headers, approval, consentData); - - if (payload.containsKey(COOKIES)) { - consentPersistData.setBrowserCookies((Map) payload.get(COOKIES)); - } - - executePersistence(consentPersistData); - - if (!approval) { - throw new ConsentException(consentData.getRedirectURI(), AuthErrorCode.ACCESS_DENIED, - "User denied the consent", consentData.getState()); - } else if (authorize != null && !StringUtils.equals("true", authorize)) { - if (StringUtils.equals(StringUtils.EMPTY, authorize) || !StringUtils.equals("false", authorize)) { - /* "authorize" parameter comes as an empty string only when a value was not defined for the parameter in - the URL. Throwing an error since a value must be present for the query parameter. Also, the value should - only be true or false */ - throw new ConsentException(consentData.getRedirectURI(), AuthErrorCode.INVALID_REQUEST, - ERROR_INVALID_VALUE_FOR_AUTHORIZE_PARAM, consentData.getState()); - } else { - return Response.ok().build(); - } - } else { - location = ConsentUtils.authorizeRequest(Boolean.toString(consentPersistData.getApproval()) - , consentPersistData.getBrowserCookies(), consentData); - } - } finally { - if (storeConsent && consentData != null) { - // remove all session data related to the consent from consent attributes - ArrayList keysToDelete = new ArrayList<>(); - - Map consentAttributes = consentCoreService. - getConsentAttributes(consentData.getConsentId()).getConsentAttributes(); - - consentAttributes.forEach((key, value) -> { - if (JSONValue.isValidJson(value) && value.contains("sessionDataKey")) { - keysToDelete.add(key); - } - }); - - consentCoreService.deleteConsentAttributes(consentData.getConsentId(), - keysToDelete); - } - } - - return Response.status(STATUS_FOUND).location(location).build(); - } - - private void executeRetrieval(ConsentData consentData, JSONObject jsonObject) throws ConsentException { - - for (ConsentRetrievalStep step : consentRetrievalSteps) { - if (log.isDebugEnabled()) { - log.debug("Executing retrieval step " + step.getClass().toString()); - } - step.execute(consentData, jsonObject); - } - } - - private void executePersistence(ConsentPersistData consentPersistData) throws ConsentException { - - for (ConsentPersistStep step : consentPersistSteps) { - if (log.isDebugEnabled()) { - log.debug("Executing persistence step " + step.getClass().toString()); - } - step.execute(consentPersistData); - } - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/api/ConsentManageEndpoint.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/api/ConsentManageEndpoint.java deleted file mode 100644 index f8054cba..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/api/ConsentManageEndpoint.java +++ /dev/null @@ -1,228 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.endpoint.api; - -import com.wso2.openbanking.accelerator.consent.endpoint.util.ConsentUtils; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionExporter; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionUtils; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.consent.extensions.manage.builder.ConsentManageBuilder; -import com.wso2.openbanking.accelerator.consent.extensions.manage.model.ConsentManageData; -import com.wso2.openbanking.accelerator.consent.extensions.manage.model.ConsentManageHandler; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import io.swagger.jaxrs.PATCH; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; -import javax.ws.rs.Consumes; -import javax.ws.rs.DELETE; -import javax.ws.rs.GET; -import javax.ws.rs.POST; -import javax.ws.rs.PUT; -import javax.ws.rs.Path; -import javax.ws.rs.Produces; -import javax.ws.rs.core.Context; -import javax.ws.rs.core.Response; -import javax.ws.rs.core.UriInfo; - -/** - * ConsentManageEndpoint. - *

- * This specifies a RESTful API for Open Banking specification based consent requests. - */ -@SuppressFBWarnings("JAXRS_ENDPOINT") -// Suppressed content - Endpoints -// Suppression reason - False Positive : These endpoints are secured with access control -// as defined in the IS deployment.toml file -// Suppressed warning count - 7 -@Path("/manage") -public class ConsentManageEndpoint { - - private static final Log log = LogFactory.getLog(ConsentManageEndpoint.class); - - private static ConsentManageHandler consentManageHandler = null; - private static final String CLIENT_ID_HEADER = "x-wso2-client-id"; - - public ConsentManageEndpoint() { - if (consentManageHandler == null) { - initializeConsentManageHandler(); - } - } - - private static void initializeConsentManageHandler() { - ConsentManageBuilder consentManageBuilder = ConsentExtensionExporter.getConsentManageBuilder(); - - if (consentManageBuilder != null) { - consentManageHandler = consentManageBuilder.getConsentManageHandler(); - } - - if (consentManageHandler != null) { - log.info("Consent manage handler " + consentManageHandler.getClass().getName() + "initialized"); - } else { - log.warn("Consent manage handler is null"); - } - } - - /** - * Consent GET requests. - */ - @GET - @Path("/{s:.*}") - @Consumes({"application/x-www-form-urlencoded"}) - @Produces({"application/json; charset=utf-8"}) - public Response manageGet(@Context HttpServletRequest request, @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - - ConsentManageData consentManageData = new ConsentManageData(ConsentExtensionUtils.getHeaders(request), - uriInfo.getQueryParameters(), uriInfo.getPathParameters().getFirst("s"), request, response); - consentManageData.setClientId(consentManageData.getHeaders().get(CLIENT_ID_HEADER)); - consentManageHandler.handleGet(consentManageData); - return sendResponse(consentManageData); - } - - /** - * Consent POST requests. - */ - @POST - @Path("/{s:.*}") - @Consumes({"application/json; charset=utf-8"}) - @Produces({"application/json; charset=utf-8"}) - public Response managePost(@Context HttpServletRequest request, @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - - ConsentManageData consentManageData = new ConsentManageData(ConsentExtensionUtils.getHeaders(request), - ConsentUtils.getPayload(request), uriInfo.getQueryParameters(), - uriInfo.getPathParameters().getFirst("s"), request, response); - consentManageData.setClientId(consentManageData.getHeaders().get(CLIENT_ID_HEADER)); - consentManageHandler.handlePost(consentManageData); - return sendResponse(consentManageData); - } - - /** - * Consent DELETE requests. - */ - @DELETE - @Path("/{s:.*}") - @Consumes({"application/json; charset=utf-8"}) - @Produces({"application/json; charset=utf-8"}) - public Response manageDelete(@Context HttpServletRequest request, @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - - ConsentManageData consentManageData = new ConsentManageData(ConsentExtensionUtils.getHeaders(request), - ConsentUtils.getPayload(request), uriInfo.getQueryParameters(), - uriInfo.getPathParameters().getFirst("s"), request, response); - consentManageData.setClientId(consentManageData.getHeaders().get(CLIENT_ID_HEADER)); - consentManageHandler.handleDelete(consentManageData); - return sendResponse(consentManageData); - } - - /** - * Consent PUT requests. - */ - @PUT - @Path("/{s:.*}") - @Consumes({"application/json; charset=utf-8"}) - @Produces({"application/json; charset=utf-8"}) - public Response managePut(@Context HttpServletRequest request, @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - - ConsentManageData consentManageData = new ConsentManageData(ConsentExtensionUtils.getHeaders(request), - ConsentUtils.getPayload(request), uriInfo.getQueryParameters(), - uriInfo.getPathParameters().getFirst("s"), request, response); - consentManageData.setClientId(consentManageData.getHeaders().get(CLIENT_ID_HEADER)); - consentManageHandler.handlePut(consentManageData); - return sendResponse(consentManageData); - } - - /** - * Consent PATCH requests. - */ - @PATCH - @Path("/{s:.*}") - @Consumes({"application/json; charset=utf-8"}) - @Produces({"application/json; charset=utf-8"}) - public Response managePatch(@Context HttpServletRequest request, @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - - ConsentManageData consentManageData = new ConsentManageData(ConsentExtensionUtils.getHeaders(request), - ConsentUtils.getPayload(request), uriInfo.getQueryParameters(), - uriInfo.getPathParameters().getFirst("s"), request, response); - consentManageData.setClientId(consentManageData.getHeaders().get(CLIENT_ID_HEADER)); - consentManageHandler.handlePatch(consentManageData); - return sendResponse(consentManageData); - } - - /** - * Consent File Upload POST requests. - */ - @POST - @Path("/fileUpload/{s:.*}") - @Consumes({"*/*"}) - @Produces({"application/json; charset=utf-8"}) - public Response manageFileUploadPost(@Context HttpServletRequest request, @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - - ConsentManageData consentManageData = new ConsentManageData(ConsentExtensionUtils.getHeaders(request), - ConsentUtils.getFileUploadPayload(request), uriInfo.getQueryParameters(), - uriInfo.getPathParameters().getFirst("s"), request, response); - consentManageData.setClientId(consentManageData.getHeaders().get(CLIENT_ID_HEADER)); - consentManageHandler.handleFileUploadPost(consentManageData); - return sendFileUploadResponse(consentManageData); - } - - /** - * Consent File GET requests. - */ - @GET - @Path("/fileUpload/{s:.*}") - @Consumes({"application/x-www-form-urlencoded"}) - @Produces({"*/*"}) - public Response manageFileGet(@Context HttpServletRequest request, @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - - ConsentManageData consentManageData = new ConsentManageData(ConsentExtensionUtils.getHeaders(request), - uriInfo.getQueryParameters(), uriInfo.getPathParameters().getFirst("s"), request, response); - consentManageData.setClientId(consentManageData.getHeaders().get(CLIENT_ID_HEADER)); - consentManageHandler.handleFileGet(consentManageData); - return sendResponse(consentManageData); - } - - private Response sendResponse(ConsentManageData consentManageData) { - if (consentManageData.getPayload() != null || consentManageData.getResponseStatus() != null) { - return Response.status(consentManageData.getResponseStatus().getStatusCode()). - entity(consentManageData.getResponsePayload()).build(); - } else { - log.debug("Response status or payload unavailable. Throwing exception"); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Response data unavailable"); - } - } - - private Response sendFileUploadResponse(ConsentManageData consentManageData) { - if (consentManageData.getPayload() != null || consentManageData.getResponseStatus() != null) { - return Response.status(consentManageData.getResponseStatus().getStatusCode()).build(); - } else { - log.debug("Response status or payload unavailable. Throwing exception"); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Response data unavailable"); - } - } - -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/api/ConsentValidationEndpoint.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/api/ConsentValidationEndpoint.java deleted file mode 100644 index 49698e92..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/api/ConsentValidationEndpoint.java +++ /dev/null @@ -1,200 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.endpoint.api; - -import com.wso2.openbanking.accelerator.common.exception.ConsentManagementException; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.consent.endpoint.util.ConsentUtils; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionExporter; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentExtensionUtils; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.consent.extensions.validate.builder.ConsentValidateBuilder; -import com.wso2.openbanking.accelerator.consent.extensions.validate.model.ConsentValidateData; -import com.wso2.openbanking.accelerator.consent.extensions.validate.model.ConsentValidationResult; -import com.wso2.openbanking.accelerator.consent.extensions.validate.model.ConsentValidator; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.DetailedConsentResource; -import com.wso2.openbanking.accelerator.consent.mgt.service.impl.ConsentCoreServiceImpl; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.net.URISyntaxException; -import java.text.ParseException; -import java.util.Iterator; -import java.util.Map; -import java.util.Set; -import java.util.TreeMap; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; -import javax.ws.rs.Consumes; -import javax.ws.rs.POST; -import javax.ws.rs.Path; -import javax.ws.rs.Produces; -import javax.ws.rs.core.Context; -import javax.ws.rs.core.Response; - -/** - * ConsentValidationEndpoint. - * - * This specifies a RESTful API for consent validation to be used at consent enforcement of resource - * retrieval/submission requests. - */ -@SuppressFBWarnings("JAXRS_ENDPOINT") -// Suppressed content - Endpoints -// Suppression reason - False Positive : These endpoints are secured with access control -// as defined in the IS deployment.toml file -// Suppressed warning count - 1 -@Path("/validate") -public class ConsentValidationEndpoint { - - private static final Log log = LogFactory.getLog(ConsentValidationEndpoint.class); - private static final ConsentCoreServiceImpl consentCoreService = new ConsentCoreServiceImpl(); - - private static ConsentValidator consentValidator = null; - private static String requestSignatureAlias; - - public ConsentValidationEndpoint() { - if (consentValidator == null) { - initializeConsentValidator(); - } - } - - private static void initializeConsentValidator() { - ConsentValidateBuilder consentValidateBuilder = ConsentExtensionExporter.getConsentValidateBuilder(); - - if (consentValidateBuilder != null) { - consentValidator = consentValidateBuilder.getConsentValidator(); - requestSignatureAlias = consentValidateBuilder.getRequestSignatureAlias(); - log.info("Consent validator " + consentValidator.getClass().getName() + "initialized"); - } - - if (consentValidator != null) { - log.info("Consent validator " + consentValidator.getClass().getName() + "initialized"); - } else { - log.warn("Consent validator is null"); - } - } - - /** - * Validate by sending consent data. - */ - @POST - @Path("/") - @Consumes({"application/jwt; charset=utf-8"}) - @Produces({"application/json; charset=utf-8"}) - public Response validate(@Context HttpServletRequest request, @Context HttpServletResponse response) { - - String payload = ConsentUtils.getStringPayload(request); - JSONObject requestData; - Object requestDataObj; - - if (IdentityCommonUtil.getConsentJWTPayloadValidatorConfigEnabled()) { - try { - IdentityCommonUtil.validateJWTSignatureWithPublicKey(payload, requestSignatureAlias); - requestData = JWTUtils.decodeRequestJWT(payload, "body"); - } catch (OpenBankingException e) { - log.error("Error while validating JWT signature", e); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Error while validating JWT " + - "signature"); - } catch (ParseException e) { - log.error("Error while decoding validation JWT", e); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Error while decoding validation JWT"); - } - } else { - try { - requestDataObj = new JSONParser(JSONParser.MODE_PERMISSIVE).parse(payload); - } catch (net.minidev.json.parser.ParseException e) { - log.error("Unable to parse the request payload", e); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Unable to parse the request payload"); - } - if (!(requestDataObj instanceof JSONObject)) { - throw new ConsentException(ResponseStatus.BAD_REQUEST, "Payload is not a JSON object"); - } else { - requestData = (JSONObject) requestDataObj; - } - } - - JSONObject requestHeaders = (JSONObject) requestData.get("headers"); - Set headerNames = requestHeaders.keySet(); - Iterator headersIterator = headerNames.iterator(); - TreeMap headersMap = new TreeMap<>(String.CASE_INSENSITIVE_ORDER); - while (headersIterator.hasNext()) { - String headerName = headersIterator.next(); - headersMap.put(headerName, requestHeaders.getAsString(headerName)); - } - - JSONObject requestPayload = (JSONObject) requestData.get("body"); - String requestPath = requestData.getAsString("electedResource"); - String consentId = requestData.getAsString("consentId"); - String userId = requestData.getAsString("userId"); - String clientId = null; - if (requestData.containsKey("clientId")) { - clientId = requestData.getAsString("clientId"); - } - Map resourceParams = (Map) requestData.get("resourceParams"); - - if (consentId == null) { - throw new ConsentException(ResponseStatus.BAD_REQUEST, "Consent Id is mandatory for consent validation"); - } - - try { - //Adding query parameters to the resource map - resourceParams = ConsentUtils.addQueryParametersToResourceParamMap(resourceParams); - } catch (URISyntaxException e) { - log.error("Error while extracting query parameters", e); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Error while extracting query parameters"); - } - - ConsentValidateData consentValidateData = new ConsentValidateData(requestHeaders, requestPayload, - requestPath, consentId, userId, clientId, resourceParams, headersMap); - - try { - DetailedConsentResource consentResource = consentCoreService.getDetailedConsent(consentId); - consentValidateData.setComprehensiveConsent(consentResource); - } catch (ConsentManagementException e) { - log.error("Exception while getting consent", e); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Exception while getting consent"); - } - - ConsentValidationResult validationResult = new ConsentValidationResult(); - consentValidator.validate(consentValidateData, validationResult); - - JSONObject information = ConsentExtensionUtils.detailedConsentToJSON( - consentValidateData.getComprehensiveConsent()); - information.put("additionalConsentInfo", validationResult.getConsentInformation()); - validationResult.setConsentInformation(information); - - JSONObject responsePayload; - try { - responsePayload = validationResult.generatePayload(); - responsePayload.appendField("consentInformation", - IdentityCommonUtil.signJWTWithDefaultKey(validationResult.getConsentInformation().toJSONString())); - } catch (Exception e) { - log.error("Error occurred while getting private key", e); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, "Error while getting private key"); - } - return Response.status(HttpServletResponse.SC_OK).entity(responsePayload).build(); - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/error/ConsentThrowableMapper.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/error/ConsentThrowableMapper.java deleted file mode 100644 index 7f9ad49a..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/error/ConsentThrowableMapper.java +++ /dev/null @@ -1,62 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.endpoint.error; - -import com.wso2.openbanking.accelerator.consent.endpoint.util.ConsentConstants; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import javax.ws.rs.ClientErrorException; -import javax.ws.rs.core.Response; -import javax.ws.rs.ext.ExceptionMapper; - -/** - * Map exceptions to custom error format. - */ -public class ConsentThrowableMapper implements ExceptionMapper { - - private static final Log log = LogFactory.getLog(ConsentThrowableMapper.class); - private static String defaultError = "A runtime error has occurred while handling the request"; - - @Override - public Response toResponse(Throwable throwable) { - - if (throwable instanceof ConsentException) { - if (((ConsentException) throwable).getErrorRedirectURI() != null) { - return Response.status(((ConsentException) throwable).getStatus().getStatusCode()). - location(((ConsentException) throwable).getErrorRedirectURI()).build(); - } else { - return Response.status(((ConsentException) throwable).getStatus().getStatusCode()) - .entity(((ConsentException) throwable).getPayload().toJSONString()) - .header(ConsentConstants.HEADER_CONTENT_TYPE, ConsentConstants.DEFAULT_RESPONSE_CONTENT_TYPE) - .build(); - } - } else { - log.error("Generic exception. Cause: " + throwable.getMessage(), throwable); - if (throwable instanceof ClientErrorException) { - return toResponse(new ConsentException(ResponseStatus.fromStatusCode(((ClientErrorException) throwable) - .getResponse().getStatus()), throwable.getMessage())); - } else { - return toResponse(new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, defaultError)); - } - } - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/util/ConsentConstants.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/util/ConsentConstants.java deleted file mode 100644 index 3dfab0eb..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/util/ConsentConstants.java +++ /dev/null @@ -1,52 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.endpoint.util; - -/** - * Constant class for consent authorize endpoints. - */ -public class ConsentConstants { - - public static final String HEADER_CONTENT_TYPE = "Content-Type"; - public static final String APPLICATION_JSON = "application/json"; - public static final String DEFAULT_RESPONSE_CONTENT_TYPE = APPLICATION_JSON; - - public static final String ERROR_PAYLOAD_READ = "Error while reading payload"; - public static final String ERROR_PAYLOAD_PARSE = "Error while parsing payload"; - public static final String RESOURCE_PATH = "ResourcePath"; - public static final String HTTP_METHOD = "HttpMethod"; - public static final String RESOURCE_CONTEXT = "ResourceContext"; - public static final String PRESERVE_CONSENT = "Consent.PreserveConsentLink"; - public static final String SENSITIVE_DATA_MAP = "sensitiveDataMap"; - public static final String LOGGED_IN_USER = "loggedInUser"; - public static final String SP_QUERY_PARAMS = "spQueryParams"; - public static final String SCOPES = "scopeString"; - public static final String APPLICATION = "application"; - public static final String REQUEST_HEADERS = "requestHeaders"; - public static final String REQUEST_URI = "redirectURI"; - public static final String USERID = "userId"; - public static final String CONSENT_ID = "consentId"; - public static final String CLIENT_ID = "clientId"; - public static final String REGULATORY = "regulatory"; - public static final String CONSENT_RESOURCE = "consentResource"; - public static final String AUTH_RESOURCE = "authResource"; - public static final String META_DATA = "metaDataMap"; - public static final String TYPE = "type"; - -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/util/ConsentUtils.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/util/ConsentUtils.java deleted file mode 100644 index dd95e1f0..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/java/com/wso2/openbanking/accelerator/consent/endpoint/util/ConsentUtils.java +++ /dev/null @@ -1,358 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.consent.endpoint.util; - -import com.google.gson.Gson; -import com.google.gson.JsonObject; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.consent.extensions.authorize.model.ConsentData; -import com.wso2.openbanking.accelerator.consent.extensions.common.AuthErrorCode; -import com.wso2.openbanking.accelerator.consent.extensions.common.ConsentException; -import com.wso2.openbanking.accelerator.consent.extensions.common.ResponseStatus; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.AuthorizationResource; -import com.wso2.openbanking.accelerator.consent.mgt.dao.models.ConsentResource; -import com.wso2.openbanking.accelerator.identity.util.HTTPClientUtils; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.commons.httpclient.util.HttpURLConnection; -import org.apache.commons.io.IOUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.apache.http.HttpResponse; -import org.apache.http.NameValuePair; -import org.apache.http.client.entity.UrlEncodedFormEntity; -import org.apache.http.client.methods.HttpPost; -import org.apache.http.client.protocol.HttpClientContext; -import org.apache.http.impl.client.BasicCookieStore; -import org.apache.http.impl.client.CloseableHttpClient; -import org.apache.http.impl.cookie.BasicClientCookie; -import org.apache.http.message.BasicNameValuePair; -import org.apache.http.protocol.BasicHttpContext; -import org.apache.http.protocol.HttpContext; -import org.wso2.carbon.context.PrivilegedCarbonContext; -import org.wso2.carbon.identity.application.common.IdentityApplicationManagementException; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.application.common.util.IdentityApplicationConstants; -import org.wso2.carbon.identity.application.mgt.ApplicationManagementService; -import org.wso2.carbon.identity.core.util.IdentityUtil; - -import java.io.IOException; -import java.net.URI; -import java.net.URISyntaxException; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.Optional; - -import javax.servlet.http.HttpServletRequest; -import javax.ws.rs.WebApplicationException; - -/** - * Utils class for consent authorize endpoints. - */ -public class ConsentUtils { - - private static final Log log = LogFactory.getLog(ConsentUtils.class); - private static final String ERROR_FETCHING_SP = "Error while fetching service provider"; - private static Gson gson = new Gson(); - - /** - * Send authorize request in order to complete the authorize flow and get the redirect. - * - * @param consent The approval/denial of the consent of the user - * @param cookies The session cookies used in auth flow - * @param consentData Consent data object which contains consent information - * @return The redirect URI to end the authorize flow - */ - public static URI authorizeRequest(String consent, Map cookies, ConsentData consentData) { - - String authorizeURL = IdentityUtil.getProperty("OAuth.OAuth2AuthzEPUrl"); - try (CloseableHttpClient client = HTTPClientUtils.getHttpsClient()) { - - BasicCookieStore cookieStore = new BasicCookieStore(); - String cookieDomain = new URI(authorizeURL).getHost(); - for (Map.Entry cookieValue : cookies.entrySet()) { - BasicClientCookie cookie = new BasicClientCookie(cookieValue.getKey(), cookieValue.getValue()); - cookie.setDomain(cookieDomain); - cookie.setPath("/"); - cookie.setSecure(true); - cookieStore.addCookie(cookie); - } - HttpPost authorizeRequest = new HttpPost(authorizeURL); - List params = new ArrayList<>(); - params.add(new BasicNameValuePair("hasApprovedAlways", "false")); - params.add(new BasicNameValuePair("sessionDataKeyConsent", consentData.getSessionDataKey())); - params.add(new BasicNameValuePair("consent", consent)); - params.add(new BasicNameValuePair("user", consentData.getUserId())); - HttpContext localContext = new BasicHttpContext(); - localContext.setAttribute(HttpClientContext.COOKIE_STORE, cookieStore); - UrlEncodedFormEntity entity = new UrlEncodedFormEntity(params); - authorizeRequest.setEntity(entity); - HttpResponse authorizeResponse = client.execute(authorizeRequest, localContext); - - if (authorizeResponse.getStatusLine().getStatusCode() != HttpURLConnection.HTTP_MOVED_TEMP) { - throw new ConsentException(consentData.getRedirectURI(), AuthErrorCode.SERVER_ERROR, - "Error while getting authorize redirect", consentData.getState()); - } else { - //Extract the location header from the authorization redirect - return new URI(authorizeResponse.getLastHeader("Location").getValue()); - } - } catch (IOException e) { - log.error("Error while sending authorize request to complete the authorize flow", e); - return null; - } catch (URISyntaxException e) { - log.error("Authorize response URI syntax error", e); - throw new ConsentException(consentData.getRedirectURI(), AuthErrorCode.SERVER_ERROR, - "Internal server error", consentData.getState()); - } catch (OpenBankingException e) { - log.error("Error while obtaining HTTP client", e); - throw new ConsentException(consentData.getRedirectURI(), AuthErrorCode.SERVER_ERROR, - "Internal server error", consentData.getState()); - } - } - - /** - * Util method to extract the payload from a HTTP request object. Can be JSONObject or JSONArray - * - * @param request The HTTP request object - * @return Object payload can be either an instance of JSONObject or JSONArray only. Can be a ConsentException if - * is and error scenario. Error is returned instead of throwing since the error response should be handled by the - * toolkit is the manage scenario. - */ - public static Object getPayload(HttpServletRequest request) { - try { - Object payload = new JSONParser(JSONParser.MODE_PERMISSIVE).parse(getStringPayload(request)); - if (payload == null) { - log.debug("Payload is empty. Returning null"); - return null; - } - if (!(payload instanceof JSONObject || payload instanceof JSONArray)) { - //Not throwing error since error should be formatted by manage toolkit - log.error("Payload is not a JSON. Returning null"); - return null; - } - return payload; - } catch (ParseException e) { - //Not throwing error since error should be formatted by manage toolkit - log.error(ConsentConstants.ERROR_PAYLOAD_PARSE + ". Returning null", e); - return null; - } catch (ConsentException e) { - //Not throwing error since error should be formatted by manage toolkit - log.error(e.getMessage() + ". Returning null", e); - return null; - } - } - - /** - * Util method to extract the payload from a HTTP request object. Can be only JSONObject - * - * @param request The request object - * @return JSONObject payload can only be an instance of JSONObject - * @throws ConsentException Parser errors and payload type is not JSON object - */ - public static JSONObject getJSONObjectPayload(HttpServletRequest request) throws ConsentException { - try { - Object payload = new JSONParser(JSONParser.MODE_PERMISSIVE).parse(getStringPayload(request)); - //JSONArray not supported here. If requirement arises, cast the object to JSONArray from here - if (payload == null) { - return null; - } - if (!(payload instanceof JSONObject)) { - return null; - } - return (JSONObject) payload; - } catch (ParseException e) { - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, ConsentConstants.ERROR_PAYLOAD_PARSE); - } catch (ConsentException e) { - //Not throwing error since error should be formatted by manage toolkit - log.error(e.getMessage() + ". Returning null", e); - return null; - } - } - - /** - * Extract string payload from request object. - * - * @param request The request object - * @return String payload - * @throws ConsentException Payload read errors - */ - public static String getStringPayload(HttpServletRequest request) throws ConsentException { - try { - return IOUtils.toString(request.getInputStream()); - } catch (IOException e) { - log.error(ConsentConstants.ERROR_PAYLOAD_READ, e); - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, ConsentConstants.ERROR_PAYLOAD_READ); - } - } - - /** - * Util method to extract the payload from a HTTP request object. - * - * @param request The HTTP request object - * @return Object payload can be xml or json. Can be a ConsentException if is and error scenario. - * Error is returned instead of throwing since the error response should be handled by the - * toolkit is the manage scenario. - */ - public static Object getFileUploadPayload(HttpServletRequest request) { - try { - String payload = getStringPayload(request); - if (payload == null) { - log.debug("Payload is empty. Returning null"); - return null; - } - return payload; - } catch (ConsentException e) { - //Not throwing error since error should be formatted by manage toolkit - log.error(e.getMessage() + ". Returning null", e); - return null; - } - } - - - /** - * Extract and add query parameters from a URL to existing resource map. - * Resource parameter map will contain the resource path(ex: /aisp/accounts/{AccountId}?queryParam=queryParamValue), - * http method, context(ex: /open-banking/v3.1/aisp) - * - * @param resourceParams Map containing the resource parameters - * @return Extracted query parameter map - */ - public static Map addQueryParametersToResourceParamMap(Map resourceParams) - throws URISyntaxException { - - if (resourceParams.isEmpty()) { - return new HashMap(); - } - - URI url = new URI((String) resourceParams.get("resource")); - - resourceParams.put(ConsentConstants.RESOURCE_PATH, url.getRawPath()); - - if (url.getRawQuery() != null) { - String[] params = url.getRawQuery().split("&"); - - for (String param : params) { - if (param.split("=").length == 2) { - String name = param.split("=")[0]; - String value = param.split("=")[1]; - resourceParams.put(name, value); - } - } - } - return resourceParams; - } - - /** - * Get Service provider from clientId. - * - * @param clientId of application. - * @return Service Provider. - * @throws WebApplicationException client error. - */ - public static ServiceProvider getOAuthServiceProvider(String clientId) throws WebApplicationException { - - ApplicationManagementService managementService = getApplicationManagementService(); - Optional serviceProvider; - try { - serviceProvider = Optional.ofNullable(managementService.getServiceProviderByClientId(clientId, - IdentityApplicationConstants.OAuth2.NAME, getTenantDomain())); - } catch (IdentityApplicationManagementException e) { - - log.error(String.format("Unable to retrieve service provider information for clientId %s", clientId), e); - // Throw Web Application exception - throw new ConsentException(ResponseStatus.INTERNAL_SERVER_ERROR, ERROR_FETCHING_SP); - } - - // Reject default service provider and empty service provider. - if (!serviceProvider.isPresent() || - serviceProvider.get().getApplicationName().equals(IdentityApplicationConstants.DEFAULT_SP_CONFIG)) { - - final String errorMessage = String.format("Unable to find application for clientId %s", clientId); - - if (log.isDebugEnabled()) { - log.debug(errorMessage); - } - - // Throw client error for not found service provider. - throw new ConsentException(ResponseStatus.NOT_FOUND, errorMessage); - } - return serviceProvider.get(); - } - - /** - * Get WSO2 IS Application Mgt Service from threadlocal carbon context. - * - * @return Application Management Service Implementation. - */ - public static ApplicationManagementService getApplicationManagementService() { - - return (ApplicationManagementService) PrivilegedCarbonContext - .getThreadLocalCarbonContext() - .getOSGiService(ApplicationManagementService.class, null); - - } - - /** - * Get Tenant Domain String from carbon context. - * - * @return tenant domain of current context. - */ - private static String getTenantDomain() { - - return PrivilegedCarbonContext.getThreadLocalCarbonContext().getTenantDomain(true); - } - - /** - * @param consentDetails json object of consent data - * @param sessionDataKey - * @return - * @throws URISyntaxException - */ - public static ConsentData getConsentDataFromAttributes(JsonObject consentDetails, String sessionDataKey) - throws URISyntaxException { - - JsonObject sensitiveDataMap = consentDetails.get(ConsentConstants.SENSITIVE_DATA_MAP).getAsJsonObject(); - ConsentData consentData = new ConsentData(sessionDataKey, - sensitiveDataMap.get(ConsentConstants.LOGGED_IN_USER).getAsString(), - sensitiveDataMap.get(ConsentConstants.SP_QUERY_PARAMS).getAsString(), - consentDetails.get(ConsentConstants.SCOPES).getAsString(), - sensitiveDataMap.get(ConsentConstants.APPLICATION).getAsString(), - gson.fromJson(consentDetails.get(ConsentConstants.REQUEST_HEADERS), Map.class)); - consentData.setSensitiveDataMap(gson.fromJson(sensitiveDataMap, Map.class)); - URI redirectURI = new URI(consentDetails.get(ConsentConstants.REQUEST_URI).getAsString()); - consentData.setRedirectURI(redirectURI); - consentData.setUserId(consentDetails.get(ConsentConstants.USERID).getAsString()); - consentData.setConsentId(consentDetails.get(ConsentConstants.CONSENT_ID).getAsString()); - consentData.setClientId(consentDetails.get(ConsentConstants.CLIENT_ID).getAsString()); - consentData.setRegulatory(Boolean.parseBoolean(consentDetails.get(ConsentConstants.REGULATORY).getAsString())); - ConsentResource consentResource = gson.fromJson(consentDetails.get(ConsentConstants.CONSENT_RESOURCE), - ConsentResource.class); - consentData.setConsentResource(consentResource); - AuthorizationResource authorizationResource = - gson.fromJson(consentDetails.get(ConsentConstants.AUTH_RESOURCE), AuthorizationResource.class); - consentData.setAuthResource(authorizationResource); - consentData.setMetaDataMap(gson.fromJson(consentDetails.get(ConsentConstants.META_DATA), Map.class)); - consentData.setType(consentDetails.get(ConsentConstants.TYPE).getAsString()); - return consentData; - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/resources/findbugs-exclude.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/resources/findbugs-exclude.xml deleted file mode 100644 index c4f8e532..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/resources/findbugs-exclude.xml +++ /dev/null @@ -1,21 +0,0 @@ - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/resources/findbugs-include.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/resources/findbugs-include.xml deleted file mode 100644 index 8932a22e..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/webapp/META-INF/webapp-classloading.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/webapp/META-INF/webapp-classloading.xml deleted file mode 100644 index 7569f0a1..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/webapp/META-INF/webapp-classloading.xml +++ /dev/null @@ -1,35 +0,0 @@ - - - - - - - - - false - - - CXF3,Carbon - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/webapp/WEB-INF/beans.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/webapp/WEB-INF/beans.xml deleted file mode 100644 index 048b8809..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/webapp/WEB-INF/beans.xml +++ /dev/null @@ -1,39 +0,0 @@ - - - - - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/webapp/WEB-INF/web.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/webapp/WEB-INF/web.xml deleted file mode 100644 index 3879430c..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint/src/main/webapp/WEB-INF/web.xml +++ /dev/null @@ -1,71 +0,0 @@ - - - - - - - WSO2 Open Banking - Consent API - WSO2 Open Banking - Consent API - - - contextConfigLocation - WEB-INF/beans.xml - - - - HttpHeaderSecurityFilter - org.apache.catalina.filters.HttpHeaderSecurityFilter - - hstsEnabled - false - - - - - HttpHeaderSecurityFilter - * - - - - - org.springframework.web.context.ContextLoaderListener - - - - - CXFServlet - - org.apache.cxf.transport.servlet.CXFServlet - - 1 - - - - CXFServlet - /* - - - - 60 - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/pom.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/pom.xml deleted file mode 100644 index 6eb99a78..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/pom.xml +++ /dev/null @@ -1,203 +0,0 @@ - - - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../../pom.xml - - 4.0.0 - - com.wso2.openbanking.accelerator.dcr.endpoint - WSO2 Open Banking - Dynamic Client Registration Endpoint - WSO2 Open Banking - Dynamic Client Endpoint - war - - - - io.swagger - swagger-jaxrs - - - javax.ws.rs - jsr311-api - - - com.google.guava - guava - - - org.yaml - snakeyaml - - - - - javax.validation - validation-api - provided - - - org.springframework - spring-web - provided - - - org.apache.cxf - cxf-bundle-jaxrs - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.identity - - - javax.ws.rs - jsr311-api - - - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - provided - - - org.wso2.carbon.identity.inbound.auth.oauth2 - org.wso2.carbon.identity.oauth.dcr - provided - - - org.wso2.carbon.identity.framework - org.wso2.carbon.identity.application.mgt - provided - - - com.fasterxml.jackson.core - jackson-databind - provided - - - io.swagger - swagger-annotations - - - net.minidev - json-smart - provided - - - - - - - org.codehaus.mojo - build-helper-maven-plugin - - - add-source-api - generate-sources - - add-source - - - - src/gen/java - - - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - false - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - maven-war-plugin - ${maven-war-plugin.version} - - - - - src/main/webapp - - - api#openbanking#dynamic-client-registration - WEB-INF/lib/axis2-kernel-1.6.1-wso2v12.jar, - WEB-INF/lib/slf4j-api-*.jar - - - - org.openapitools - openapi-generator-maven-plugin - ${openapi.generator.plugin.version} - - - - generate - - - - true - true - ${project.basedir}/src/main/resources/dynamic.client.registration.yaml - - jaxrs-cxf - src/gen/java - true - - false - com.wso2.openbanking.accelerator.dynamic.client.registration.src.main.model - com.wso2.openbanking.accelerator.dynamic.client.registration.src.main.api - impl - DTO - ${project.basedir} - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/gen/java/com/wso2/openbanking/accelerator/dynamic/client/registration/dto/RegistrationErrorDTO.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/gen/java/com/wso2/openbanking/accelerator/dynamic/client/registration/dto/RegistrationErrorDTO.java deleted file mode 100644 index c3e3766e..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/gen/java/com/wso2/openbanking/accelerator/dynamic/client/registration/dto/RegistrationErrorDTO.java +++ /dev/null @@ -1,74 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.endpoint.impl.dto; - -import com.fasterxml.jackson.annotation.JsonProperty; -import io.swagger.annotations.ApiModelProperty; - -/** - * Error model for DCR - */ -public class RegistrationErrorDTO { - - private String error = null; - - private String errorDescription = null; - - /** - * - **/ - @ApiModelProperty(value = "") - @JsonProperty("error") - public String getError() { - - return error; - } - - public void setError(String error) { - - this.error = error; - } - - /** - * - **/ - @ApiModelProperty(value = "") - @JsonProperty("error_description") - public String getError_description() { - - return errorDescription; - } - - public void setErrorDescription(String errorDescription) { - - this.errorDescription = errorDescription; - } - - @Override - public String toString() { - - StringBuilder sb = new StringBuilder(); - sb.append("class ErrorDTO {\n"); - - sb.append(" error: ").append(error).append("\n"); - sb.append(" error_description: ").append(errorDescription).append("\n"); - sb.append("}\n"); - return sb.toString(); - } -} - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/RegistrationConstants.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/RegistrationConstants.java deleted file mode 100644 index 1836dc54..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/RegistrationConstants.java +++ /dev/null @@ -1,30 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.endpoint.impl; - -/** - * Constants for DCR. - */ -public class RegistrationConstants { - - public static final String CLIENT_ID_ISSUED_AT = "client_id_issued_at"; - public static final String CLIENT_ID = "client_id"; - public static final String REGISTRATION_ACCESS_TOKEN = "registration_access_token"; - public static final String CLIENT_NOT_FOUND = "NOT_FOUND"; - public static final String INTERNAL_SERVER_ERROR = "Internal Server error"; -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/api/ClientRegistrationApiImpl.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/api/ClientRegistrationApiImpl.java deleted file mode 100644 index e7e37a9a..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/api/ClientRegistrationApiImpl.java +++ /dev/null @@ -1,321 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.endpoint.impl.api; - -import com.google.gson.Gson; -import com.google.gson.JsonElement; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.identity.dcr.endpoint.impl.RegistrationConstants; -import com.wso2.openbanking.accelerator.identity.dcr.endpoint.impl.service.RegistrationServiceHandler; -import com.wso2.openbanking.accelerator.identity.dcr.endpoint.impl.util.RegistrationUtils; -import com.wso2.openbanking.accelerator.identity.dcr.exception.DCRValidationException; -import com.wso2.openbanking.accelerator.identity.dcr.model.RegistrationRequest; -import com.wso2.openbanking.accelerator.identity.dcr.validation.DCRCommonConstants; -import com.wso2.openbanking.accelerator.identity.dcr.validation.RegistrationValidator; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonConstants; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonHelper; -import com.wso2.openbanking.accelerator.identity.util.IdentityCommonUtil; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.application.common.IdentityApplicationManagementException; -import org.wso2.carbon.identity.oauth.dcr.exception.DCRMException; -import org.wso2.carbon.identity.oauth2.IdentityOAuth2Exception; - -import java.io.IOException; -import java.security.cert.CertificateEncodingException; -import java.text.ParseException; -import java.util.Enumeration; -import java.util.HashMap; -import java.util.Map; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; -import javax.ws.rs.Consumes; -import javax.ws.rs.DELETE; -import javax.ws.rs.GET; -import javax.ws.rs.POST; -import javax.ws.rs.PUT; -import javax.ws.rs.Path; -import javax.ws.rs.Produces; -import javax.ws.rs.core.Context; -import javax.ws.rs.core.Response; -import javax.ws.rs.core.UriInfo; - -/** - * Implementation class for the DCR API. - */ -@Path("/register") -public class ClientRegistrationApiImpl { - - private static final Log log = LogFactory.getLog(ClientRegistrationApiImpl.class); - - private final IdentityCommonHelper identityCommonHelper = new IdentityCommonHelper(); - private RegistrationServiceHandler registrationServiceHandler = new RegistrationServiceHandler(); - private static Gson gson = new Gson(); - - @DELETE - @Path("/{s:.*}") - @Produces({"application/json; charset=utf-8"}) - public Response registerClientIdDelete(@Context HttpServletRequest request, @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - - RegistrationValidator validator = RegistrationValidator.getRegistrationValidator(); - if (log.isDebugEnabled()) { - log.debug("Invoking the configured registration validator:" + validator); - } - String clientId = uriInfo.getPathParameters().getFirst("s"); - - try { - validator.validateDelete(clientId); - identityCommonHelper.revokeAccessTokensByClientId(clientId); - return registrationServiceHandler.deleteRegistration(clientId); - } catch (DCRMException e) { - log.error("Error while deleting the application", e); - if (e.getErrorCode().contains(RegistrationConstants.CLIENT_NOT_FOUND)) { - return Response.status(Response.Status.UNAUTHORIZED).build(); - } - } catch (DCRValidationException e) { - log.error("Error occurred while validating request", e); - return Response.status(Response.Status.BAD_REQUEST).entity(RegistrationUtils.getErrorDTO(e.getErrorCode(), - e.getErrorDescription())).build(); - } catch (IdentityOAuth2Exception e) { - log.error("Error occurred while revoking application access tokens", e); - } - return Response.status(Response.Status.INTERNAL_SERVER_ERROR).entity(RegistrationUtils - .getErrorDTO(RegistrationConstants.INTERNAL_SERVER_ERROR, - "Error occurred while deleting the application")).build(); - } - - @GET - @Path("/{s:.*}") - @Consumes({"application/x-www-form-urlencoded"}) - @Produces({"application/json; charset=utf-8"}) - public Response registerClientIdGet(@Context HttpServletRequest request, @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - - try { - String clientId = uriInfo.getPathParameters().getFirst("s"); - - RegistrationValidator validator = RegistrationValidator.getRegistrationValidator(); - if (log.isDebugEnabled()) { - log.debug("Invoking the configured registration validator:" + validator); - } - validator.validateGet(clientId); - Map headers = getHeaders(request); - String accessToken = headers.get(RegistrationConstants.REGISTRATION_ACCESS_TOKEN); - - Map additionalAttributes = new HashMap<>(); - - try { - String tlsCert = new IdentityCommonHelper().encodeCertificateContent( - IdentityCommonUtil.getCertificateFromAttribute( - request.getAttribute(IdentityCommonConstants.JAVAX_SERVLET_REQUEST_CERTIFICATE))); - - // add TLS cert as additional attribute - additionalAttributes.put(IdentityCommonConstants.TLS_CERT, tlsCert); - } catch (CertificateEncodingException e) { - log.error("Certificate not valid", e); - } - - return registrationServiceHandler.retrieveRegistration(additionalAttributes, clientId, accessToken); - } catch (DCRMException e) { - log.error("Error while retrieving application", e); - if (e.getErrorCode().contains(RegistrationConstants.CLIENT_NOT_FOUND)) { - return Response.status(Response.Status.UNAUTHORIZED).build(); - } - } catch (IdentityApplicationManagementException e) { - log.error("Error while retrieving Service Provider details", e); - } catch (DCRValidationException e) { - log.error("Error occurred while validating request", e); - return Response.status(Response.Status.BAD_REQUEST).entity(RegistrationUtils.getErrorDTO(e.getErrorCode(), - e.getErrorDescription())).build(); - } - return Response.status(Response.Status.INTERNAL_SERVER_ERROR).entity( - RegistrationUtils.getErrorDTO(RegistrationConstants.INTERNAL_SERVER_ERROR, - "Error occurred while processing the request")).build(); - } - - @PUT - @Path("/{s:.*}") - @Consumes({"application/json; charset=utf-8"}) - @Produces({"application/json; charset=utf-8"}) - public Response registerClientIdPut(@Context HttpServletRequest request, @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - - //RegistrationRequest registrationRequest = RegistrationUtils.getRegistrationRequest(requestBody); - - RegistrationValidator registrationValidator = RegistrationValidator.getRegistrationValidator(); - if (log.isDebugEnabled()) { - log.debug("Invoking the configured registration validator:" + registrationValidator); - } - try { - JsonElement registrationRequestDetails = gson.toJsonTree(RegistrationUtils.getPayload(request)); - - RegistrationRequest registrationRequest = - gson.fromJson(registrationRequestDetails, RegistrationRequest.class); - - Map requestAttributes = (Map) - gson.fromJson(registrationRequestDetails, Map.class); - - registrationRequest.setRequestParameters(requestAttributes); - - if (StringUtils.isNotEmpty(registrationRequest.getSoftwareStatement())) { - //decode SSA if provided in the registration request - String ssaBody = JWTUtils.decodeRequestJWT(registrationRequest.getSoftwareStatement(), - OpenBankingConstants.JWT_BODY) - .toString(); - Map ssaAttributesMap = gson.fromJson(ssaBody, Map.class); - registrationRequest.setSsaParameters(ssaAttributesMap); - } - - String clientId = uriInfo.getPathParameters().getFirst("s"); - RegistrationUtils.validateRegistrationCreation(registrationRequest); - log.debug("Invoking specific validations"); - RegistrationValidator.getRegistrationValidator().validateUpdate(registrationRequest); - Map headers = getHeaders(request); - String accessToken = headers.get(RegistrationConstants.REGISTRATION_ACCESS_TOKEN); - - Map additionalAttributes = new HashMap<>(); - - try { - String tlsCert = new IdentityCommonHelper().encodeCertificateContent( - IdentityCommonUtil.getCertificateFromAttribute( - request.getAttribute(IdentityCommonConstants.JAVAX_SERVLET_REQUEST_CERTIFICATE))); - - // add TLS cert as additional attribute - additionalAttributes.put(IdentityCommonConstants.TLS_CERT, tlsCert); - } catch (CertificateEncodingException e) { - log.error("Certificate not valid", e); - } - - return registrationServiceHandler. - updateRegistration(registrationRequest, additionalAttributes, clientId, accessToken); - } catch (ParseException e) { - log.error("Error while parsing the softwareStatement", e); - } catch (DCRMException e) { - log.error("Error occurred while creating the Service provider", e); - if (e.getErrorCode().contains(RegistrationConstants.CLIENT_NOT_FOUND)) { - return Response.status(Response.Status.UNAUTHORIZED).build(); - } - } catch (IdentityApplicationManagementException e) { - log.error("Error occurred while retrieving the Service provider details", e); - } catch (DCRValidationException e) { - log.error("Error occurred while validating request", e); - return Response.status(Response.Status.BAD_REQUEST).entity(RegistrationUtils.getErrorDTO(e.getErrorCode(), - e.getErrorDescription())).build(); - } catch (net.minidev.json.parser.ParseException | IOException e) { - log.error("Error occurred while parsing the request", e); - } - return Response.status(Response.Status.INTERNAL_SERVER_ERROR).entity( - RegistrationUtils.getErrorDTO(RegistrationConstants.INTERNAL_SERVER_ERROR, - "Error occurred while processing the request")).build(); - - } - - @POST - @Path("/{s:.*}") - @Consumes({"application/json; charset=utf-8"}) - @Produces({"application/json; charset=utf-8"}) - public Response registerPost(@Context HttpServletRequest request, @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - - RegistrationValidator registrationValidator = RegistrationValidator.getRegistrationValidator(); - //invoke the configured registration VALIDATOR - if (log.isDebugEnabled()) { - log.debug("Invoking the configured registration validator:" + registrationValidator); - } - try { - JsonElement registrationRequestDetails = gson.toJsonTree(RegistrationUtils.getPayload(request)); - - RegistrationRequest registrationRequest = - gson.fromJson(registrationRequestDetails, RegistrationRequest.class); - - Map requestAttributes = (Map) - gson.fromJson(registrationRequestDetails, Map.class); - - Map additionalAttributes = new HashMap<>(); - - try { - String tlsCert = new IdentityCommonHelper().encodeCertificateContent( - IdentityCommonUtil.getCertificateFromAttribute( - request.getAttribute(IdentityCommonConstants.JAVAX_SERVLET_REQUEST_CERTIFICATE))); - - // add TLS cert as additional attribute - additionalAttributes.put(IdentityCommonConstants.TLS_CERT, tlsCert); - } catch (CertificateEncodingException e) { - log.error("Certificate not valid", e); - } - - registrationRequest.setRequestParameters(requestAttributes); - if (StringUtils.isNotEmpty(registrationRequest.getSoftwareStatement())) { - //decode SSA if provided in the registration request - String ssaBody = JWTUtils.decodeRequestJWT(registrationRequest.getSoftwareStatement(), - OpenBankingConstants.JWT_BODY) - .toString(); - Map ssaAttributesMap = gson.fromJson(ssaBody, Map.class); - registrationRequest.setSsaParameters(ssaAttributesMap); - } - - RegistrationUtils.validateRegistrationCreation(registrationRequest); - //do specific validations - registrationValidator.validatePost(registrationRequest); - return registrationServiceHandler.createRegistration(registrationRequest, additionalAttributes); - } catch (ParseException e) { - log.error("Error while parsing the softwareStatement", e); - } catch (DCRMException e) { - log.error("Error occurred while creating the Service provider", e); - if (DCRCommonConstants.DUPLICATE_APPLICATION_NAME.equalsIgnoreCase(e.getErrorCode())) { - return Response.status(Response.Status.BAD_REQUEST).entity(RegistrationUtils - .getErrorDTO(DCRCommonConstants.INVALID_META_DATA, e.getErrorDescription())) - .build(); - } - } catch (IdentityApplicationManagementException e) { - log.error("Error occurred while retrieving the Service provider details", e); - } catch (DCRValidationException e) { - log.error("Error occurred while validating request", e); - return Response.status(Response.Status.BAD_REQUEST).entity(RegistrationUtils - .getErrorDTO(e.getErrorCode(), e.getErrorDescription())) - .build(); - } catch (net.minidev.json.parser.ParseException | IOException e) { - log.error("Error occurred while parsing the request", e); - } - return Response.status(Response.Status.INTERNAL_SERVER_ERROR).entity( - RegistrationUtils.getErrorDTO(RegistrationConstants.INTERNAL_SERVER_ERROR, - "Error occurred while processing the request")).build(); - } - - /** - * Extract headers from a request object. - * - * @param request The request object - * @return Map of header key value pairs - */ - public static Map getHeaders(HttpServletRequest request) { - Map headers = new HashMap<>(); - Enumeration headerNames = request.getHeaderNames(); - while (headerNames.hasMoreElements()) { - String headerName = headerNames.nextElement(); - headers.put(headerName, request.getHeader(headerName)); - } - return headers; - } - -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/model/DCRRequestData.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/model/DCRRequestData.java deleted file mode 100644 index 7efda05b..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/model/DCRRequestData.java +++ /dev/null @@ -1,144 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.endpoint.impl.model; - -import com.wso2.openbanking.accelerator.identity.dcr.endpoint.impl.util.ResponseStatus; - -import java.util.Map; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; - -/** - * Model class for DCR request data. - */ -public class DCRRequestData { - - public Map getHeaders() { - - return headers; - } - - public void setHeaders(Map headers) { - - this.headers = headers; - } - - public Object getPayload() { - - return payload; - } - - public void setPayload(Object payload) { - - this.payload = payload; - } - - public Map getQueryParams() { - - return queryParams; - } - - public void setQueryParams(Map queryParams) { - - this.queryParams = queryParams; - } - - public String getRequestPath() { - - return requestPath; - } - - public void setRequestPath(String requestPath) { - - this.requestPath = requestPath; - } - - public String getClientId() { - - return clientId; - } - - public void setClientId(String clientId) { - - this.clientId = clientId; - } - - public HttpServletRequest getRequest() { - - return request; - } - - public void setRequest(HttpServletRequest request) { - - this.request = request; - } - - public HttpServletResponse getResponse() { - - return response; - } - - public void setResponse(HttpServletResponse response) { - - this.response = response; - } - - public ResponseStatus getResponseStatus() { - - return responseStatus; - } - - public void setResponseStatus(ResponseStatus responseStatus) { - - this.responseStatus = responseStatus; - } - - public Object getResponsePayload() { - - return responsePayload; - } - - public void setResponsePayload(Object responsePayload) { - - this.responsePayload = responsePayload; - } - - private Map headers; - //Payload can either be a JSONObject or a JSONArray - private Object payload; - private Map queryParams; - private String requestPath; - private String clientId; - private HttpServletRequest request; - private HttpServletResponse response; - private ResponseStatus responseStatus; - private Object responsePayload; - - public DCRRequestData(Map headers, Object payload, Map queryParams, - String requestPath, HttpServletRequest request, HttpServletResponse response) { - - this.headers = headers; - this.payload = payload; - this.queryParams = queryParams; - this.requestPath = requestPath; - this.request = request; - this.response = response; - } - -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/service/RegistrationServiceHandler.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/service/RegistrationServiceHandler.java deleted file mode 100644 index 6fdaa3f4..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/service/RegistrationServiceHandler.java +++ /dev/null @@ -1,267 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.endpoint.impl.service; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.identity.dcr.endpoint.impl.RegistrationConstants; -import com.wso2.openbanking.accelerator.identity.dcr.endpoint.impl.util.RegistrationUtils; -import com.wso2.openbanking.accelerator.identity.dcr.exception.DCRValidationException; -import com.wso2.openbanking.accelerator.identity.dcr.model.RegistrationRequest; -import com.wso2.openbanking.accelerator.identity.dcr.validation.DCRCommonConstants; -import com.wso2.openbanking.accelerator.identity.dcr.validation.RegistrationValidator; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.context.PrivilegedCarbonContext; -import org.wso2.carbon.identity.application.common.IdentityApplicationManagementException; -import org.wso2.carbon.identity.application.common.model.ServiceProvider; -import org.wso2.carbon.identity.application.common.model.ServiceProviderProperty; -import org.wso2.carbon.identity.application.mgt.ApplicationManagementService; -import org.wso2.carbon.identity.oauth.dcr.bean.Application; -import org.wso2.carbon.identity.oauth.dcr.exception.DCRMException; -import org.wso2.carbon.identity.oauth.dcr.service.DCRMService; - -import java.text.ParseException; -import java.time.Instant; -import java.util.Arrays; -import java.util.List; -import java.util.Map; - -import javax.ws.rs.core.Response; - -/** - * Service class to invoke spec specific validators and manage storing, retrieving, updating and - * deleting registrations. - */ -public class RegistrationServiceHandler { - - private static final Log log = LogFactory.getLog(RegistrationServiceHandler.class); - private DCRMService oAuth2DCRMService; - private OpenBankingConfigurationService openBankingConfigurationService; - - public Response createRegistration(RegistrationRequest registrationRequest, - Map additionalAttributes) - throws DCRMException, IdentityApplicationManagementException, IllegalArgumentException, ParseException { - - DCRMService dcrmService = getDCRServiceInstance(); - OpenBankingConfigurationService openBankingConfigurationService = getOBConfigService(); - RegistrationValidator registrationValidator = RegistrationValidator.getRegistrationValidator(); - String tenantDomain = PrivilegedCarbonContext.getThreadLocalCarbonContext().getTenantDomain(); - String userName = PrivilegedCarbonContext.getThreadLocalCarbonContext().getUsername(); - boolean useSoftwareIdAsAppName = false; - String jwksEndpointName = ""; - if (openBankingConfigurationService != null) { - Map configurations = openBankingConfigurationService.getConfigurations(); - useSoftwareIdAsAppName = Boolean.parseBoolean(configurations - .get(OpenBankingConstants.DCR_USE_SOFTWAREID_AS_APPNAME).toString()); - if (configurations.containsKey(OpenBankingConstants.DCR_JWKS_NAME)) { - jwksEndpointName = configurations - .get(OpenBankingConstants.DCR_JWKS_NAME).toString(); - } - } - String applicationName = RegistrationUtils.getApplicationName(registrationRequest, useSoftwareIdAsAppName); - Application application = dcrmService.registerApplication(RegistrationUtils - .getApplicationRegistrationRequest(registrationRequest, applicationName)); - if (log.isDebugEnabled()) { - log.debug("Created application with name :" + application.getClientName()); - } - ApplicationManagementService applicationManagementService = ApplicationManagementService.getInstance(); - ServiceProvider serviceProvider = applicationManagementService - .getServiceProvider(application.getClientName(), tenantDomain); - - //get JWKS URI from the request - String jwksUri = RegistrationUtils.getJwksUriFromRequest(registrationRequest, jwksEndpointName); - serviceProvider.setJwksUri(jwksUri); - - Long clientIdIssuedTime = Instant.now().getEpochSecond(); - //store the client details as SP meta data - Map registrationRequestData = RegistrationUtils - .getAlteredApplicationAttributes(registrationRequest); - registrationRequestData.put(RegistrationConstants.CLIENT_ID_ISSUED_AT, clientIdIssuedTime.toString()); - // Adding SP property to identify create request. Will be removed when setting up authenticators. - registrationRequestData.put("AppCreateRequest", "true"); - List spMetaData = RegistrationUtils.getServiceProviderPropertyList - (registrationRequestData); - serviceProvider.setSpProperties(spMetaData.toArray(new ServiceProviderProperty[0])); - applicationManagementService.updateApplication(serviceProvider, tenantDomain, userName); - - if (log.isDebugEnabled()) { - log.debug("Updated Service Provider " + serviceProvider.getApplicationName() + " with the client data"); - } - Map registrationData = registrationRequest.getRequestParameters(); - registrationData.put(RegistrationConstants.CLIENT_ID, application.getClientId()); - registrationData.put(RegistrationConstants.CLIENT_ID_ISSUED_AT, clientIdIssuedTime.toString()); - if (registrationRequest.getSsaParameters() != null) { - registrationData.putAll(registrationRequest.getSsaParameters()); - } - registrationData.putAll(additionalAttributes); - String registrationResponse = registrationValidator.getRegistrationResponse(registrationData); - return Response.status(Response.Status.CREATED).entity(registrationResponse).build(); - } - - public Response retrieveRegistration(Map additionalAttributes, String clientId, String accessToken) - throws DCRMException, IdentityApplicationManagementException { - - DCRMService dcrmService = getDCRServiceInstance(); - Application application = dcrmService.getApplication(clientId); - - if (log.isDebugEnabled()) { - log.debug("Retrieved Application with name " + application.getClientName()); - } - String tenantDomain = PrivilegedCarbonContext.getThreadLocalCarbonContext().getTenantDomain(); - - ApplicationManagementService applicationManagementService = ApplicationManagementService.getInstance(); - ServiceProvider serviceProvider = applicationManagementService - .getServiceProvider(application.getClientName(), tenantDomain); - ServiceProviderProperty[] serviceProviderProperties = serviceProvider.getSpProperties(); - - if (log.isDebugEnabled()) { - log.debug("Retrieved client meta data for application " + application.getClientName()); - } - - List spPropertyList = Arrays.asList(serviceProviderProperties); - Map spMetaData = RegistrationUtils.getSpMetaDataMap(spPropertyList); - spMetaData.put(RegistrationConstants.CLIENT_ID, application.getClientId()); - spMetaData.put(RegistrationConstants.REGISTRATION_ACCESS_TOKEN, accessToken); - spMetaData.putAll(additionalAttributes); - - String registrationResponseJson = RegistrationValidator.getRegistrationValidator() - .getRegistrationResponse(spMetaData); - return Response.status(Response.Status.OK).entity(registrationResponseJson).build(); - } - - public Response updateRegistration(RegistrationRequest request, Map additionalAttributes, - String clientId, String accessToken) - throws DCRMException, IdentityApplicationManagementException, DCRValidationException, ParseException { - - DCRMService dcrmService = getDCRServiceInstance(); - OpenBankingConfigurationService openBankingConfigurationService = getOBConfigService(); - boolean useSoftwareIdAsAppName = false; - String jwksEndpointName = ""; - if (openBankingConfigurationService != null) { - Map configurations = openBankingConfigurationService.getConfigurations(); - useSoftwareIdAsAppName = Boolean.parseBoolean(configurations - .get(OpenBankingConstants.DCR_USE_SOFTWAREID_AS_APPNAME).toString()); - if (configurations.containsKey(OpenBankingConstants.DCR_JWKS_NAME)) { - jwksEndpointName = configurations - .get(OpenBankingConstants.DCR_JWKS_NAME).toString(); - } - } - Application applicationToUpdate = dcrmService.getApplication(clientId); - String applicationNameInRequest; - if (useSoftwareIdAsAppName) { - applicationNameInRequest = (request.getSoftwareStatement() != null) ? - request.getSoftwareStatementBody().getSoftwareId() : - request.getSoftwareId(); - } else { - applicationNameInRequest = request.getSoftwareStatementBody().getClientName(); - } - if (!applicationToUpdate.getClientName().equals(applicationNameInRequest)) { - throw new DCRValidationException(DCRCommonConstants.INVALID_META_DATA, "Invalid application name"); - } - String applicationName = RegistrationUtils.getApplicationName(request, useSoftwareIdAsAppName); - Application application = dcrmService.updateApplication - (RegistrationUtils.getApplicationUpdateRequest(request, applicationName), clientId); - if (log.isDebugEnabled()) { - log.debug("Updated Application with name " + application.getClientName()); - } - String tenantDomain = PrivilegedCarbonContext.getThreadLocalCarbonContext().getTenantDomain(); - String userName = PrivilegedCarbonContext.getThreadLocalCarbonContext().getUsername(); - - //retrieve stored client meta data - ApplicationManagementService applicationManagementService = ApplicationManagementService.getInstance(); - ServiceProvider serviceProvider = applicationManagementService - .getServiceProvider(application.getClientName(), tenantDomain); - - //get JWKS URI from the request - String jwksUri = RegistrationUtils.getJwksUriFromRequest(request, jwksEndpointName); - serviceProvider.setJwksUri(jwksUri); - - ServiceProviderProperty[] serviceProviderProperties = serviceProvider.getSpProperties(); - if (log.isDebugEnabled()) { - log.debug("Retrieved client meta data for application " + application.getClientName()); - } - List spPropertyList = Arrays.asList(serviceProviderProperties); - Map storedSPMetaData = RegistrationUtils.getSpMetaDataMap(spPropertyList); - - String clientIdIssuedAt = ""; - if (storedSPMetaData.containsKey(RegistrationConstants.CLIENT_ID_ISSUED_AT)) { - clientIdIssuedAt = storedSPMetaData.get(RegistrationConstants.CLIENT_ID_ISSUED_AT).toString(); - } - - //update Service provider with new client data - Map updateRequestData = RegistrationUtils.getAlteredApplicationAttributes(request); - Map updateRegistrationData = request.getRequestParameters(); - if (request.getSsaParameters() != null) { - updateRegistrationData.putAll(request.getSsaParameters()); - } - updateRequestData.put(RegistrationConstants.CLIENT_ID_ISSUED_AT, clientIdIssuedAt); - // Adding SP property to identify update request. Will be removed when updating authenticators. - updateRequestData.put("AppCreateRequest", "false"); - List spMetaData = RegistrationUtils.getServiceProviderPropertyList(updateRequestData); - serviceProvider.setSpProperties(spMetaData.toArray(new ServiceProviderProperty[0])); - applicationManagementService.updateApplication(serviceProvider, tenantDomain, userName); - - if (log.isDebugEnabled()) { - log.debug("Updated Service Provider meta data for application " + application.getClientName()); - } - - updateRegistrationData.put(RegistrationConstants.CLIENT_ID, application.getClientId()); - updateRegistrationData.put(RegistrationConstants.CLIENT_ID_ISSUED_AT, clientIdIssuedAt); - updateRegistrationData.put(RegistrationConstants.REGISTRATION_ACCESS_TOKEN, accessToken); - updateRegistrationData.putAll(additionalAttributes); - String registrationResponse = RegistrationValidator.getRegistrationValidator() - .getRegistrationResponse(updateRegistrationData); - return Response.status(Response.Status.OK).entity(registrationResponse).build(); - } - - public Response deleteRegistration(String clientId) throws DCRMException { - - DCRMService dcrmService = getDCRServiceInstance(); - dcrmService.deleteApplication(clientId); - if (log.isDebugEnabled()) { - log.debug("Deleted application with client Id :" + clientId); - } - return Response.status(Response.Status.NO_CONTENT).build(); - } - - public DCRMService getDCRServiceInstance() { - - if (this.oAuth2DCRMService == null) { - DCRMService oAuth2DCRMService = (DCRMService) PrivilegedCarbonContext. - getThreadLocalCarbonContext().getOSGiService(DCRMService.class, null); - if (oAuth2DCRMService != null) { - this.oAuth2DCRMService = oAuth2DCRMService; - } - } - return this.oAuth2DCRMService; - } - - public OpenBankingConfigurationService getOBConfigService() { - - if (this.openBankingConfigurationService == null) { - OpenBankingConfigurationService openBankingConfigurationService = - (OpenBankingConfigurationService) PrivilegedCarbonContext.getThreadLocalCarbonContext() - .getOSGiService(OpenBankingConfigurationService.class, null); - if (openBankingConfigurationService != null) { - this.openBankingConfigurationService = openBankingConfigurationService; - } - } - return this.openBankingConfigurationService; - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/util/RegistrationUtils.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/util/RegistrationUtils.java deleted file mode 100644 index eef4a113..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/util/RegistrationUtils.java +++ /dev/null @@ -1,359 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.endpoint.impl.util; - -import com.google.gson.Gson; -import com.google.gson.JsonObject; -import com.google.gson.JsonParser; -import com.wso2.openbanking.accelerator.common.util.JWTUtils; -import com.wso2.openbanking.accelerator.identity.dcr.endpoint.impl.dto.RegistrationErrorDTO; -import com.wso2.openbanking.accelerator.identity.dcr.exception.DCRValidationException; -import com.wso2.openbanking.accelerator.identity.dcr.model.RegistrationError; -import com.wso2.openbanking.accelerator.identity.dcr.model.RegistrationRequest; -import com.wso2.openbanking.accelerator.identity.dcr.utils.ValidatorUtils; -import com.wso2.openbanking.accelerator.identity.dcr.validation.DCRCommonConstants; -import com.wso2.openbanking.accelerator.identity.dcr.validation.RegistrationValidator; -import net.minidev.json.JSONArray; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import org.apache.commons.io.IOUtils; -import org.apache.commons.lang.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.wso2.carbon.identity.application.common.model.ServiceProviderProperty; -import org.wso2.carbon.identity.oauth.dcr.bean.ApplicationRegistrationRequest; -import org.wso2.carbon.identity.oauth.dcr.bean.ApplicationUpdateRequest; - -import java.io.IOException; -import java.text.ParseException; -import java.util.ArrayList; -import java.util.Enumeration; -import java.util.HashMap; -import java.util.Iterator; -import java.util.List; -import java.util.Map; -import java.util.stream.Collectors; -import java.util.stream.Stream; - -import javax.servlet.http.HttpServletRequest; - -/** - * Util class which includes helper methods required for DCR. - */ -public class RegistrationUtils { - - private static final Log log = LogFactory.getLog(RegistrationUtils.class); - - private static final String DISALLOWED_CHARS_PATTERN = "([~!#$;%^&*+={}\\s\\|\\\\<>\\\"\\'\\/,\\]\\[\\(\\)])"; - private static final String SUBSTITUTE_STRING = "_"; - private static final int ABBREVIATED_STRING_LENGTH = 70; - private static Gson gson = new Gson(); - - /** - * this method invokes the configured registration VALIDATOR class - * by default the DefaultRegistrationValidatorImpl class will be configured. - * - * @param registrationRequest object containing the client registration request details - */ - public static void validateRegistrationCreation(RegistrationRequest registrationRequest) - throws ParseException, DCRValidationException { - - RegistrationValidator dcrRequestValidator; - dcrRequestValidator = RegistrationValidator.getRegistrationValidator(); - - if (StringUtils.isNotEmpty(registrationRequest.getSoftwareStatement())) { - // set the ssa payload according to the specification format - String decodedSSA = JWTUtils - .decodeRequestJWT(registrationRequest.getSoftwareStatement(), "body").toJSONString(); - dcrRequestValidator.setSoftwareStatementPayload(registrationRequest, decodedSSA); - } - - // do common validations - ValidatorUtils.getValidationViolations(registrationRequest); - - } - - public static RegistrationErrorDTO getErrorDTO(String errorCode, String errorMessage) { - - RegistrationErrorDTO registrationErrorDTO = new RegistrationErrorDTO(); - registrationErrorDTO.setError(errorCode); - registrationErrorDTO.setErrorDescription(errorMessage); - return registrationErrorDTO; - } - - public static RegistrationError getRegistrationError(String errorCode, String errorMessage) { - - RegistrationError registrationError = new RegistrationError(); - registrationError.setErrorCode(errorCode); - registrationError.setErrorMessage(errorMessage); - return registrationError; - } - - public static ApplicationRegistrationRequest getApplicationRegistrationRequest( - RegistrationRequest registrationRequest, String applicationName) { - - ApplicationRegistrationRequest appRegistrationRequest = new ApplicationRegistrationRequest(); - appRegistrationRequest.setClientName(applicationName); - appRegistrationRequest.setGrantTypes(registrationRequest.getGrantTypes()); - - // Get the redirect URIs based on the presence of software statement - List redirectUris = StringUtils.isEmpty(registrationRequest.getSoftwareStatement()) - ? registrationRequest.getCallbackUris() - : registrationRequest.getSoftwareStatementBody().getCallbackUris(); - - appRegistrationRequest.setRedirectUris(redirectUris); - - return appRegistrationRequest; - } - - public static ApplicationUpdateRequest getApplicationUpdateRequest(RegistrationRequest registrationRequest, - String applicationName) { - - ApplicationUpdateRequest applicationUpdateRequest = new ApplicationUpdateRequest(); - applicationUpdateRequest.setClientName(applicationName); - applicationUpdateRequest.setGrantTypes(registrationRequest.getGrantTypes()); - - // Get the redirect URIs based on the presence of software statement - List redirectUris = StringUtils.isEmpty(registrationRequest.getSoftwareStatement()) - ? registrationRequest.getCallbackUris() - : registrationRequest.getSoftwareStatementBody().getCallbackUris(); - - applicationUpdateRequest.setRedirectUris(redirectUris); - - return applicationUpdateRequest; - } - /** - * Retrieves the application name from the registration request. - * - * @param request registration or update request - * @param useSoftwareIdAsAppName Indicates whether to use the software ID as the application name - * @return The application name - */ - public static String getApplicationName(RegistrationRequest request, boolean useSoftwareIdAsAppName) { - if (useSoftwareIdAsAppName) { - // If the request does not contain a software statement, get the software Id directly from the request - if (StringUtils.isEmpty(request.getSoftwareStatement())) { - return request.getSoftwareId(); - } - return request.getSoftwareStatementBody().getSoftwareId(); - } - return RegistrationUtils.getSafeApplicationName(request.getSoftwareStatementBody().getClientName()); - } - - /** - * Retrieves the JWKS URI from the registration request based on the presence of the software statement. - * - * @param registrationRequest registration or update request - * @param jwksEndpointName name used for the JWKS endpoint in the software statement - * @return JWKS URI. - */ - public static String getJwksUriFromRequest(RegistrationRequest registrationRequest, String jwksEndpointName) { - if (StringUtils.isEmpty(registrationRequest.getSoftwareStatement())) { - return registrationRequest.getJwksURI(); - } - if (StringUtils.isNotEmpty(jwksEndpointName)) { - return registrationRequest.getSsaParameters().get(jwksEndpointName).toString(); - } - return registrationRequest.getSoftwareStatementBody().getJwksURI(); - } - - public static ArrayList getServiceProviderPropertyList - (Map clientMetaData) { - - ArrayList spPropList = new ArrayList<>(); - for (Map.Entry entry : clientMetaData.entrySet()) { - String key = entry.getKey(); - String value = entry.getValue(); - ServiceProviderProperty serviceProviderproperty = new ServiceProviderProperty(); - if (value != null) { - serviceProviderproperty.setDisplayName(key); - serviceProviderproperty.setName(key); - serviceProviderproperty.setValue(value); - spPropList.add(serviceProviderproperty); - } - } - return spPropList; - - } - - public static Map getSpMetaDataMap(List spPropertyList) { - - Map spMetaDataMap = new HashMap<>(); - for (ServiceProviderProperty spProperty : spPropertyList) { - if (spProperty.getValue().contains(DCRCommonConstants.ARRAY_ELEMENT_SEPERATOR)) { - List metaDataList = Stream.of(spProperty.getValue() - .split(DCRCommonConstants.ARRAY_ELEMENT_SEPERATOR)) - .map(String::trim) - .collect(Collectors.toList()); - getJsonElementListFromString(metaDataList); - spMetaDataMap.put(spProperty.getName(), metaDataList); - } else if (spProperty.getValue().contains("{")) { - JsonParser jsonParser = new JsonParser(); - JsonObject jsonObject = ((JsonObject) jsonParser - .parse(spProperty.getValue())); - spMetaDataMap.put(spProperty.getName(), jsonObject); - } else { - spMetaDataMap.put(spProperty.getName(), spProperty.getValue()); - } - - } - return spMetaDataMap; - } - - public static String getSafeApplicationName(String applicationName) { - - if (StringUtils.isEmpty(applicationName)) { - throw new IllegalArgumentException("Application name should be a valid string"); - } - - String sanitizedInput = applicationName.trim().replaceAll(DISALLOWED_CHARS_PATTERN, SUBSTITUTE_STRING); - return StringUtils.abbreviate(sanitizedInput, ABBREVIATED_STRING_LENGTH); - - } - - public static Map getAlteredApplicationAttributes(RegistrationRequest registrationRequest) { - - Map alteredAppAttributeMap = new HashMap<>(); - addAttributes(registrationRequest.getRequestParameters(), alteredAppAttributeMap); - - if (StringUtils.isNotEmpty(registrationRequest.getSoftwareStatement())) { - //add ssa attributes - addAttributes(registrationRequest.getSsaParameters(), alteredAppAttributeMap); - //add ssa issuer - alteredAppAttributeMap.put("ssaIssuer", registrationRequest.getSsaParameters().get("iss").toString()); - } - - return alteredAppAttributeMap; - } - - public static void addAttributes(Map requestAttributes, - Map alteredAttributes) { - - String alteredValue = ""; - for (Map.Entry entry : requestAttributes.entrySet()) { - alteredValue = ""; - if (entry.getValue() instanceof ArrayList) { - - ArrayList list = ((ArrayList) entry.getValue()); - Object lastListElement = new Object(); - if (list.size() > 0) { - lastListElement = list.get(list.size() - 1); - } - getJsonElementList(list); - if (list.size() == 1) { - alteredValue = list.get(0).toString().concat(DCRCommonConstants.ARRAY_ELEMENT_SEPERATOR); - alteredAttributes.put(entry.getKey().toString(), alteredValue); - } else if (list.size() > 0) { - for (Object listElement : list) { - if (!lastListElement.equals(listElement)) { - alteredValue = alteredValue.concat( - listElement.toString().concat(DCRCommonConstants.ARRAY_ELEMENT_SEPERATOR)); - } else { - alteredValue = alteredValue.concat(lastListElement.toString()); - } - } - alteredAttributes.put(entry.getKey().toString(), alteredValue); - } - } else if (entry.getValue() instanceof Map) { - alteredAttributes.put(entry.getKey().toString(), gson.toJson(entry.getValue())); - } else { - //remove unnecessary inverted commas. - if (entry.getValue() != null) { - // This is to handle optional nullable params. - // Ex: "software_on_behalf_of_org":null - alteredAttributes.put(entry.getKey().toString(), entry.getValue().toString()); - } - } - } - } - - public static Map getRegistrationDetailsForResponse(RegistrationRequest registrationRequest) { - - String registrationRequestJson = gson.toJson(registrationRequest); - return gson.fromJson(registrationRequestJson, Map.class); - } - - /** - * Extract headers from a request object. - * - * @param request The request object - * @return Map of header key value pairs - */ - public static Map getHeaders(HttpServletRequest request) { - - Map headers = new HashMap<>(); - Enumeration headerNames = request.getHeaderNames(); - while (headerNames.hasMoreElements()) { - String headerName = headerNames.nextElement(); - headers.put(headerName, request.getHeader(headerName)); - } - return headers; - } - - /** - * Util method to extract the payload from a HTTP request object. Can be JSONObject or JSONArray - * - * @param request The HTTP request object - * @return Object payload can be either an instance of JSONObject or JSONArray only. Can be a ConsentException if - * is and error scenario. Error is returned instead of throwing since the error response should be handled by the - * toolkit is the manage scenario. - */ - public static Object getPayload(HttpServletRequest request) throws IOException, - net.minidev.json.parser.ParseException { - - Object payload = new JSONParser(JSONParser.MODE_PERMISSIVE) - .parse(IOUtils.toString(request.getInputStream())); - if (payload == null) { - log.debug("Payload is empty. Returning null"); - return null; - } - if (!(payload instanceof JSONObject || payload instanceof JSONArray)) { - //Not throwing error since error should be formatted by manage toolkit - log.error("Payload is not a JSON. Returning null"); - return null; - } - return payload; - } - - /** - * check whether the elemet is a json and convert to a json. - * - * @param metaDataList meta data property list - */ - public static void getJsonElementList(List metaDataList) { - - for (Iterator iterator = metaDataList.iterator(); iterator.hasNext(); ) { - Object element = iterator.next(); - if (element.toString().contains("{")) { - //Object elementToRemove = element; - metaDataList.set(metaDataList.indexOf(element), gson.toJson(element)); - } - } - } - - public static void getJsonElementListFromString(List metaDataList) { - - for (Iterator iterator = metaDataList.iterator(); iterator.hasNext(); ) { - Object element = iterator.next(); - if (element.toString().contains("{")) { - metaDataList.set(metaDataList.indexOf(element), - new JsonParser().parse(element.toString()).getAsJsonObject()); - } - } - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/util/ResponseStatus.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/util/ResponseStatus.java deleted file mode 100644 index ab002635..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/java/com/wso2/openbanking/accelerator/identity/dcr/endpoint/impl/util/ResponseStatus.java +++ /dev/null @@ -1,231 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.identity.dcr.endpoint.impl.util; - -/** - * Enum of the supported response status in accelerator. - */ -public enum ResponseStatus { - - /** - * 200 OK, see .... - */ - OK(200, "OK"), - /** - * 201 Created, see .... - */ - CREATED(201, "Created"), - /** - * 202 Accepted, see .... - */ - ACCEPTED(202, "Accepted"), - /** - * 204 No Content, see .... - */ - NO_CONTENT(204, "No Content"), - /** - * 205 Reset Content, see .... - * - * @since 2.0 - */ - RESET_CONTENT(205, "Reset Content"), - /** - * 206 Reset Content, see .... - * - * @since 2.0 - */ - PARTIAL_CONTENT(206, "Partial Content"), - /** - * 301 Moved Permanently, - * see .... - */ - MOVED_PERMANENTLY(301, "Moved Permanently"), - /** - * 302 Found, see .... - * - * @since 2.0 - */ - FOUND(302, "Found"), - /** - * 303 See Other, see .... - */ - SEE_OTHER(303, "See Other"), - /** - * 304 Not Modified, see .... - */ - NOT_MODIFIED(304, "Not Modified"), - /** - * 305 Use Proxy, see .... - * - * @since 2.0 - */ - USE_PROXY(305, "Use Proxy"), - /** - * 307 Temporary Redirect, see .... - */ - TEMPORARY_REDIRECT(307, "Temporary Redirect"), - /** - * 400 Bad Request, see .... - */ - BAD_REQUEST(400, "Bad Request"), - /** - * 401 Unauthorized, see .... - */ - UNAUTHORIZED(401, "Unauthorized"), - /** - * 402 Payment Required, see .... - * - * @since 2.0 - */ - PAYMENT_REQUIRED(402, "Payment Required"), - /** - * 403 Forbidden, see .... - */ - FORBIDDEN(403, "Forbidden"), - /** - * 404 Not Found, see .... - */ - NOT_FOUND(404, "Not Found"), - /** - * 405 Method Not Allowed, see .... - * - * @since 2.0 - */ - METHOD_NOT_ALLOWED(405, "Method Not Allowed"), - /** - * 406 Not Acceptable, see .... - */ - NOT_ACCEPTABLE(406, "Not Acceptable"), - /** - * 409 Conflict, see .... - */ - CONFLICT(409, "Conflict"), - /** - * 410 Gone, see .... - */ - GONE(410, "Gone"), - /** - * 411 Length Required, see .... - * - * @since 2.0 - */ - LENGTH_REQUIRED(411, "Length Required"), - /** - * 412 Precondition Failed, see .... - */ - PRECONDITION_FAILED(412, "Precondition Failed"), - /** - * 413 Request Entity Too Large, see HTTP/1.1 documentation. - * - * @since 2.0 - */ - REQUEST_ENTITY_TOO_LARGE(413, "Request Entity Too Large"), - /** - * 414 Request-URI Too Long, - * see .... - * - * @since 2.0 - */ - REQUEST_URI_TOO_LONG(414, "Request-URI Too Long"), - /** - * 415 Unsupported Media Type, see HTTP/1.1 documentation. - */ - UNSUPPORTED_MEDIA_TYPE(415, "Unsupported Media Type"), - /** - * 416 Requested Range Not Satisfiable, see HTTP/1.1 documentation. - * - * @since 2.0 - */ - REQUESTED_RANGE_NOT_SATISFIABLE(416, "Requested Range Not Satisfiable"), - /** - * 417 Expectation Failed, see .... - * - * @since 2.0 - */ - EXPECTATION_FAILED(417, "Expectation Failed"), - /** - * 500 Internal Server Error, - * see .... - */ - INTERNAL_SERVER_ERROR(500, "Internal Server Error"), - /** - * 501 Not Implemented, see .... - * - * @since 2.0 - */ - NOT_IMPLEMENTED(501, "Not Implemented"), - /** - * 503 Service Unavailable, see .... - */ - SERVICE_UNAVAILABLE(503, "Service Unavailable"); - - private final int code; - private final String reason; - - ResponseStatus(final int statusCode, final String reasonPhrase) { - this.code = statusCode; - this.reason = reasonPhrase; - } - - /** - * Get the associated status code. - * - * @return the status code. - */ - public int getStatusCode() { - return code; - } - - /** - * Get the reason phrase. - * - * @return the reason phrase. - */ - public String getReasonPhrase() { - return toString(); - } - - /** - * Get the reason phrase. - * - * @return the reason phrase. - */ - @Override - public String toString() { - return reason; - } - - /** - * Convert a numerical status code into the corresponding Status. - * - * @param statusCode the numerical status code. - * @return the matching Status or null is no matching Status is defined. - */ - public static ResponseStatus fromStatusCode(final int statusCode) { - for (ResponseStatus s : ResponseStatus.values()) { - if (s.code == statusCode) { - return s; - } - } - return null; - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/resources/findbugs-include.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/resources/findbugs-include.xml deleted file mode 100644 index 8932a22e..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/webapp/META-INF/MANIFEST.mf b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/webapp/META-INF/MANIFEST.mf deleted file mode 100644 index 9d885be5..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/webapp/META-INF/MANIFEST.mf +++ /dev/null @@ -1 +0,0 @@ -Manifest-Version: 1.0 diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/webapp/META-INF/webapp-classloading.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/webapp/META-INF/webapp-classloading.xml deleted file mode 100644 index b212826c..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/webapp/META-INF/webapp-classloading.xml +++ /dev/null @@ -1,35 +0,0 @@ - - - - - - - - - false - - - Carbon,CXF3 - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/webapp/WEB-INF/beans.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/webapp/WEB-INF/beans.xml deleted file mode 100644 index 32e98763..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/webapp/WEB-INF/beans.xml +++ /dev/null @@ -1,32 +0,0 @@ - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/webapp/WEB-INF/web.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/webapp/WEB-INF/web.xml deleted file mode 100644 index 6eb2b9c3..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint/src/main/webapp/WEB-INF/web.xml +++ /dev/null @@ -1,69 +0,0 @@ - - - - - - WSO2 Open Banking - Dynamic Client Registration API - WSO2 Open Banking - Dynamic Client Registration API - - - contextConfigLocation - WEB-INF/beans.xml - - - - HttpHeaderSecurityFilter - org.apache.catalina.filters.HttpHeaderSecurityFilter - - hstsEnabled - false - - - - - HttpHeaderSecurityFilter - * - - - - - org.springframework.web.context.ContextLoaderListener - - - - - CXFServlet - - org.apache.cxf.transport.servlet.CXFServlet - - 1 - - - - CXFServlet - /* - - - - 60 - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/pom.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/pom.xml deleted file mode 100644 index c5cae4ff..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/pom.xml +++ /dev/null @@ -1,93 +0,0 @@ - - - - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../../pom.xml - - 4.0.0 - - open-banking-backend - war - WSO2 Open Banking - Demo Backend - - - - javax.ws.rs - jsr311-api - - - com.github.spotbugs - spotbugs-annotations - ${spotbugs.annotations.version} - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - - - - - - - maven-war-plugin - - WEB-INF/lib/*.jar - api#openbanking#backend - - ${maven-war-plugin.version} - - - org.apache.maven.plugins - maven-compiler-plugin - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - true - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/BankException.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/BankException.java deleted file mode 100644 index 96fc5f57..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/BankException.java +++ /dev/null @@ -1,37 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.demo.backend; - -/** - * BankException class. - */ -public class BankException extends Exception { - - public BankException(String msg) { - super(msg); - } - - public BankException(String msg, Throwable e) { - super(msg, e); - } - - public BankException(Throwable throwable) { - super(throwable); - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/BankExceptionHandler.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/BankExceptionHandler.java deleted file mode 100644 index f53f5612..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/BankExceptionHandler.java +++ /dev/null @@ -1,35 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.demo.backend; - -import javax.ws.rs.core.MediaType; -import javax.ws.rs.core.Response; -import javax.ws.rs.core.Response.Status; -import javax.ws.rs.ext.ExceptionMapper; - -/** - * BankExceptionHandler class. - */ -public class BankExceptionHandler implements ExceptionMapper { - - public Response toResponse(BankException exception) { - return Response.status(Status.BAD_REQUEST).entity(exception.getMessage()) - .type(MediaType.APPLICATION_JSON).build(); - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/services/AccountService.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/services/AccountService.java deleted file mode 100644 index 3ac2652d..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/services/AccountService.java +++ /dev/null @@ -1,203 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.demo.backend.services; - -import com.wso2.openbanking.accelerator.demo.backend.BankException; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; - -import java.util.UUID; - -import javax.ws.rs.GET; -import javax.ws.rs.HeaderParam; -import javax.ws.rs.Path; -import javax.ws.rs.PathParam; -import javax.ws.rs.Produces; -import javax.ws.rs.core.Response; - -/** - * AccountService class. - */ -@Path("/accountservice/") -public class AccountService { - - @SuppressFBWarnings("JAXRS_ENDPOINT") - // Suppressed content - Endpoint - // Suppression reason - False Positive : This endpoint is a demo endpoint that is not exposed in production - // Suppressed warning count - 1 - @GET - @Path("/accounts/{AccountId}") - @Produces("application/json; charset=utf-8") - public Response getOneAccount(@PathParam("AccountId") String accountId, - @HeaderParam("x-fapi-interaction-id") String xFapiInteractionId, - @HeaderParam("Account-Request-Information") String accountRequestInfo) - throws BankException { - - String response = "{\n" + - " \"Data\": {\n" + - " \"Account\": [\n" + - " {\n" + - " \"AccountId\": \"" + accountId + "\",\n" + - " \"Status\": \"Enabled\",\n" + - " \"StatusUpdateDateTime\": \"2020-04-16T06:06:06+00:00\",\n" + - " \"Currency\": \"GBP\",\n" + - " \"AccountType\": \"Personal\",\n" + - " \"AccountSubType\": \"CurrentAccount\",\n" + - " \"Nickname\": \"Bills\",\n" + - " \"Account\": [{\n" + - " \"SchemeName\": \"SortCodeAccountNumber\",\n" + - " \"Identification\": \"" + accountId + "\",\n" + - " \"Name\": \"Mr Kevin\",\n" + - " \"SecondaryIdentification\": \"00021\"\n" + - " }]\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Links\": {\n" + - " \"Self\": \"https://api.alphabank.com/open-banking/v3.0/accounts/" + accountId + - "\"\n" + - " },\n" + - " \"Meta\": {\n" + - " \"TotalPages\": 1\n" + - " }\n" + - "}"; - - if (xFapiInteractionId == null) { - xFapiInteractionId = UUID.randomUUID().toString(); - } - return Response.status(200).entity(response) - .header("x-fapi-interaction-id", xFapiInteractionId).build(); - - } - - @SuppressFBWarnings("JAXRS_ENDPOINT") - // Suppressed content - Endpoint - // Suppression reason - False Positive : This endpoint is a demo endpoint that is not exposed in production - // Suppressed warning count - 1 - @GET - @Path("/accounts/{AccountId}/transactions") - @Produces("application/json; charset=utf-8") - public Response getAccountTransactions(@PathParam("AccountId") String accountId, - @HeaderParam("x-fapi-interaction-id") String xFapiInteractionId, - @HeaderParam("Account-Request-Information") String accountRequestInfo) - throws BankException { - - String response = "{\n" + - " \"Data\": {\n" + - " \"Transaction\": [\n" + - " {\n" + - " \"AccountId\": \"" + accountId + "\",\n" + - " \"TransactionId\": \"123\",\n" + - " \"TransactionReference\": \"Ref 1\",\n" + - " \"Amount\": {\n" + - " \"Amount\": \"10.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"CreditDebitIndicator\": \"" + "Credit" + "\",\n" + - " \"Status\": \"Booked\",\n" + - " \"BookingDateTime\": \"2017-04-05T10:43:07+00:00\",\n" + - " \"ValueDateTime\": \"2017-04-05T10:45:22+00:00\",\n" + - " \"TransactionInformation\": \"Cash from Aubrey\",\n" + - " \"BankTransactionCode\": {\n" + - " \"Code\": \"str\",\n" + - " \"SubCode\": \"str\"\n" + - " },\n" + - " \"ProprietaryBankTransactionCode\": {\n" + - " \"Code\": \"Transfer\",\n" + - " \"Issuer\": \"AlphaBank\"\n" + - " },\n" + - " \"Balance\": {\n" + - " \"Amount\": {\n" + - " \"Amount\": \"230.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"CreditDebitIndicator\": \"Credit\",\n" + - " \"Type\": \"InterimBooked\"\n" + - " }\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Links\": {\n" + - " \"Self\": \"https://api.alphabank.com/open-banking/v3.0/accounts/" + accountId + - "/transactions/\"\n" + - " },\n" + - " \"Meta\": {\n" + - " \"TotalPages\": 1,\n" + - " \"FirstAvailableDateTime\": \"2017-05-03T00:00:00+00:00\",\n" + - " \"LastAvailableDateTime\": \"2017-12-03T00:00:00+00:00\"\n" + - " }\n" + - "}"; - if (xFapiInteractionId == null) { - xFapiInteractionId = UUID.randomUUID().toString(); - } - return Response.status(200).entity(response) - .header("x-fapi-interaction-id", xFapiInteractionId).build(); - } - - @SuppressFBWarnings("JAXRS_ENDPOINT") - // Suppressed content - Endpoint - // Suppression reason - False Positive : This endpoint is a demo endpoint that is not exposed in production - // Suppressed warning count - 1 - @GET - @Path("/accounts/{AccountId}/balances") - @Produces("application/json; charset=utf-8") - public Response getAccountBalance(@PathParam("AccountId") String accountId, - @HeaderParam("x-fapi-interaction-id") String xFapiInteractionId, - @HeaderParam("Account-Request-Information") String accountRequestInfo) - throws BankException { - - String response = "{\n" + - " \"Data\": {\n" + - " \"Balance\": [\n" + - " {\n" + - " \"AccountId\": \"" + accountId + "\",\n" + - " \"Amount\": {\n" + - " \"Amount\": \"1230.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"CreditDebitIndicator\": \"Credit\",\n" + - " \"Type\": \"InterimAvailable\",\n" + - " \"DateTime\": \"2017-04-05T10:43:07+00:00\",\n" + - " \"CreditLine\": [\n" + - " {\n" + - " \"Included\": true,\n" + - " \"Amount\": {\n" + - " \"Amount\": \"1000.00\",\n" + - " \"Currency\": \"GBP\"\n" + - " },\n" + - " \"Type\": \"Pre-Agreed\"\n" + - " }\n" + - " ]\n" + - " }\n" + - " ]\n" + - " },\n" + - " \"Links\": {\n" + - " \"Self\": \"https://api.alphabank.com/open-banking/v3.0/accounts/" + accountId + - "/balances/\"\n" + - " },\n" + - " \"Meta\": {\n" + - " \"TotalPages\": 1\n" + - " }\n" + - "}"; - if (xFapiInteractionId == null) { - xFapiInteractionId = UUID.randomUUID().toString(); - } - return Response.status(200).entity(response) - .header("x-fapi-interaction-id", xFapiInteractionId).build(); - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/services/FundsConfirmationService.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/services/FundsConfirmationService.java deleted file mode 100644 index b1b540cd..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/services/FundsConfirmationService.java +++ /dev/null @@ -1,82 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.demo.backend.services; - -import com.wso2.openbanking.accelerator.demo.backend.BankException; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; - -import javax.ws.rs.HeaderParam; -import javax.ws.rs.POST; -import javax.ws.rs.Path; -import javax.ws.rs.Produces; -import javax.ws.rs.core.Response; - -/** - * FundsConfirmationService class. - */ -@Path("/fundsconfirmationservice/") -public class FundsConfirmationService { - - @SuppressFBWarnings("JAXRS_ENDPOINT") - // Suppressed content - JAXRS_ENDPOINT - // Suppression reason - False Positive : This endpoint is a demo endpoint that is not exposed in production - // Suppressed warning count - 1 - - @POST - @Path("/funds-confirmations") - @Produces("application/json; charset=utf-8") - public Response getAccountBalance(String requestString, - @HeaderParam("x-fapi-interaction-id") String xFapiInteractionId) - throws BankException { - - JSONObject request; - try { - JSONParser parser = new JSONParser(JSONParser.MODE_PERMISSIVE); - request = (JSONObject) parser.parse(requestString); - } catch (ParseException e) { - throw new BankException("Error in casting JSON body " + e); - } - - String consentId = ((JSONObject) request.get("Data")).getAsString("ConsentId"); - String response = "{\n" + - " \"Data\": {\n" + - " \"FundsConfirmationId\": \"836403\",\n" + - " \"ConsentId\": \"" + consentId + "\",\n" + - " \"CreationDateTime\": \"2017-06-02T00:00:00+00:00\",\n" + - " \"FundsAvailable\": true,\n" + - " \"Reference\": \"Purchase02\",\n" + - " \"InstructedAmount\": {\n" + - " \"Amount\": \"20.00\",\n" + - " \"Currency\": \"USD\"\n" + - " }\n" + - " },\n" + - " \"Links\": {\n" + - " \"Self\": \"https://api.alphabank.com/open-banking/v3.0/funds-confirmations/836403\"\n" + - " },\n" + - " \"Meta\": {\n" + - " }\n" + - "}"; - return Response.status(201).entity(response) - .header("x-fapi-interaction-id", xFapiInteractionId).build(); - } -} - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/services/PaymentService.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/services/PaymentService.java deleted file mode 100644 index ffc077fc..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/services/PaymentService.java +++ /dev/null @@ -1,277 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.demo.backend.services; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.demo.backend.BankException; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.commons.lang3.StringUtils; - -import java.nio.charset.StandardCharsets; -import java.text.DateFormat; -import java.text.SimpleDateFormat; -import java.time.Instant; -import java.time.LocalDate; -import java.time.OffsetDateTime; -import java.time.OffsetTime; -import java.time.format.DateTimeFormatter; -import java.util.Base64; -import java.util.Date; -import java.util.HashMap; -import java.util.LinkedList; -import java.util.Map; -import java.util.Queue; -import java.util.Random; -import java.util.UUID; - -import javax.ws.rs.GET; -import javax.ws.rs.HeaderParam; -import javax.ws.rs.POST; -import javax.ws.rs.Path; -import javax.ws.rs.PathParam; -import javax.ws.rs.Produces; -import javax.ws.rs.core.Response; - -/** - * Payments Service class. - */ -@Path("/paymentservice/") -public class PaymentService { - - public static final String EXPECTED_EXECUTION_TIME = "ExpectedExecutionDateTime"; - public static final String EXPECTED_SETTLEMENT_TIME = "ExpectedSettlementDateTime"; - private static final Map domesticPayments = new HashMap<>(); - private static final int MAX_LIMIT = 500; - private static final Queue domesticPaymentsIdQueue = new LinkedList<>(); - - @SuppressFBWarnings("JAXRS_ENDPOINT") - // Suppressed content - Endpoint - // Suppression reason - False Positive : This endpoint is a demo endpoint that is not exposed in production - // Suppressed warning count - 1 - @GET - @Path("/payment-consents/{ConsentId}/funds-confirmation") - @Produces("application/json; charset=utf-8") - public Response getPaymentTypeFundsConfirmation(@PathParam("ConsentId") String paymentId) { - - Instant currentDate = Instant.now(); - - String response = "{\n" + - " \"Data\": {\n" + - " \"FundsAvailableResult\": {\n" + - " \"FundsAvailableDateTime\": \"" + currentDate.toString() + "\",\n" + - " \"FundsAvailable\": true\n" + - " }\n" + - " },\n" + - " \"Links\": {\n" + - " \"Self\": \"/pisp/payments/" + paymentId + "/funds-confirmation\"\n" + - " },\n" + - " \"Meta\": {}\n" + - "}"; - - return Response.status(200).entity(response) - .header("x-fapi-interaction-id", UUID.randomUUID().toString()) - .build(); - } - - @SuppressFBWarnings("JAXRS_ENDPOINT") - // Suppressed content - Endpoint - // Suppression reason - False Positive : This endpoint is a demo endpoint that is not exposed in production - // Suppressed warning count - 1 - - @POST - @Path("/payments") - @Produces("application/json; charset=utf-8") - public Response paymentSubmission(String requestString, @HeaderParam("x-fapi-interaction-id") String fid, - @HeaderParam("Account-Request-Information") String accountRequestInfo) - throws BankException { - - JSONObject jsonObject; - JSONObject accountRequestInformation; - - try { - accountRequestInformation = getRequest(accountRequestInfo); - JSONParser parser = new JSONParser(JSONParser.MODE_PERMISSIVE); - jsonObject = (JSONObject) parser.parse(requestString); - } catch (ParseException e) { - throw new BankException("Error in casting JSON body " + e); - } - - JSONObject additionalConsentInfo = (JSONObject) accountRequestInformation.get("additionalConsentInfo"); - - JSONObject response = cacheAndGetPaymentResponse(jsonObject, additionalConsentInfo); - return Response.status(201).entity(response.toString()) - .header("x-fapi-interaction-id", fid) - .build(); - - } - - @SuppressFBWarnings("JAXRS_ENDPOINT") - // Suppressed content - Endpoint - // Suppression reason - False Positive : This endpoint is a demo endpoint that is not exposed in production - // Suppressed warning count - 1 - - @GET - @Path("/payments/{paymentId}") - @Produces("application/json; charset=utf-8") - public Response getPaymentTypePayment(@PathParam("paymentId") String paymentId) { - - JSONObject responseObject = null; - if (StringUtils.isNotBlank(paymentId)) { - - responseObject = domesticPayments.get(paymentId); - - } - if (responseObject == null) { - responseObject = new JSONObject(); - } - - - return Response.status(200).entity(responseObject.toString()) - .header("x-fapi-interaction-id", "93bac548-d2de-4546-b106-880a5018460d") - .build(); - } - - private static JSONObject getRequest(String json) throws ParseException { - - String[] splitString = json.split("\\."); - String base64EncodedBody = splitString[1]; - String decodedString = null; - decodedString = new String(Base64.getUrlDecoder() - .decode(base64EncodedBody.getBytes(StandardCharsets.UTF_8)), StandardCharsets.UTF_8); - - JSONParser parser = new JSONParser(JSONParser.MODE_PERMISSIVE); - JSONObject jsonObject = (JSONObject) parser.parse(decodedString); - return jsonObject; - } - - @SuppressFBWarnings("PREDICTABLE_RANDOM") - // Suppressed content - PREDICTABLE_RANDOM - // Suppression reason - False Positive : This endpoint is a demo endpoint that is not exposed in production - // Suppressed warning count - 1 - private JSONObject cacheAndGetPaymentResponse(JSONObject requestObject, - JSONObject additionalConsentInfo) - throws BankException { - - JSONObject responseObject; - - int randomPIN = new Random().nextInt(100); - - String status; - String paymentIdValue; - - paymentIdValue = ((JSONObject) requestObject.get("Data")).getAsString("ConsentId"); - paymentIdValue = paymentIdValue + "-" + randomPIN; - - status = "AcceptedSettlementCompleted"; - - - DateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssXXX"); - Date date = new Date(); - String currentDate = dateFormat.format(date); - - String readRefundAccount = additionalConsentInfo.getAsString("ReadRefundAccount"); - String cutOffTimeAcceptable = additionalConsentInfo.getAsString("CutOffTimeAcceptable"); - - try { - JSONParser parser = new JSONParser(JSONParser.MODE_PERMISSIVE); - responseObject = (JSONObject) parser.parse(requestObject.toString()); - - JSONObject dataObject = (JSONObject) responseObject.get("Data"); - - dataObject.put("PaymentId", paymentIdValue); - dataObject.put("Status", status); - dataObject.put("CreationDateTime", currentDate); - dataObject.put("StatusUpdateDateTime", currentDate); - - // Add refund account details if requested during consent initiation - if (Boolean.parseBoolean(readRefundAccount)) { - addRefundAccount(dataObject); - } - - if (Boolean.parseBoolean(cutOffTimeAcceptable)) { - dataObject.put(EXPECTED_EXECUTION_TIME, constructDateTime(1L, - OpenBankingConstants.EXPECTED_EXECUTION_TIME)); - dataObject.put(EXPECTED_SETTLEMENT_TIME, constructDateTime(1L, - OpenBankingConstants.EXPECTED_SETTLEMENT_TIME)); - } - - JSONObject linksObject = new JSONObject(); - linksObject.put("Self", "/payments/" + paymentIdValue); - responseObject.put("Links", linksObject); - - JSONObject metaObject = new JSONObject(); - responseObject.put("Meta", metaObject); - - responseObject.remove("Risk"); - - } catch (ParseException e) { - throw new BankException(e); - } - addToCache(paymentIdValue, responseObject); - return responseObject; - } - - /** - * Add Refund account details to the response. - * - * @param dataObject - */ - private void addRefundAccount(JSONObject dataObject) { - - String schemeName = "OB.SortCodeAccountNumber"; - String identification = "Identification"; - String name = "NTPC Inc"; - - JSONObject accountData = new JSONObject(); - accountData.put("SchemeName", schemeName); - accountData.put("Identification", identification); - accountData.put("Name", name); - - JSONObject account = new JSONObject(); - account.put("Account", accountData); - - dataObject.put("Refund", account); - } - - public static String constructDateTime(long daysToAdd, String configToRead) { - - OpenBankingConfigParser parser = OpenBankingConfigParser.getInstance(); - String time = (String) parser.getConfiguration().get(configToRead); - String dateValue = LocalDate.now().plusDays(daysToAdd) + "T" + (OffsetTime.parse(time)); - - OffsetDateTime offSetDateVal = OffsetDateTime.parse(dateValue); - DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ssXXX"); - return dateTimeFormatter.format(offSetDateVal); - } - - private void addToCache(String paymentIdValue, JSONObject responseObject) { - - if (domesticPayments.size() > MAX_LIMIT) { - // Max limit reached - domesticPayments.remove(domesticPaymentsIdQueue.poll()); - } - domesticPayments.put(paymentIdValue, responseObject); - domesticPaymentsIdQueue.add(paymentIdValue); - - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/services/VrpService.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/services/VrpService.java deleted file mode 100644 index 14ce852c..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/java/com/wso2/openbanking/accelerator/demo/backend/services/VrpService.java +++ /dev/null @@ -1,290 +0,0 @@ -/** - * Copyright (c) 2024, WSO2 LLC. (https://www.wso2.com). - *

- * WSO2 LLC. 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 com.wso2.openbanking.accelerator.demo.backend.services; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.demo.backend.BankException; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.commons.lang3.StringUtils; - -import java.nio.charset.StandardCharsets; -import java.text.DateFormat; -import java.text.SimpleDateFormat; -import java.time.Instant; -import java.time.LocalDate; -import java.time.OffsetDateTime; -import java.time.OffsetTime; -import java.time.format.DateTimeFormatter; -import java.util.Base64; -import java.util.Date; -import java.util.HashMap; -import java.util.LinkedList; -import java.util.Map; -import java.util.Queue; -import java.util.Random; -import java.util.UUID; - - -import javax.ws.rs.GET; -import javax.ws.rs.HeaderParam; -import javax.ws.rs.POST; -import javax.ws.rs.Path; -import javax.ws.rs.PathParam; -import javax.ws.rs.Produces; -import javax.ws.rs.core.Response; - - -/** - * Vrp Service class. - */ -@Path("/vrpservice/") -public class VrpService { - - public static final String EXPECTED_EXECUTION_TIME = "ExpectedExecutionDateTime"; - public static final String EXPECTED_SETTLEMENT_TIME = "ExpectedSettlementDateTime"; - private static final int MAX_LIMIT = 500; - private static final Queue domesticVRPsIdQueue = new LinkedList<>(); - private static final Map domesticVRPs = new HashMap<>(); - - - @SuppressFBWarnings("JAXRS_ENDPOINT") - // Suppressed content - Endpoint - // Suppression reason - False Positive : This endpoint is a demo endpoint that is not exposed in production - // Suppressed warning count - 1 - @GET - @Path("/domestic-vrp-consents/{ConsentId}/funds-confirmation") - @Produces("application/json; charset=utf-8") - public Response getPaymentTypeFundsConfirmation(@PathParam("ConsentId") String domesticVRPId) { - - Instant currentDate = Instant.now(); - - String response = "{\n" + - " \"Data\": {\n" + - " \"FundsAvailableResult\": {\n" + - " \"FundsAvailableDateTime\": \"" + currentDate.toString() + "\",\n" + - " \"FundsAvailable\": true\n" + - " }\n" + - " },\n" + - " \"Links\": {\n" + - " \"Self\": \"/vrp/domestic-vrps/" + domesticVRPId + "/funds-confirmation\"\n" + - " },\n" + - " \"Meta\": {}\n" + - "}"; - - return Response.status(200).entity(response) - .header("x-fapi-interaction-id", UUID.randomUUID().toString()) - .build(); - } - - @SuppressFBWarnings("JAXRS_ENDPOINT") - // Suppressed content - Endpoint - // Suppression reason - False Positive : This endpoint is a demo endpoint that is not exposed in production - // Suppressed warning count - 1 - - @POST - @Path("/domestic-vrps") - @Produces("application/json; charset=utf-8") - public Response paymentSubmission(String requestString, @PathParam("paymentType") String paymentType, - @HeaderParam("x-fapi-interaction-id") String fid, - @HeaderParam("Account-Request-Information") String accountRequestInfo) - throws BankException { - - JSONObject jsonObject; - JSONObject accountRequestInformation; - - try { - accountRequestInformation = getRequest(paymentType, accountRequestInfo); - JSONParser parser = new JSONParser(JSONParser.MODE_PERMISSIVE); - jsonObject = (JSONObject) parser.parse(requestString); - } catch (ParseException e) { - throw new BankException("Error in casting JSON body " + e); - } - - JSONObject additionalConsentInfo = (JSONObject) accountRequestInformation.get("additionalConsentInfo"); - - JSONObject response = cacheAndGetPaymentResponse(paymentType, jsonObject, additionalConsentInfo); - return Response.status(201).entity(response.toString()) - .header("x-fapi-interaction-id", fid) - .build(); - - } - - @SuppressFBWarnings("JAXRS_ENDPOINT") - // Suppressed content - Endpoint - // Suppression reason - False Positive : This endpoint is a demo endpoint that is not exposed in production - // Suppressed warning count - 1 - - @GET - @Path("/domestic-vrps/{domesticVRPId}") - @Produces("application/json; charset=utf-8") - public Response getPaymentTypePayment(@PathParam("domesticVRPId") String domesticVRPId) { - - JSONObject responseObject = null; - if (StringUtils.isNotBlank(domesticVRPId)) { - - responseObject = domesticVRPs.get(domesticVRPId); - - } - if (responseObject == null) { - responseObject = new JSONObject(); - } - - - return Response.status(200).entity(responseObject.toString()) - .header("x-fapi-interaction-id", "93bac548-d2de-4546-b106-880a5018460d") - .build(); - } - - - private static JSONObject getRequest(String paymentType, String json) throws ParseException { - - String[] splitString = json.split("\\."); - String base64EncodedBody = splitString[1]; - String decodedString = null; - decodedString = new String(Base64.getUrlDecoder() - .decode(base64EncodedBody.getBytes(StandardCharsets.UTF_8)), StandardCharsets.UTF_8); - - JSONParser parser = new JSONParser(JSONParser.MODE_PERMISSIVE); - JSONObject jsonObject = (JSONObject) parser.parse(decodedString); - return jsonObject; - } - - - @SuppressFBWarnings("PREDICTABLE_RANDOM") - // Suppressed content - PREDICTABLE_RANDOM - // Suppression reason - False Positive : This endpoint is a demo endpoint that is not exposed in production - // Suppressed warning count - 1 - private JSONObject cacheAndGetPaymentResponse(String paymentType, JSONObject requestObject, - JSONObject additionalConsentInfo) - throws BankException { - - JSONObject responseObject; - - int randomPIN = new Random().nextInt(100); - - String status; - String paymentIdValue; - - paymentIdValue = ((JSONObject) requestObject.get("Data")).getAsString("ConsentId"); - paymentIdValue = paymentIdValue + "-" + randomPIN; - - status = "AcceptedSettlementCompleted"; - - DateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssXXX"); - Date date = new Date(); - String currentDate = dateFormat.format(date); - - String readRefundAccount = additionalConsentInfo.getAsString("ReadRefundAccount"); - String cutOffTimeAcceptable = additionalConsentInfo.getAsString("CutOffTimeAcceptable"); - - try { - JSONParser parser = new JSONParser(JSONParser.MODE_PERMISSIVE); - responseObject = (JSONObject) parser.parse(requestObject.toString()); - - JSONObject dataObject = (JSONObject) responseObject.get("Data"); - - dataObject.put("DomesticVRPId", paymentIdValue); - dataObject.put("Status", status); - dataObject.put("CreationDateTime", currentDate); - dataObject.put("StatusUpdateDateTime", currentDate); - - if ("domestic-vrps".equals(paymentType)) { - JSONObject debtorAccount = new JSONObject(); - debtorAccount.put("SchemeName", "SortCodeAccountNumber"); - debtorAccount.put("SecondaryIdentification", "Roll 2901"); - debtorAccount.put("Name", "Deb Mal"); - debtorAccount.put("Identification", additionalConsentInfo.getAsString("AccountIds") - .split(":")[0].replace("[\"", "")); - - dataObject.put("DebtorAccount", debtorAccount); - - } - - // Add refund account details if requested during consent initiation - if (Boolean.parseBoolean(readRefundAccount)) { - addRefundAccount(dataObject); - } - - if (Boolean.parseBoolean(cutOffTimeAcceptable)) { - dataObject.put(EXPECTED_EXECUTION_TIME, constructDateTime(1L, - OpenBankingConstants.EXPECTED_EXECUTION_TIME)); - dataObject.put(EXPECTED_SETTLEMENT_TIME, constructDateTime(1L, - OpenBankingConstants.EXPECTED_SETTLEMENT_TIME)); - } - - JSONObject linksObject = new JSONObject(); - linksObject.put("Self", "/domestic-vrps/" + paymentIdValue); - responseObject.put("Links", linksObject); - - JSONObject metaObject = new JSONObject(); - responseObject.put("Meta", metaObject); - - } catch (ParseException e) { - throw new BankException(e); - } - addToCache(paymentIdValue, responseObject); - return responseObject; - } - - /** - * Add Refund account details to the response. - * - * @param dataObject - */ - private void addRefundAccount(JSONObject dataObject) { - - String schemeName = "OB.SortCodeAccountNumber"; - String identification = "Identification"; - String name = "NTPC Inc"; - - JSONObject accountData = new JSONObject(); - accountData.put("SchemeName", schemeName); - accountData.put("Identification", identification); - accountData.put("Name", name); - - JSONObject account = new JSONObject(); - account.put("Account", accountData); - - dataObject.put("Refund", account); - } - - public static String constructDateTime(long daysToAdd, String configToRead) { - - OpenBankingConfigParser parser = OpenBankingConfigParser.getInstance(); - String time = (String) parser.getConfiguration().get(configToRead); - String dateValue = LocalDate.now().plusDays(daysToAdd) + "T" + (OffsetTime.parse(time)); - - OffsetDateTime offSetDateVal = OffsetDateTime.parse(dateValue); - DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ssXXX"); - return dateTimeFormatter.format(offSetDateVal); - } - - private void addToCache(String paymentIdValue, JSONObject responseObject) { - - if (domesticVRPs.size() > MAX_LIMIT) { - // Max limit reached - domesticVRPs.remove(domesticVRPsIdQueue.poll()); - } - domesticVRPs.put(paymentIdValue, responseObject); - domesticVRPsIdQueue.add(paymentIdValue); - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/resources/findbugs-include.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/resources/findbugs-include.xml deleted file mode 100644 index 8932a22e..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/webapp/META-INF/webapp-classloading.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/webapp/META-INF/webapp-classloading.xml deleted file mode 100644 index fe6f12f8..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/webapp/META-INF/webapp-classloading.xml +++ /dev/null @@ -1,34 +0,0 @@ - - - - - - - - false - - - Carbon,CXF3 - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/webapp/WEB-INF/cxf-servlet.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/webapp/WEB-INF/cxf-servlet.xml deleted file mode 100644 index 71138c28..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/webapp/WEB-INF/cxf-servlet.xml +++ /dev/null @@ -1,59 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/webapp/WEB-INF/web.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/webapp/WEB-INF/web.xml deleted file mode 100644 index 7fa90a98..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend/src/main/webapp/WEB-INF/web.xml +++ /dev/null @@ -1,55 +0,0 @@ - - - - - - Open-Banking - - - JAXServlet - JAX-WS/JAX-RS Servlet - JAX-WS/JAX-RS Endpoint - - org.apache.cxf.transport.servlet.CXFServlet - - - service-list-stylesheet - servicelist.css - - - jersey.config.server.provider.classnames - org.glassfish.jersey.media.multipart.MultiPartFeature - - - 1 - - - - JAXServlet - /services/* - - - - 60 - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/pom.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/pom.xml deleted file mode 100644 index 24cf3bb7..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/pom.xml +++ /dev/null @@ -1,123 +0,0 @@ - - - - - open-banking - com.wso2.openbanking.accelerator - 3.2.0-SNAPSHOT - ../../../pom.xml - - 4.0.0 - - openbanking-accelerator-demosite-endpoint - war - WSO2 Open Banking - Demo-site Backend Endpoint - - - - commons-logging - commons-logging - - - io.rest-assured - rest-assured - compile - - - org.springframework - spring-web - provided - - - org.apache.cxf - cxf-bundle-jaxrs - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - provided - - - javax.ws.rs - jsr311-api - - - - - com.fasterxml.jackson.core - jackson-databind - provided - - - net.minidev - json-smart - provided - - - - - - - maven-war-plugin - ${maven-war-plugin.version} - - - - - src/main/webapp - - - api#openbanking#demosite - WEB-INF/lib/slf4j-api-*.jar - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - High - true - true - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/java/com/wso2/openbanking/accelerator/demosite/endpoint/api/JWTGeneratorEndpoint.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/java/com/wso2/openbanking/accelerator/demosite/endpoint/api/JWTGeneratorEndpoint.java deleted file mode 100644 index fad588b5..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/java/com/wso2/openbanking/accelerator/demosite/endpoint/api/JWTGeneratorEndpoint.java +++ /dev/null @@ -1,98 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.demosite.endpoint.api; - -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.databind.ObjectMapper; -import com.google.gson.Gson; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.demosite.endpoint.model.JWTGeneratorEndpointErrorResponse; -import com.wso2.openbanking.accelerator.demosite.endpoint.model.PayloadData; -import com.wso2.openbanking.accelerator.demosite.endpoint.util.GeneratorUtil; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; -import javax.ws.rs.Consumes; -import javax.ws.rs.GET; -import javax.ws.rs.POST; -import javax.ws.rs.Path; -import javax.ws.rs.Produces; -import javax.ws.rs.core.Context; -import javax.ws.rs.core.MultivaluedMap; -import javax.ws.rs.core.Response; - -/** - * Demo-site Endpoint - * This specifies the RESTful APIs to generate the payloads required to try out the OB flows in the demo-site. - */ -@Path("/") -public class JWTGeneratorEndpoint { - - private static Log log = LogFactory.getLog(JWTGeneratorEndpoint.class); - - /** - * Generate the RequestObject, DCR Payload and Token Assertion - */ - @POST - @Path("/getJWT") - @Consumes({"application/x-www-form-urlencoded"}) - @Produces({"application/json; charset=utf-8"}) - public Response getJWT(@Context HttpServletRequest request, @Context HttpServletResponse response, - MultivaluedMap parameterMap) { - String requestPayload; - try { - requestPayload = new ObjectMapper().writeValueAsString(parameterMap); - } catch (JsonProcessingException e) { - String error = "Error in formatting the request payload"; - log.error(error, e); - JWTGeneratorEndpointErrorResponse errorResponse = GeneratorUtil.createErrorResponse(400, error); - return Response.status(400).entity(errorResponse.getPayload()).build(); - } - PayloadData data = new Gson().fromJson(requestPayload.replaceAll("\\\\r|\\\\n|\\r|\\n|\\[|]", ""), - PayloadData.class); - try { - return Response.status(201).entity(GeneratorUtil.generateJWT(data)).build(); - } catch (OpenBankingException e) { - String error = "Error occurred while building the JWT"; - log.error(error, e); - JWTGeneratorEndpointErrorResponse errorResponse = GeneratorUtil.createErrorResponse(500, error); - return Response.status(500).entity(errorResponse.getPayload()).build(); - } - } - - /** - * Update key and certificate used to sign the JWT content - */ - @GET - @Path("/updateCerts") - @Produces({"application/json; charset=utf-8"}) - public Response updateCertificates(@Context HttpServletRequest request, @Context HttpServletResponse response) { - try { - return Response.status(201).entity(GeneratorUtil.updateConfigurations()).build(); - } catch (OpenBankingException e) { - String error = "Error occurred while updating the certificates"; - log.error(error, e); - JWTGeneratorEndpointErrorResponse errorResponse = GeneratorUtil.createErrorResponse(500, error); - return Response.status(500).entity(errorResponse.getPayload()).build(); - } - } - -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/java/com/wso2/openbanking/accelerator/demosite/endpoint/model/JWTGeneratorEndpointErrorResponse.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/java/com/wso2/openbanking/accelerator/demosite/endpoint/model/JWTGeneratorEndpointErrorResponse.java deleted file mode 100644 index e6f07eb4..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/java/com/wso2/openbanking/accelerator/demosite/endpoint/model/JWTGeneratorEndpointErrorResponse.java +++ /dev/null @@ -1,50 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.demosite.endpoint.model; - -import net.minidev.json.JSONObject; - -/** - * Demo-site JWT generator endpoint error response - */ -public class JWTGeneratorEndpointErrorResponse { - - private int httpStatusCode = 0; - private JSONObject payload = null; - - public int getHttpStatusCode() { - - return httpStatusCode; - } - public void setHttpStatusCode(int httpStatusCode) { - - this.httpStatusCode = httpStatusCode; - } - - public JSONObject getPayload() { - - return payload; - } - public void setPayload(JSONObject payload) { - - this.payload = payload; - } - -} - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/java/com/wso2/openbanking/accelerator/demosite/endpoint/model/PayloadData.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/java/com/wso2/openbanking/accelerator/demosite/endpoint/model/PayloadData.java deleted file mode 100644 index b5a42dcf..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/java/com/wso2/openbanking/accelerator/demosite/endpoint/model/PayloadData.java +++ /dev/null @@ -1,108 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.demosite.endpoint.model; - -/** - * Data wrapper for the request payload data. - */ -public class PayloadData { - - private String clientId; - private String payload; - private String redirectUri; - private String consentId; - private String scopes; - private String type; - private String apiName; - private String ssa; - private String softwareId; - - public String getClientId() { - return clientId; - } - - public void setClientId(String clientId) { - this.clientId = clientId; - } - - public String getPayload() { - return payload; - } - - public void setPayload(String payload) { - this.payload = payload; - } - - public String getRedirectUri() { - return redirectUri; - } - - public void setRedirectUri(String redirectUri) { - this.redirectUri = redirectUri; - } - - public String getConsentId() { - return consentId; - } - - public void setConsentId(String consentId) { - this.consentId = consentId; - } - - public String getScopes() { - return scopes; - } - - public void setScopes(String scopes) { - this.scopes = scopes; - } - - public String getType() { - return type; - } - - public void setType(String type) { - this.type = type; - } - - public String getApiName() { - return apiName; - } - - public void setApiName(String apiName) { - this.apiName = apiName; - } - - public String getSsa() { - return ssa; - } - - public void setSsa(String ssa) { - this.ssa = ssa; - } - - public String getSoftwareId() { - return softwareId; - } - - public void setSoftwareId(String softwareId) { - this.softwareId = softwareId; - } - -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/java/com/wso2/openbanking/accelerator/demosite/endpoint/util/GeneratorUtil.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/java/com/wso2/openbanking/accelerator/demosite/endpoint/util/GeneratorUtil.java deleted file mode 100644 index 2423ff79..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/java/com/wso2/openbanking/accelerator/demosite/endpoint/util/GeneratorUtil.java +++ /dev/null @@ -1,434 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.demosite.endpoint.util; - -import com.nimbusds.jose.JOSEException; -import com.nimbusds.jose.JOSEObjectType; -import com.nimbusds.jose.JWSAlgorithm; -import com.nimbusds.jose.JWSHeader; -import com.nimbusds.jose.JWSObject; -import com.nimbusds.jose.JWSSigner; -import com.nimbusds.jose.Payload; -import com.nimbusds.jose.crypto.RSASSASigner; -import com.nimbusds.jose.jwk.RSAKey; -import com.wso2.openbanking.accelerator.common.exception.OpenBankingException; -import com.wso2.openbanking.accelerator.demosite.endpoint.model.JWTGeneratorEndpointErrorResponse; -import com.wso2.openbanking.accelerator.demosite.endpoint.model.PayloadData; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; -import org.json.JSONObject; - -import java.io.ByteArrayInputStream; -import java.io.FileInputStream; -import java.io.IOException; -import java.io.InputStream; -import java.net.URL; -import java.nio.charset.StandardCharsets; -import java.nio.file.Files; -import java.nio.file.Paths; -import java.nio.file.StandardCopyOption; -import java.security.KeyFactory; -import java.security.NoSuchAlgorithmException; -import java.security.PrivateKey; -import java.security.cert.Certificate; -import java.security.cert.CertificateException; -import java.security.cert.CertificateFactory; -import java.security.cert.X509Certificate; -import java.security.spec.InvalidKeySpecException; -import java.security.spec.PKCS8EncodedKeySpec; -import java.time.Instant; -import java.util.Base64; -import java.util.Locale; -import java.util.Properties; -import java.util.UUID; - -/** - * Utils class for the demo-site JWTGenerator endpoint - */ -public class GeneratorUtil { - - private static final Log log = LogFactory.getLog(GeneratorUtil.class); - private static final JWSAlgorithm DEFAULT_ALGORITHM = JWSAlgorithm.PS256; - private static String certPath = ""; - private static String keyPath = ""; - private static String serverDomain = ""; - private static String port = ""; - private static boolean isExternalLink = false; - private static String kid = ""; - private static Certificate certificate = null; - private static JWSSigner signer = null; - - /** - * Generate the JWT required for token assertion, request object and DCR payload - * - * @param requestData Request data object which contains request parameters - * @return The signed JWT - * @throws OpenBankingException - */ - public static String generateJWT(PayloadData requestData) throws OpenBankingException { - - JWSHeader header; - Payload payload; - String appName = null; - - if (certificate == null || signer == null || kid.equals("") || serverDomain.equals("") || port.equals("")) { - try { - updateConfigurations(); - } catch (OpenBankingException e) { - String error = "Error in updating certificates"; - log.error(error, e); - throw new OpenBankingException(error, e); - } - } - - if (requestData.getType().toLowerCase(Locale.ENGLISH).contains("dcr")) { - // For DCR app registrations the software ID is set to newApp from the frontend - if (requestData.getSoftwareId().equals("newApp")) { - appName = UUID.randomUUID().toString(); - } else { - appName = requestData.getSoftwareId(); - } - requestData.setSsa(generateSSA(requestData, appName)); - } - header = generateHeader(requestData); - payload = generatePayload(requestData, appName); - - try { - return signJWT(header, payload); - } catch (OpenBankingException e) { - String error = "Error while signing JWT/JWS"; - log.error(error, e); - throw new OpenBankingException(error, e); - } - } - - /** - * Download and update the stored certificates - * - * @throws OpenBankingException - */ - public static String updateConfigurations() throws OpenBankingException { - try { - InputStream configurations = GeneratorUtil.class.getClassLoader() - .getResourceAsStream("configurations.properties"); - Properties configurationProperties = new Properties(); - configurationProperties.load(configurations); - certPath = configurationProperties.getProperty("CertificateConfigs.CertUrl"); - keyPath = configurationProperties.getProperty("CertificateConfigs.KeyUrl"); - isExternalLink = Boolean.parseBoolean( - configurationProperties.getProperty("CertificateConfigs.IsExternalLink")); - serverDomain = configurationProperties.getProperty("PayloadConfigs.IamDomain"); - port = configurationProperties.getProperty("PayloadConfigs.Port"); - } catch (IOException e) { - String error = "Error occurred while reading the configurations"; - log.error(error, e); - throw new OpenBankingException(error, e); - } - - try { - certificate = getPublicSigningCert(); - signer = new RSASSASigner((PrivateKey) getSigningKey()); - try { - kid = getThumbPrint(certificate); - log.info("The certificates were updated successfully"); - return "Certificates updated successfully"; - } catch (OpenBankingException e) { - String error = "Error when getting thumbprint of primary public cert"; - log.error(error, e); - throw new OpenBankingException(error, e); - } - } catch (OpenBankingException e) { - String error = "Error when retrieving primary public cert"; - log.error(error, e); - throw new OpenBankingException(error, e); - } - } - - /** - * Generate SHA-1 DER Thumbprint. - * - * @param certificate - * @return Thumbprint - * @throws OpenBankingException - */ - private static String getThumbPrint(Certificate certificate) throws OpenBankingException { - - try { - X509Certificate x509cert = (X509Certificate) CertificateFactory.getInstance("X.509") - .generateCertificate(new ByteArrayInputStream(certificate.getEncoded())); - return RSAKey.parse(x509cert).computeThumbprint("SHA-1").toString(); - } catch (CertificateException | JOSEException e) { - String error = "Error occurred while generating SHA-1 JWK thumbprint"; - log.error(error, e); - throw new OpenBankingException(error, e); - } - } - - /** - * Generate the payload of the JWT - * - * @param requestData Request data object which contains request parameters - * @param appName Name of the application to be created - * @return The payload of the JWT - */ - private static Payload generatePayload(PayloadData requestData, String appName) { - - long initiationTime = Instant.now().getEpochSecond(); - long expirationTime = initiationTime + 3600; - long jtiValue = initiationTime + 10; - String apiName = requestData.getApiName().toLowerCase(Locale.ENGLISH); - String payloadString = ""; - - if (apiName.contains("authorize")) { - payloadString = "{\n" + - " \"max_age\": 86400,\n" + - " \"aud\": \"" + serverDomain + ":" + port + "/oauth2/token\",\n" + - " \"scope\": \"" + requestData.getScopes() + "\",\n" + - " \"iss\": \"" + requestData.getClientId() + "\",\n" + - " \"claims\": {\n" + - " \"id_token\": {\n" + - " \"acr\": {\n" + - " \"values\": [\n" + - " \"urn:openbanking:psd2:sca\",\n" + - " \"urn:openbanking:psd2:ca\"\n" + - " ],\n" + - " \"essential\": true\n" + - " },\n" + - " \"openbanking_intent_id\": {\n" + - " \"value\": \"" + requestData.getConsentId() + "\",\n" + - " \"essential\": true\n" + - " }\n" + - " },\n" + - " \"userinfo\": {\n" + - " \"openbanking_intent_id\": {\n" + - " \"value\": \"" + requestData.getConsentId() + "\",\n" + - " \"essential\": true\n" + - " }\n" + - " }\n" + - " },\n" + - " \"response_type\": \"code id_token\",\n" + - " \"redirect_uri\": \"" + requestData.getRedirectUri() + "\",\n" + - " \"state\": \"YWlzcDozMTQ2\",\n" + - " \"exp\": " + expirationTime + ",\n" + - " \"nonce\": \"n-0S6_WzA2M0000025\",\n" + - " \"client_id\": \"" + requestData.getClientId() + "\"\n" + - "}"; - } else if (apiName.contains("token")) { - payloadString = "{\n" + - " \"iss\": \"" + requestData.getClientId() + "\",\n" + - " \"sub\": \"" + requestData.getClientId() + "\",\n" + - " \"exp\": " + expirationTime + ",\n" + - " \"iat\": " + initiationTime + ",\n" + - " \"jti\": \"" + jtiValue + "\",\n" + - " \"aud\": \"" + serverDomain + ":" + port + "/oauth2/token\"\n" + - "}"; - } else if (apiName.contains("dynamic")) { - payloadString = "{\n" + - " \"iss\": \"" + appName + "\",\n" + - " \"iat\": " + initiationTime + ",\n" + - " \"exp\": " + expirationTime + ",\n" + - " \"jti\": \"" + jtiValue + "\",\n" + - " \"aud\": \"https://localbank.com\",\n" + - " \"scope\": \"accounts payments\",\n" + - " \"token_endpoint_auth_method\": \"private_key_jwt\",\n" + - " \"token_endpoint_auth_signing_alg\": \"PS256\",\n" + - " \"grant_types\": [\n" + - " \"authorization_code\",\n" + - " \"client_credentials\",\n" + - " \"refresh_token\"\n" + - " ],\n" + - " \"response_types\": [\n" + - " \"code id_token\"\n" + - " ],\n" + - " \"id_token_signed_response_alg\": \"PS256\",\n" + - " \"request_object_signing_alg\": \"PS256\",\n" + - " \"application_type\": \"web\",\n" + - " \"software_id\": \"" + appName + "\",\n" + - " \"redirect_uris\": [\n" + - " \"" + serverDomain + "/ob/authenticationendpoint/auth_code.do\"\n" + - " ],\n" + - " \"software_statement\": \"" + requestData.getSsa() + "\"\n" + - "}"; - } - return new Payload(payloadString); - } - - /** - * Extract the signing key from the keystore - * - * @return The signing key - * @throws OpenBankingException - */ - private static PrivateKey getSigningKey() throws OpenBankingException { - try { - if (isExternalLink) { - Files.copy(new URL(keyPath).openStream(), Paths.get("key.key"), StandardCopyOption.REPLACE_EXISTING); - } else { - Files.copy(GeneratorUtil.class.getClassLoader().getResourceAsStream("signing.key"), - Paths.get("key.key"), StandardCopyOption.REPLACE_EXISTING); - } - String privateKeyPath = "key.key"; - String keyContent = new String(Files.readAllBytes(Paths.get(privateKeyPath)), StandardCharsets.UTF_8); - keyContent = keyContent.replace("-----BEGIN PRIVATE KEY-----", ""); - keyContent = keyContent.replace("-----END PRIVATE KEY-----", ""); - keyContent = keyContent.replace("\n", ""); - return KeyFactory.getInstance("RSA") - .generatePrivate(new PKCS8EncodedKeySpec(Base64.getDecoder().decode(keyContent))); - } catch (NoSuchAlgorithmException | IOException | InvalidKeySpecException e) { - String error = "Error in extracting the signing private key"; - log.error(error, e); - throw new OpenBankingException(error, e); - } - } - - /** - * Extract the certificate from the keystore - * - * @return The certificate - * @throws OpenBankingException - */ - private static Certificate getPublicSigningCert() throws OpenBankingException { - try { - if (isExternalLink) { - Files.copy(new URL(certPath).openStream(), Paths.get("cert.pem"), StandardCopyOption.REPLACE_EXISTING); - } else { - Files.copy(GeneratorUtil.class.getClassLoader().getResourceAsStream("signing.pem"), - Paths.get("cert.pem"), StandardCopyOption.REPLACE_EXISTING); - } - String publicCertPath = "cert.pem"; - return CertificateFactory.getInstance("X.509").generateCertificate( - new FileInputStream(publicCertPath)); - } catch (CertificateException | IOException e) { - String error = "Error in extracting the signing certificate"; - log.error(error, e); - throw new OpenBankingException(error, e); - } - } - - /** - * Sign the JWT including header and payload content - * - * @param header Header content of the JWT - * @param payload Payload content of the JWT - * @return signed JWT - * @throws OpenBankingException - */ - private static String signJWT(JWSHeader header, Payload payload) throws OpenBankingException { - - JWSObject jwsObject = new JWSObject(header, payload); - try { - jwsObject.sign(signer); - log.info("The JWT was generated successfully"); - return jwsObject.serialize(); - } catch (JOSEException e) { - String error = "Unable to sign JWT with signer"; - log.error(error, e); - throw new OpenBankingException(error, e); - } - } - - /** - * Generate SSA payload of the DCR App - * - * @param requestData Request data object which contains request parameters - * @param appName Name of the application to be created - * @return SSA payload - * @throws OpenBankingException - */ - private static String generateSSA(PayloadData requestData, String appName) throws OpenBankingException { - long initiationTime = Instant.now().getEpochSecond(); - long expirationTime = initiationTime + 3600; - long jtiValue = initiationTime + 10; - String requestType = requestData.getType(); - - String ssaContent = new String(Base64.getUrlDecoder() - .decode(requestData.getSsa().split("\\.")[1]), StandardCharsets.UTF_8); - JSONObject ssaContentObject = new JSONObject(ssaContent); - ssaContentObject.remove("iat"); - ssaContentObject.put("iat", initiationTime); - ssaContentObject.remove("exp"); - ssaContentObject.put("exp", expirationTime); - ssaContentObject.remove("jti"); - ssaContentObject.put("jti", String.valueOf(jtiValue)); - ssaContentObject.remove("software_id"); - ssaContentObject.put("software_id", appName); - ssaContentObject.remove("software_client_id"); - ssaContentObject.put("software_client_id", appName); - ssaContentObject.remove("software_redirect_uris"); - String[] redirectUris = { serverDomain + "/ob/authenticationendpoint/auth_code.do" }; - ssaContentObject.put("software_redirect_uris", redirectUris); - requestData.setType("ssa"); - - JWSHeader header = generateHeader(requestData); - requestData.setType(requestType); - try { - return signJWT(header, new Payload(ssaContentObject.toString())); - } catch (OpenBankingException e) { - String error = "Error while signing JWT/JWS"; - log.error(error, e); - throw new OpenBankingException(error, e); - } - } - - /** - * Generate the header of the JWT - * - * @param requestData Request data object which contains request parameters - * @return The header of the JWT - */ - private static JWSHeader generateHeader(PayloadData requestData) { - - String type = requestData.getType().toLowerCase(Locale.ENGLISH); - JWSHeader header; - if (type.contains("ssa")) { - header = new JWSHeader.Builder(DEFAULT_ALGORITHM) - .keyID(kid) - .type(JOSEObjectType.JWT) - .build(); - } else if (type.contains("dcr")) { - header = new JWSHeader.Builder(DEFAULT_ALGORITHM) - .keyID(kid) - .build(); - } else { - header = new JWSHeader.Builder(DEFAULT_ALGORITHM) - .keyID(kid) - .build(); - } - return header; - } - - /** - * Generate error response in case of an exception that may occur when processing the request - * - * @param httpStatusCode Status code of the error encountered - * @param errorDescription Description of the error encountered - * @return Error response - */ - public static JWTGeneratorEndpointErrorResponse createErrorResponse(int httpStatusCode, String errorDescription) { - - JWTGeneratorEndpointErrorResponse demositeErrorResponse = new JWTGeneratorEndpointErrorResponse(); - net.minidev.json.JSONObject errorResponse = new net.minidev.json.JSONObject(); - errorResponse.put("error_description", errorDescription); - demositeErrorResponse.setPayload(errorResponse); - demositeErrorResponse.setHttpStatusCode(httpStatusCode); - - return demositeErrorResponse; - } - -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/resources/configurations.properties b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/resources/configurations.properties deleted file mode 100644 index 713ec04f..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/resources/configurations.properties +++ /dev/null @@ -1,21 +0,0 @@ -# Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). -# -# WSO2 LLC. 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. - -CertificateConfigs.CertUrl = null -CertificateConfigs.KeyUrl = null -CertificateConfigs.IsExternalLink = false -PayloadConfigs.IamDomain = https://localhost -PayloadConfigs.Port = 9446 diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/resources/findbugs-include.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/resources/findbugs-include.xml deleted file mode 100644 index 8932a22e..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/webapp/META-INF/webapp-classloading.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/webapp/META-INF/webapp-classloading.xml deleted file mode 100644 index b212826c..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/webapp/META-INF/webapp-classloading.xml +++ /dev/null @@ -1,35 +0,0 @@ - - - - - - - - - false - - - Carbon,CXF3 - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/webapp/WEB-INF/beans.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/webapp/WEB-INF/beans.xml deleted file mode 100644 index 73766b03..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/webapp/WEB-INF/beans.xml +++ /dev/null @@ -1,32 +0,0 @@ - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/webapp/WEB-INF/web.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/webapp/WEB-INF/web.xml deleted file mode 100644 index b86062ad..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demosite.endpoint/src/main/webapp/WEB-INF/web.xml +++ /dev/null @@ -1,69 +0,0 @@ - - - - - - WSO2 Open Banking - Demo-site Backend Endpoint - WSO2 Open Banking - Demo-site Backend Endpoint - - - contextConfigLocation - WEB-INF/beans.xml - - - - HttpHeaderSecurityFilter - org.apache.catalina.filters.HttpHeaderSecurityFilter - - hstsEnabled - false - - - - - HttpHeaderSecurityFilter - * - - - - - org.springframework.web.context.ContextLoaderListener - - - - - CXFServlet - - org.apache.cxf.transport.servlet.CXFServlet - - 1 - - - - CXFServlet - /* - - - - 60 - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/findbugs-exclude.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/findbugs-exclude.xml deleted file mode 100644 index 262dbed5..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/findbugs-exclude.xml +++ /dev/null @@ -1,20 +0,0 @@ - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/pom.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/pom.xml deleted file mode 100644 index cdd54d0a..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/pom.xml +++ /dev/null @@ -1,172 +0,0 @@ - - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../../pom.xml - - 4.0.0 - - openbanking-event-notifications-endpoint - war - WSO2 Open Banking - Event Notifications Endpoint - - - - io.swagger - swagger-jaxrs - - - javax.ws.rs - jsr311-api - - - com.google.guava - guava - - - org.yaml - snakeyaml - - - - - javax.validation - validation-api - provided - - - org.springframework - spring-web - provided - - - org.apache.cxf - cxf-bundle-jaxrs - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.common - provided - - - javax.ws.rs - jsr311-api - - - - - org.wso2.carbon.identity.inbound.auth.oauth2 - org.wso2.carbon.identity.oauth.dcr - provided - - - org.wso2.carbon.identity.framework - org.wso2.carbon.identity.application.mgt - provided - - - com.fasterxml.jackson.core - jackson-databind - provided - - - io.swagger - swagger-annotations - - - javax.ws.rs - jsr311-api - - - - - net.minidev - json-smart - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.event.notifications.service - - - javax.ws.rs - jsr311-api - - - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.consent.extensions - provided - - - - - - - maven-war-plugin - ${maven-war-plugin.version} - - - - - src/main/webapp - - - api#openbanking#event-notifications - WEB-INF/lib/slf4j-api-*.jar - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - true - ${project.build.directory}/spotbugs - ${project.basedir}/findbugs-exclude.xml - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/api/EventCreationEndpoint.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/api/EventCreationEndpoint.java deleted file mode 100644 index 31103497..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/api/EventCreationEndpoint.java +++ /dev/null @@ -1,154 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.endpoint.api; - -import com.wso2.openbanking.accelerator.event.notifications.endpoint.constants.EventNotificationEndPointConstants; -import com.wso2.openbanking.accelerator.event.notifications.endpoint.util.EventNotificationUtils; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.NotificationCreationDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.handler.EventCreationServiceHandler; -import com.wso2.openbanking.accelerator.event.notifications.service.response.EventCreationResponse; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import io.swagger.annotations.ApiOperation; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.nio.charset.StandardCharsets; -import java.util.Base64; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; -import javax.ws.rs.Consumes; -import javax.ws.rs.POST; -import javax.ws.rs.Path; -import javax.ws.rs.Produces; -import javax.ws.rs.core.Context; -import javax.ws.rs.core.MultivaluedMap; -import javax.ws.rs.core.Response; - -/** - * Events creation API. - */ -@Path("/") -public class EventCreationEndpoint { - - private static final Log log = LogFactory.getLog(EventCreationEndpoint.class); - private static final EventCreationServiceHandler eventCreationServiceHandler = EventNotificationUtils. - getEventNotificationCreationServiceHandler(); - private static final String specialChars = "!@#$%&*()'+,./:;<=>?[]^_`{|}"; - - /** - * This API will be used to create events. - */ - @SuppressFBWarnings({"JAXRS_ENDPOINT", "SERVLET_HEADER"}) - // Suppressed content - Endpoint - // Suppression reason - False Positive : This endpoint is secured with access control lists in the configuration - // Suppressed content - request.getHeader() - // Suppression reason - False Positive : Header is properly validated to ensure no special characters are passed - // Suppressed warning count - 4 - @POST - @Path("/create-events") - @Consumes({"application/x-www-form-urlencoded"}) - @Produces({"application/json; charset=utf-8"}) - - @ApiOperation(value = "Create Events", tags = {" Create Events"}) - public Response createEvents(@Context HttpServletRequest request, @Context HttpServletResponse response, - MultivaluedMap parameterMap) { - - - NotificationCreationDTO notificationCreationDTO = new NotificationCreationDTO(); - String requestData = StringUtils.EMPTY; - JSONObject notificationEvents; - - try { - //Check if the request pay load is empty - if (!parameterMap.isEmpty() && parameterMap.containsKey(EventNotificationEndPointConstants.REQUEST)) { - - requestData = parameterMap.get(EventNotificationEndPointConstants.REQUEST). - toString().replaceAll("\\\\r|\\\\n|\\r|\\n|\\[|]| ", StringUtils.EMPTY); - - byte[] decodedBytes = Base64.getDecoder().decode(requestData); - String decodedString = new String(decodedBytes, StandardCharsets.UTF_8); - notificationEvents = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE).parse(decodedString); - log.debug("Decoded payload string : " + decodedString.replaceAll("[\r\n]", "")); - - } else { - - return Response.status(Response.Status.BAD_REQUEST).entity(EventNotificationUtils.getErrorDTO( - EventNotificationEndPointConstants.MISSING_REQUEST_PAYLOAD, - EventNotificationConstants.MISSING_REQ_PAYLOAD)).build(); - } - - //check if the client id is present in the header - String clientId = request.getHeader(EventNotificationEndPointConstants.X_WSO2_CLIENT_ID); - if (!StringUtils.isBlank(clientId)) { - notificationCreationDTO.setClientId(request.getHeader( - EventNotificationEndPointConstants.X_WSO2_CLIENT_ID)); - } else { - return Response.status(Response.Status.BAD_REQUEST).entity(EventNotificationUtils.getErrorDTO( - EventNotificationEndPointConstants.MISSING_REQUEST_HEADER, - EventNotificationConstants.MISSING_HEADER_PARAM_CLIENT_ID)).build(); - } - - //check if the resource id is present in the header - String resourceId = request.getHeader(EventNotificationEndPointConstants.X_WSO2_RESOURCE_ID); - if (!StringUtils.isBlank(resourceId)) { - if (StringUtils.containsAny(resourceId, specialChars)) { - return Response.status(Response.Status.BAD_REQUEST).entity(EventNotificationUtils.getErrorDTO( - EventNotificationEndPointConstants.INVALID_REQUEST_HEADER, - EventNotificationConstants.INVALID_CHARS_IN_HEADER_ERROR)).build(); - } - notificationCreationDTO.setResourceId(request.getHeader( - EventNotificationEndPointConstants.X_WSO2_RESOURCE_ID));; - } else { - return Response.status(Response.Status.BAD_REQUEST).entity(EventNotificationUtils.getErrorDTO( - EventNotificationEndPointConstants.MISSING_REQUEST_HEADER, - EventNotificationConstants.MISSING_HEADER_PARAM_RESOURCE_ID)).build(); - } - - //set events to notificationCreationDTO - JSONObject finalNotificationEvents = notificationEvents; - notificationEvents.keySet().forEach(eventName -> { - JSONObject eventInformation = (JSONObject) finalNotificationEvents.get(eventName); - notificationCreationDTO.setEventPayload(eventName, eventInformation); - }); - - } catch (ParseException e) { - log.error("Error while parsing the request payload", e); - return Response.status(Response.Status.BAD_REQUEST).entity(EventNotificationUtils - .getErrorDTO(EventNotificationEndPointConstants.INVALID_REQUEST_PAYLOAD, - EventNotificationEndPointConstants.REQUEST_PAYLOAD_ERROR)).build(); - } catch (ClassCastException e) { - log.error(EventNotificationEndPointConstants.REQUEST_PAYLOAD_ERROR, e); - return Response.status(Response.Status.BAD_REQUEST).entity(EventNotificationUtils - .getErrorDTO(EventNotificationEndPointConstants.INVALID_REQUEST_PAYLOAD, - EventNotificationEndPointConstants.REQUEST_PAYLOAD_ERROR)).build(); - } - - EventCreationResponse eventCreationResponse = eventCreationServiceHandler. - publishOBEvent(notificationCreationDTO); - - return EventNotificationUtils.mapEventCreationServiceResponse(eventCreationResponse); - } - -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/api/EventPollingEndpoint.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/api/EventPollingEndpoint.java deleted file mode 100644 index 9ab95c6e..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/api/EventPollingEndpoint.java +++ /dev/null @@ -1,130 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.endpoint.api; - -import com.wso2.openbanking.accelerator.event.notifications.endpoint.constants.EventNotificationEndPointConstants; -import com.wso2.openbanking.accelerator.event.notifications.endpoint.util.EventNotificationUtils; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.handler.EventPollingServiceHandler; -import com.wso2.openbanking.accelerator.event.notifications.service.response.EventPollingResponse; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import io.swagger.annotations.ApiOperation; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.nio.charset.StandardCharsets; -import java.util.Base64; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; -import javax.ws.rs.Consumes; -import javax.ws.rs.POST; -import javax.ws.rs.Path; -import javax.ws.rs.Produces; -import javax.ws.rs.core.Context; -import javax.ws.rs.core.MultivaluedMap; -import javax.ws.rs.core.Response; - -/** - * Aggregated Event Polling API Specification. - * - *

Swagger for Aggregated Event Polling API Specification - */ -@Path("/events") -public class EventPollingEndpoint { - - private static final Log log = LogFactory.getLog(EventCreationEndpoint.class); - private static final EventPollingServiceHandler eventPollingServiceHandler = EventNotificationUtils. - getEventPollingServiceHandler(); - - /** - * Retrieve Event Notifications Using Aggregated Polling. - */ - @SuppressFBWarnings({"JAXRS_ENDPOINT", "SERVLET_HEADER"}) - // Suppressed content - Endpoint - // Suppression reason - False Positive : This endpoint is secured with access control lists in the configuration - // Suppressed content - request.getHeader() - // Suppression reason - False Positive : Header is properly validated to ensure no special characters are passed - // Suppressed warning count - 4 - @POST - @Path("/{s:.*}") - @Consumes({"application/x-www-form-urlencoded"}) - @Produces({"application/json; charset=utf-8", "application/jose+jwe"}) - @ApiOperation(value = "Retrieve Events", tags = {"Events"}) - - public Response pollEvents(@Context HttpServletRequest request, @Context HttpServletResponse response, - MultivaluedMap parameterMap) { - - String eventPollingData; - JSONObject eventPollingRequest; - - if (!parameterMap.isEmpty() && parameterMap.containsKey(EventNotificationEndPointConstants.REQUEST)) { - - eventPollingData = parameterMap.get(EventNotificationEndPointConstants.REQUEST). - toString().replaceAll("\\\\r|\\\\n|\\r|\\n|\\[|]| ", StringUtils.EMPTY); - - if (StringUtils.isNotBlank(eventPollingData)) { - byte[] decodedBytes = Base64.getDecoder().decode(eventPollingData); - String decodedString = new String(decodedBytes, StandardCharsets.UTF_8); - try { - eventPollingRequest = (JSONObject) new JSONParser(JSONParser.MODE_PERMISSIVE).parse(decodedString); - - //check if the client id is present in the header - String clientId = request.getHeader(EventNotificationConstants.X_WSO2_CLIENT_ID); - if (!StringUtils.isBlank(clientId)) { - eventPollingRequest.put(EventNotificationConstants.X_WSO2_CLIENT_ID, request. - getHeader(EventNotificationConstants.X_WSO2_CLIENT_ID)); - } else { - return Response.status(Response.Status.BAD_REQUEST).entity(EventNotificationUtils.getErrorDTO( - EventNotificationEndPointConstants.MISSING_REQUEST_HEADER, - EventNotificationConstants.MISSING_HEADER_PARAM_CLIENT_ID)).build(); - } - - EventPollingResponse eventPollingResponse = eventPollingServiceHandler. - pollEvents(eventPollingRequest); - - return EventNotificationUtils.mapEventPollingServiceResponse(eventPollingResponse); - - } catch (ParseException e) { - log.error("Exception when parsing the request payload", e); - return Response.status(Response.Status.BAD_REQUEST).entity(EventNotificationUtils.getErrorDTO( - EventNotificationEndPointConstants.INVALID_REQUEST_PAYLOAD, - EventNotificationEndPointConstants.REQUEST_PAYLOAD_ERROR)).build(); - } catch (ClassCastException e) { - log.error(EventNotificationEndPointConstants.REQUEST_PAYLOAD_ERROR, e); - return Response.status(Response.Status.BAD_REQUEST).entity(EventNotificationUtils.getErrorDTO( - EventNotificationEndPointConstants.INVALID_REQUEST_PAYLOAD, - EventNotificationEndPointConstants.REQUEST_PAYLOAD_ERROR)).build(); - } - } else { - return Response.status(Response.Status.BAD_REQUEST).entity(EventNotificationUtils.getErrorDTO( - EventNotificationEndPointConstants.INVALID_REQUEST_PAYLOAD, - EventNotificationEndPointConstants.EMPTY_REQ_PAYLOAD)).build(); - } - } else { - return Response.status(Response.Status.BAD_REQUEST).entity(EventNotificationUtils.getErrorDTO( - EventNotificationEndPointConstants.MISSING_REQUEST_PAYLOAD, - EventNotificationConstants.MISSING_REQ_PAYLOAD)).build(); - } - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/api/EventSubscriptionEndpoint.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/api/EventSubscriptionEndpoint.java deleted file mode 100644 index c5f2693b..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/api/EventSubscriptionEndpoint.java +++ /dev/null @@ -1,258 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.endpoint.api; - -import com.wso2.openbanking.accelerator.event.notifications.endpoint.constants.EventNotificationEndPointConstants; -import com.wso2.openbanking.accelerator.event.notifications.endpoint.util.EventNotificationUtils; -import com.wso2.openbanking.accelerator.event.notifications.endpoint.util.EventSubscriptionUtils; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.EventSubscriptionDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.handler.EventSubscriptionServiceHandler; -import com.wso2.openbanking.accelerator.event.notifications.service.response.EventSubscriptionResponse; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import io.swagger.annotations.ApiOperation; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.ParseException; -import org.apache.commons.lang3.StringUtils; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import java.io.IOException; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; -import javax.ws.rs.Consumes; -import javax.ws.rs.DELETE; -import javax.ws.rs.GET; -import javax.ws.rs.POST; -import javax.ws.rs.PUT; -import javax.ws.rs.Path; -import javax.ws.rs.Produces; -import javax.ws.rs.core.Context; -import javax.ws.rs.core.Response; -import javax.ws.rs.core.UriInfo; - -/** - * Events Notification Subscription API. - */ -@Path("/subscription") -public class EventSubscriptionEndpoint { - private static final Log log = LogFactory.getLog(EventSubscriptionEndpoint.class); - - private static final EventSubscriptionServiceHandler eventSubscriptionServiceHandler = EventSubscriptionUtils. - getEventSubscriptionServiceHandler(); - - /** - * Register an Event Notification Subscription. - */ - @SuppressFBWarnings({"JAXRS_ENDPOINT", "SERVLET_HEADER"}) - @POST - @Path("/") - @Consumes({"application/json; charset=utf-8"}) - @Produces({"application/json; charset=utf-8"}) - @ApiOperation(value = "Create Subscriptions", tags = {" Create Subscriptions"}) - public Response registerSubscription(@Context HttpServletRequest request, @Context HttpServletResponse response) { - - EventSubscriptionDTO eventSubscriptionDTO = new EventSubscriptionDTO(); - - //check if the client id is present in the header - String clientId = request.getHeader(EventNotificationEndPointConstants.X_WSO2_CLIENT_ID); - if (StringUtils.isBlank(clientId)) { - return Response.status(Response.Status.BAD_REQUEST).entity(EventNotificationUtils. - getErrorDTO(EventNotificationEndPointConstants.MISSING_REQUEST_HEADER, - EventNotificationConstants.MISSING_HEADER_PARAM_CLIENT_ID)).build(); - } - eventSubscriptionDTO.setClientId(request.getHeader(EventNotificationEndPointConstants.X_WSO2_CLIENT_ID)); - - // extract the payload from the request - try { - JSONObject requestData = EventSubscriptionUtils.getJSONObjectPayload(request); - if (requestData != null) { - eventSubscriptionDTO.setRequestData(requestData); - } else { - log.error("Subscription request payload is missing"); - return Response.status(Response.Status.BAD_REQUEST). - entity(EventNotificationUtils. - getErrorDTO(EventNotificationEndPointConstants.MISSING_REQUEST_PAYLOAD, - EventNotificationEndPointConstants.MISSING_JSON_REQUEST_PAYLOAD)).build(); - } - } catch (IOException e) { - log.error("Invalid Payload received", e); - return Response.status(Response.Status.BAD_REQUEST). - entity(EventNotificationUtils. - getErrorDTO(EventNotificationEndPointConstants.INVALID_REQUEST_PAYLOAD, - EventNotificationEndPointConstants.REQUEST_PAYLOAD_ERROR)).build(); - } catch (ParseException e) { - log.error("Failed to parse the payload", e); - return Response.status(Response.Status.BAD_REQUEST). - entity(EventNotificationUtils. - getErrorDTO(EventNotificationEndPointConstants.INVALID_REQUEST_PAYLOAD, - EventNotificationEndPointConstants.ERROR_PAYLOAD_PARSE)).build(); - } - EventSubscriptionResponse eventSubscriptionResponse = eventSubscriptionServiceHandler. - createEventSubscription(eventSubscriptionDTO); - return EventSubscriptionUtils.mapEventSubscriptionServiceResponse(eventSubscriptionResponse); - } - - /** - * Retrieve a Single Event Subscription. - */ - @SuppressFBWarnings({"JAXRS_ENDPOINT", "SERVLET_HEADER"}) - @GET - @Path("/{subscriptionId}") - @Consumes({"application/json; charset=utf-8"}) - @Produces({"application/json; charset=utf-8"}) - public Response retrieveSubscription(@Context HttpServletRequest request, @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - //check if the client id is present in the header - String clientId = request.getHeader(EventNotificationEndPointConstants.X_WSO2_CLIENT_ID); - if (StringUtils.isBlank(clientId)) { - return Response.status(Response.Status.BAD_REQUEST).entity(EventNotificationUtils. - getErrorDTO(EventNotificationEndPointConstants.MISSING_REQUEST_HEADER, - EventNotificationConstants.MISSING_HEADER_PARAM_CLIENT_ID)).build(); - } - - EventSubscriptionResponse eventSubscriptionResponse = eventSubscriptionServiceHandler. - getEventSubscription(clientId, uriInfo.getPathParameters().getFirst("subscriptionId")); - return EventSubscriptionUtils.mapEventSubscriptionServiceResponse(eventSubscriptionResponse); - } - - /** - * Retrieve All Events Subscriptions of a Client. - */ - @SuppressFBWarnings({"JAXRS_ENDPOINT", "SERVLET_HEADER"}) - @GET - @Path("/") - @Produces({"application/json; charset=utf-8"}) - public Response retrieveAllSubscriptions(@Context HttpServletRequest request, - @Context HttpServletResponse response) { - - //check if the client id is present in the header - String clientId = request.getHeader(EventNotificationEndPointConstants.X_WSO2_CLIENT_ID); - if (StringUtils.isBlank(clientId)) { - return Response.status(Response.Status.BAD_REQUEST).entity(EventNotificationUtils. - getErrorDTO(EventNotificationEndPointConstants.MISSING_REQUEST_HEADER, - EventNotificationConstants.MISSING_HEADER_PARAM_CLIENT_ID)).build(); - } - - EventSubscriptionResponse eventSubscriptionResponse = eventSubscriptionServiceHandler. - getAllEventSubscriptions(clientId); - return EventSubscriptionUtils.mapEventSubscriptionServiceResponse(eventSubscriptionResponse); - } - - /** - * Retrieve All Events Subscriptions by an event type. - */ - @SuppressFBWarnings({"JAXRS_ENDPOINT", "SERVLET_HEADER"}) - @GET - @Path("/type/{eventType}") - @Produces({"application/json; charset=utf-8"}) - public Response retrieveAllSubscriptionsByEventType(@Context HttpServletRequest request, - @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - - //check if the client id is present in the header - String clientId = request.getHeader(EventNotificationEndPointConstants.X_WSO2_CLIENT_ID); - if (StringUtils.isBlank(clientId)) { - return Response.status(Response.Status.BAD_REQUEST).entity(EventNotificationUtils. - getErrorDTO(EventNotificationEndPointConstants.MISSING_REQUEST_HEADER, - EventNotificationConstants.MISSING_HEADER_PARAM_CLIENT_ID)).build(); - } - - EventSubscriptionResponse eventSubscriptionResponse = eventSubscriptionServiceHandler. - getEventSubscriptionsByEventType(clientId, uriInfo.getPathParameters().getFirst("eventType")); - return EventSubscriptionUtils.mapEventSubscriptionServiceResponse(eventSubscriptionResponse); - } - - /** - * Update an Event Subscription. - */ - @SuppressFBWarnings({"JAXRS_ENDPOINT", "SERVLET_HEADER"}) - @PUT - @Path("/{subscriptionId}") - @Consumes({"application/json; charset=utf-8"}) - @Produces({"application/json; charset=utf-8"}) - public Response updateSubscription(@Context HttpServletRequest request, @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - EventSubscriptionDTO eventSubscriptionDTO = new EventSubscriptionDTO(); - - //check if the client id is present in the header - String clientId = request.getHeader(EventNotificationEndPointConstants.X_WSO2_CLIENT_ID); - if (StringUtils.isBlank(clientId)) { - return Response.status(Response.Status.BAD_REQUEST).entity(EventNotificationUtils. - getErrorDTO(EventNotificationEndPointConstants.MISSING_REQUEST_HEADER, - EventNotificationConstants.MISSING_HEADER_PARAM_CLIENT_ID)).build(); - } - eventSubscriptionDTO.setClientId(request.getHeader(EventNotificationConstants.X_WSO2_CLIENT_ID)); - - // extract the payload from the request - try { - JSONObject requestData = EventSubscriptionUtils.getJSONObjectPayload(request); - if (requestData != null) { - eventSubscriptionDTO.setRequestData(requestData); - } else { - log.error("Subscription request payload is missing"); - return Response.status(Response.Status.BAD_REQUEST). - entity(EventNotificationUtils. - getErrorDTO(EventNotificationEndPointConstants.MISSING_REQUEST_PAYLOAD, - EventNotificationEndPointConstants.MISSING_JSON_REQUEST_PAYLOAD)).build(); - } - } catch (IOException e) { - log.error("Invalid Payload received", e); - return Response.status(Response.Status.BAD_REQUEST). - entity(EventNotificationUtils. - getErrorDTO(EventNotificationEndPointConstants.INVALID_REQUEST_PAYLOAD, - EventNotificationEndPointConstants.REQUEST_PAYLOAD_ERROR)).build(); - } catch (ParseException e) { - return Response.status(Response.Status.BAD_REQUEST). - entity(EventNotificationUtils. - getErrorDTO(EventNotificationEndPointConstants.INVALID_REQUEST_PAYLOAD, - EventNotificationEndPointConstants.ERROR_PAYLOAD_PARSE)).build(); - } - - eventSubscriptionDTO.setSubscriptionId(uriInfo.getPathParameters().getFirst("subscriptionId")); - EventSubscriptionResponse eventSubscriptionResponse = eventSubscriptionServiceHandler. - updateEventSubscription(eventSubscriptionDTO); - return EventSubscriptionUtils.mapEventSubscriptionServiceResponse(eventSubscriptionResponse); - } - - /** - * Delete an Event Subscription. - */ - @SuppressFBWarnings({"JAXRS_ENDPOINT", "SERVLET_HEADER"}) - @DELETE - @Path("/{subscriptionId}") - @Consumes({"application/json; charset=utf-8"}) - @Produces({"application/json; charset=utf-8"}) - public Response deleteSubscription(@Context HttpServletRequest request, @Context HttpServletResponse response, - @Context UriInfo uriInfo) { - //check if the client id is present in the header - String clientId = request.getHeader(EventNotificationEndPointConstants.X_WSO2_CLIENT_ID); - if (StringUtils.isBlank(clientId)) { - return Response.status(Response.Status.BAD_REQUEST).entity(EventNotificationUtils. - getErrorDTO(EventNotificationEndPointConstants.MISSING_REQUEST_HEADER, - EventNotificationConstants.MISSING_HEADER_PARAM_CLIENT_ID)).build(); - } - - EventSubscriptionResponse eventSubscriptionResponse = eventSubscriptionServiceHandler. - deleteEventSubscription(clientId, uriInfo.getPathParameters().getFirst("subscriptionId")); - return EventSubscriptionUtils.mapEventSubscriptionServiceResponse(eventSubscriptionResponse); - } - -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/constants/EventNotificationEndPointConstants.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/constants/EventNotificationEndPointConstants.java deleted file mode 100644 index 90530c99..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/constants/EventNotificationEndPointConstants.java +++ /dev/null @@ -1,41 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.endpoint.constants; - -/** - * Constants in Endpoint. - */ -public class EventNotificationEndPointConstants { - public static final String X_WSO2_CLIENT_ID = "x-wso2-client_id"; - public static final String X_WSO2_RESOURCE_ID = "x-wso2-resource_id"; - public static final String REQUEST = "request"; - public static final String NOT_FOUND_RESPONSE = "No OPEN notifications founds for the given clientID"; - public static final String POLLING_ERROR_RESPONSE = "OB Event Notification Polling error"; - public static final String EVENT_CREATION_ERROR_RESPONSE = "OB Event Notification Creation error"; - public static final String REQUEST_PAYLOAD_ERROR = "Error in the request payload"; - public static final String EMPTY_REQ_PAYLOAD = "Request payload cannot be empty"; - public static final String INVALID_REQUEST = "invalid_request"; - public static final String INVALID_REQUEST_PAYLOAD = "invalid_request_payload"; - public static final String MISSING_REQUEST_PAYLOAD = "missing_request_payload"; - public static final String MISSING_JSON_REQUEST_PAYLOAD = "missing_Json_request_payload"; - public static final String INVALID_REQUEST_HEADER = "invalid_request_header"; - public static final String MISSING_REQUEST_HEADER = "missing_request_header"; - public static final String ERROR_PAYLOAD_PARSE = "Error while parsing payload"; - public static final String NOTIFICATIONS_NOT_FOUND = "notification_not_found"; -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/util/EventNotificationUtils.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/util/EventNotificationUtils.java deleted file mode 100644 index 9f7c038d..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/util/EventNotificationUtils.java +++ /dev/null @@ -1,142 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.endpoint.util; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.util.OpenBankingUtils; -import com.wso2.openbanking.accelerator.event.notifications.endpoint.constants.EventNotificationEndPointConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.dto.EventNotificationErrorDTO; -import com.wso2.openbanking.accelerator.event.notifications.service.handler.EventCreationServiceHandler; -import com.wso2.openbanking.accelerator.event.notifications.service.handler.EventPollingServiceHandler; -import com.wso2.openbanking.accelerator.event.notifications.service.response.EventCreationResponse; -import com.wso2.openbanking.accelerator.event.notifications.service.response.EventPollingResponse; -import org.apache.commons.logging.Log; -import org.apache.commons.logging.LogFactory; - -import javax.ws.rs.core.Response; - -/** - * This class will have util methods needed for event notifcations. - */ -public class EventNotificationUtils { - - private static final Log log = LogFactory.getLog(EventNotificationUtils.class); - - /** - * This method is to get the event creation service handler as per the config. - * @return - */ - public static EventCreationServiceHandler getEventNotificationCreationServiceHandler() { - - EventCreationServiceHandler eventCreationServiceHandler = (EventCreationServiceHandler) - OpenBankingUtils.getClassInstanceFromFQN(OpenBankingConfigParser.getInstance(). - getConfiguration().get(OpenBankingConstants.EVENT_CREATION_HANDLER).toString()); - - return eventCreationServiceHandler; - } - - /** - * This method is to get the event polling service handler as per the config. - * @return - */ - public static EventPollingServiceHandler getEventPollingServiceHandler() { - - EventPollingServiceHandler eventPollingServiceHandler = (EventPollingServiceHandler) - OpenBankingUtils.getClassInstanceFromFQN(OpenBankingConfigParser.getInstance(). - getConfiguration().get(OpenBankingConstants.EVENT_POLLING_HANDLER).toString()); - - return eventPollingServiceHandler; - } - - /** - * Method to map the Event Creation Service Response to API response. - * @param eventCreationResponse - * @return - */ - public static Response mapEventCreationServiceResponse(EventCreationResponse eventCreationResponse) { - - if (EventNotificationConstants.CREATED.equals(eventCreationResponse.getStatus())) { - - return Response.status(Response.Status.CREATED).entity(eventCreationResponse.getResponseBody()).build(); - - } else if (EventNotificationConstants.BAD_REQUEST.equals(eventCreationResponse.getStatus())) { - - return Response.status(Response.Status.BAD_REQUEST).entity(EventNotificationUtils.getErrorDTO( - EventNotificationEndPointConstants.INVALID_REQUEST, - eventCreationResponse.getErrorResponse())).build(); - } - - return Response.status(Response.Status.BAD_REQUEST).entity(getErrorDTO( - EventNotificationEndPointConstants.INVALID_REQUEST, - EventNotificationEndPointConstants.EVENT_CREATION_ERROR_RESPONSE)).build(); - } - - /** - * Method to map Event Polling Service to API response. - * @param eventPollingResponse - * @return - */ - public static Response mapEventPollingServiceResponse(EventPollingResponse eventPollingResponse) { - - if (EventNotificationConstants.OK.equals(eventPollingResponse.getStatus())) { - return Response.status(Response.Status.OK).entity(eventPollingResponse.getResponseBody()).build(); - } else if (EventNotificationConstants.NOT_FOUND.equals(eventPollingResponse.getStatus())) { - return Response.status(Response.Status.NOT_FOUND).entity(eventPollingResponse.getResponseBody()).build(); - } else { - if (eventPollingResponse.getErrorResponse() instanceof String) { - return Response.status(getErrorResponseStatus(eventPollingResponse.getStatus())) - .entity(EventNotificationUtils.getErrorDTO(EventNotificationEndPointConstants.INVALID_REQUEST, - eventPollingResponse.getErrorResponse().toString())).build(); - } else { - return Response.status(getErrorResponseStatus(eventPollingResponse.getStatus())) - .entity(eventPollingResponse.getErrorResponse()) - .build(); - } - } - } - - /** - * Method to map Event Polling Service error to API response. - * @return EventNotificationErrorDTO - */ - public static EventNotificationErrorDTO getErrorDTO(String error, String errorDescription) { - EventNotificationErrorDTO eventNotificationErrorDTO = new EventNotificationErrorDTO(); - eventNotificationErrorDTO.setError(error); - eventNotificationErrorDTO.setErrorDescription(errorDescription); - return eventNotificationErrorDTO; - } - - /** - * Get mapped Response.Status for the given status value. - * @param status status value - * @return Mapped Response.Status - */ - private static Response.Status getErrorResponseStatus(String status) { - - if (EventNotificationConstants.NOT_FOUND.equals(status)) { - return Response.Status.NOT_FOUND; - } else if (EventNotificationConstants.BAD_REQUEST.equals(status)) { - return Response.Status.BAD_REQUEST; - } else { - return Response.Status.INTERNAL_SERVER_ERROR; - } - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/util/EventSubscriptionUtils.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/util/EventSubscriptionUtils.java deleted file mode 100644 index 0b621b44..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/java/com/wso2/openbanking/accelerator/event/notifications/endpoint/util/EventSubscriptionUtils.java +++ /dev/null @@ -1,99 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.event.notifications.endpoint.util; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.common.util.OpenBankingUtils; -import com.wso2.openbanking.accelerator.event.notifications.service.constants.EventNotificationConstants; -import com.wso2.openbanking.accelerator.event.notifications.service.handler.EventSubscriptionServiceHandler; -import com.wso2.openbanking.accelerator.event.notifications.service.response.EventSubscriptionResponse; -import com.wso2.openbanking.accelerator.event.notifications.service.util.EventNotificationServiceUtil; -import net.minidev.json.JSONObject; -import net.minidev.json.parser.JSONParser; -import net.minidev.json.parser.ParseException; -import org.apache.commons.io.IOUtils; -import org.apache.http.HttpStatus; - -import java.io.IOException; - -import javax.servlet.http.HttpServletRequest; -import javax.ws.rs.core.Response; - - -/** - * Events Notification Subscription API Utils. - */ -public class EventSubscriptionUtils { - - /** - * Extract string payload from request object. - */ - public static EventSubscriptionServiceHandler getEventSubscriptionServiceHandler() { - - EventSubscriptionServiceHandler eventSubscriptionServiceHandler = (EventSubscriptionServiceHandler) - OpenBankingUtils.getClassInstanceFromFQN(OpenBankingConfigParser.getInstance().getConfiguration(). - get(OpenBankingConstants.EVENT_SUBSCRIPTION_HANDLER).toString()); - return eventSubscriptionServiceHandler; - } - - /** - * Extract string payload from request object. - * - * @param request The request object - * @return String payload - */ - public static JSONObject getJSONObjectPayload(HttpServletRequest request) throws IOException, ParseException { - Object payload = new JSONParser(JSONParser.MODE_PERMISSIVE).parse(IOUtils. - toString(request.getInputStream())); - if (payload == null || !(payload instanceof JSONObject)) { - return null; - } - return (JSONObject) payload; - } - - /** - * Method to map the Event Creation Service Response to API response. - * - * @param eventSubscriptionResponse - * @return Response - */ - public static Response mapEventSubscriptionServiceResponse(EventSubscriptionResponse eventSubscriptionResponse) { - int status = eventSubscriptionResponse.getStatus(); - if (HttpStatus.SC_NO_CONTENT == status) { - return Response.status(status) - .build(); - } else if (eventSubscriptionResponse.getErrorResponse() == null) { - if (eventSubscriptionResponse.getResponseBody() != null) { - return Response.status(status) - .entity(eventSubscriptionResponse.getResponseBody()) - .build(); - } else { - return Response.status(HttpStatus.SC_INTERNAL_SERVER_ERROR) - .entity(EventNotificationServiceUtil.getErrorDTO(EventNotificationConstants.INVALID_REQUEST, - EventNotificationConstants.ERROR_HANDLING_EVENT_SUBSCRIPTION)) - .build(); - } - } else { - return Response.status(status) - .entity(eventSubscriptionResponse.getErrorResponse()) - .build(); - } - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/resources/findbugs-include.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/resources/findbugs-include.xml deleted file mode 100644 index 8932a22e..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/webapp/META-INF/webapp-classloading.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/webapp/META-INF/webapp-classloading.xml deleted file mode 100644 index b212826c..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/webapp/META-INF/webapp-classloading.xml +++ /dev/null @@ -1,35 +0,0 @@ - - - - - - - - - false - - - Carbon,CXF3 - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/webapp/WEB-INF/beans.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/webapp/WEB-INF/beans.xml deleted file mode 100644 index 97e65a79..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/webapp/WEB-INF/beans.xml +++ /dev/null @@ -1,34 +0,0 @@ - - - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/webapp/WEB-INF/web.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/webapp/WEB-INF/web.xml deleted file mode 100644 index 3f2f9dab..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint/src/main/webapp/WEB-INF/web.xml +++ /dev/null @@ -1,69 +0,0 @@ - - - - - - WSO2 Open Banking - Event Notifications API - WSO2 Open Banking - Event Notifications API - - - contextConfigLocation - WEB-INF/beans.xml - - - - HttpHeaderSecurityFilter - org.apache.catalina.filters.HttpHeaderSecurityFilter - - hstsEnabled - false - - - - - HttpHeaderSecurityFilter - * - - - - - org.springframework.web.context.ContextLoaderListener - - - - - CXFServlet - - org.apache.cxf.transport.servlet.CXFServlet - - 1 - - - - CXFServlet - /* - - - - 60 - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/pom.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/pom.xml deleted file mode 100644 index 79a0f7fb..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/pom.xml +++ /dev/null @@ -1,122 +0,0 @@ - - - - - 4.0.0 - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../../pom.xml - - - - com.wso2.openbanking.accelerator.push.authorization.endpoint - WSO2 Open Banking - Push Authorization - war - - - - io.swagger - swagger-jaxrs - - - javax.ws.rs - jsr311-api - - - com.google.guava - guava - - - org.yaml - snakeyaml - - - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.identity - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.runtime.identity.authn.filter - provided - - - org.wso2.carbon.identity.inbound.auth.oauth2 - org.wso2.carbon.identity.oauth.client.authn.filter - provided - - - com.fasterxml.jackson.core - jackson-databind - provided - - - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - true - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-include.xml - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - maven-war-plugin - ${maven-war-plugin.version} - - - - - src/main/webapp - - - api#openbanking#push-authorization - WEB-INF/lib/slf4j-api-*.jar - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/java/com/wso2/openbanking/accelerator/push/authorization/endpoint/api/PushAuthorisationEndpoint.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/java/com/wso2/openbanking/accelerator/push/authorization/endpoint/api/PushAuthorisationEndpoint.java deleted file mode 100644 index df7b39ee..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/java/com/wso2/openbanking/accelerator/push/authorization/endpoint/api/PushAuthorisationEndpoint.java +++ /dev/null @@ -1,185 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.push.authorization.endpoint.api; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigurationService; -import com.wso2.openbanking.accelerator.common.constant.OpenBankingConstants; -import com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.PushAuthRequestValidator; -import com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.exception.PushAuthRequestValidatorException; -import com.wso2.openbanking.accelerator.identity.push.auth.extension.request.validator.model.PushAuthErrorResponse; -import com.wso2.openbanking.accelerator.push.authorization.endpoint.model.PushAuthorisationResponse; -import com.wso2.openbanking.accelerator.runtime.identity.authn.filter.OBOAuthClientAuthenticatorProxy; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import org.apache.commons.lang3.RandomStringUtils; -import org.apache.commons.lang3.StringUtils; -import org.apache.cxf.interceptor.InInterceptors; -import org.apache.http.HttpStatus; -import org.wso2.carbon.context.PrivilegedCarbonContext; -import org.wso2.carbon.identity.oauth.cache.SessionDataCache; -import org.wso2.carbon.identity.oauth.cache.SessionDataCacheEntry; -import org.wso2.carbon.identity.oauth.cache.SessionDataCacheKey; -import org.wso2.carbon.identity.oauth2.bean.OAuthClientAuthnContext; -import org.wso2.carbon.identity.oauth2.model.OAuth2Parameters; - -import java.time.Instant; -import java.util.List; -import java.util.Map; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; -import javax.ws.rs.Consumes; -import javax.ws.rs.POST; -import javax.ws.rs.Path; -import javax.ws.rs.Produces; -import javax.ws.rs.core.Context; -import javax.ws.rs.core.MultivaluedMap; -import javax.ws.rs.core.Response; - -/** - * Pushed Authorization Requests API - * - *

This specification defines the pushed authorization request endpoint, which allows clients to push the payload of - * an OAuth 2.0 authorization request to the authorization server via a direct request and provides them with a request - * URI that is used as reference to the data in a subsequent authorization request. - * Life cycle : - * This endpoint creates and returns request_uri to the user. The request object is stored in IDN_AUTH_SESSION_STORE - * DB fronted by IS SessionDataCache. During auth call, request_uri is resolved using - * DefaultOBRequestUriRequestObjectBuilder extension implementation. - * Finally when the request_uri is used once, it should be removed from cache in consent-authorize-steps. - */ -@Path("/") -@InInterceptors(classes = OBOAuthClientAuthenticatorProxy.class) -public class PushAuthorisationEndpoint { - - private static final String REQUEST = "request"; - public static final String CLIENT_AUTHENTICATION_CONTEXT = "oauth.client.authentication.context"; - private OpenBankingConfigurationService openBankingConfigurationService; - - /** - * Push an OAuth authorisation request object in exchange for a request_uri. - *

- * Endpoint maybe secured with basic auth in base 64 - */ - @SuppressFBWarnings("JAXRS_ENDPOINT") - // Suppressed content - Endpoint - // Suppression reason - False Positive : This endpoint is secured with access control lists in the configuration - // Suppressed warning count - 1 - @POST - @Path("/par") - @Consumes({"application/x-www-form-urlencoded"}) - @Produces({"application/json"}) - public Response parPost(@Context HttpServletRequest request, @Context HttpServletResponse response, - MultivaluedMap parameterMap) { - - PushAuthRequestValidator pushAuthRequestValidator = PushAuthRequestValidator.getPushAuthRequestValidator(); - - Map paramMap; - String requestJWT = StringUtils.EMPTY; - - OAuthClientAuthnContext clientAuthnContext = (OAuthClientAuthnContext) - request.getAttribute(CLIENT_AUTHENTICATION_CONTEXT); - - // Check if the client authentication is successful - if (!clientAuthnContext.isAuthenticated()) { - // create error response - PushAuthErrorResponse errorResponse = pushAuthRequestValidator - .createErrorResponse(HttpServletResponse.SC_UNAUTHORIZED, - clientAuthnContext.getErrorCode(), clientAuthnContext.getErrorMessage()); - return Response.status(errorResponse.getHttpStatusCode()) - .entity(errorResponse.getPayload()).build(); - } - - try { - paramMap = pushAuthRequestValidator.validateParams(request, (Map>) parameterMap); - } catch (PushAuthRequestValidatorException exception) { - // create error response - PushAuthErrorResponse errorResponse = pushAuthRequestValidator - .createErrorResponse(exception.getHttpStatusCode(), exception.getErrorCode(), - exception.getErrorDescription()); - return Response.status(errorResponse.getHttpStatusCode() != 0 ? - errorResponse.getHttpStatusCode() : exception.getHttpStatusCode()) - .entity(errorResponse.getPayload()).build(); - } - - if (!paramMap.isEmpty() && paramMap.containsKey(REQUEST)) { - - requestJWT = paramMap.get(REQUEST).toString(); - } - - // Generate a urn with cryptographically strong pseudo random algorithm - String urn = RandomStringUtils.randomAlphanumeric(32); - - OpenBankingConfigurationService openBankingConfigurationService = getOBConfigService(); - - int expiryTime = Integer.parseInt(openBankingConfigurationService.getConfigurations() - .get(OpenBankingConstants.PUSH_AUTH_EXPIRY_TIME).toString()); - - // Add to auth cache - addToIdnOAuthCache(requestJWT, urn, expiryTime); - - return Response.status(HttpStatus.SC_CREATED) - .entity(getSuccessResponse("urn" + ":" + openBankingConfigurationService.getConfigurations() - .get(OpenBankingConstants.PUSH_AUTH_REQUEST_URI_SUBSTRING).toString() + ":" + urn, expiryTime)) - .build(); - - } - - /** - * Add Request Object to Session-Key Cache (Database). Validation is set as one minute. - */ - private static void addToIdnOAuthCache(String requestJWT, String sessionKey, int expiry) { - - SessionDataCacheKey cacheKey = new SessionDataCacheKey(sessionKey); - SessionDataCacheEntry sessionDataCacheEntry = new SessionDataCacheEntry(); - OAuth2Parameters oAuth2Parameters = new OAuth2Parameters(); - long expiryTimestamp = Instant.now().getEpochSecond() + expiry; - oAuth2Parameters.setEssentialClaims(requestJWT + ":" + expiryTimestamp); - sessionDataCacheEntry.setoAuth2Parameters(oAuth2Parameters); - - SessionDataCache.getInstance().addToCache(cacheKey, sessionDataCacheEntry); - } - - /** - * Create success response. - */ - private static PushAuthorisationResponse getSuccessResponse(String requestUri, int expiry) { - - PushAuthorisationResponse pushAuthorisationResponse = new PushAuthorisationResponse(); - - pushAuthorisationResponse.setRequestUri(requestUri); - pushAuthorisationResponse.setExpiresIn(expiry); - return pushAuthorisationResponse; - } - - /** - * Retrieve Open Banking configuration service. - */ - private OpenBankingConfigurationService getOBConfigService() { - - if (this.openBankingConfigurationService == null) { - OpenBankingConfigurationService openBankingConfigurationService = - (OpenBankingConfigurationService) PrivilegedCarbonContext.getThreadLocalCarbonContext() - .getOSGiService(OpenBankingConfigurationService.class, null); - if (openBankingConfigurationService != null) { - this.openBankingConfigurationService = openBankingConfigurationService; - } - } - return this.openBankingConfigurationService; - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/java/com/wso2/openbanking/accelerator/push/authorization/endpoint/model/PushAuthorisationResponse.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/java/com/wso2/openbanking/accelerator/push/authorization/endpoint/model/PushAuthorisationResponse.java deleted file mode 100644 index 09270eb1..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/java/com/wso2/openbanking/accelerator/push/authorization/endpoint/model/PushAuthorisationResponse.java +++ /dev/null @@ -1,86 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.push.authorization.endpoint.model; - -import com.fasterxml.jackson.annotation.JsonProperty; - -/** - * Model class for push authorisation response. - */ -public class PushAuthorisationResponse { - - private String requestUri = null; - private Integer expiresIn = null; - - @JsonProperty("request_uri") - public String getRequest_uri() { - - return requestUri; - } - - public void setRequestUri(String requestUri) { - - this.requestUri = requestUri; - } - - public PushAuthorisationResponse requestUri(String requestUri) { - - this.requestUri = requestUri; - return this; - } - - @JsonProperty("expires_in") - public Integer getExpires_in() { - - return expiresIn; - } - - public void setExpiresIn(Integer expiresIn) { - - this.expiresIn = expiresIn; - } - - public PushAuthorisationResponse expiresIn(Integer expiresIn) { - - this.expiresIn = expiresIn; - return this; - } - - @Override - public String toString() { - StringBuilder sb = new StringBuilder(); - sb.append("class PushAuthorisationResponse {\n"); - - sb.append(" requestUri: ").append(toIndentedString(requestUri)).append("\n"); - sb.append(" expiresIn: ").append(toIndentedString(expiresIn)).append("\n"); - sb.append("}"); - return sb.toString(); - } - - /** - * Convert the given object to string with each line indented by 4 spaces - * (except the first line). - */ - private static String toIndentedString(Object object) { - if (object == null) { - return "null"; - } - return object.toString().replace("\n", "\n "); - } -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/resources/findbugs-include.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/resources/findbugs-include.xml deleted file mode 100644 index 8932a22e..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/webapp/META-INF/MANIFEST.mf b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/webapp/META-INF/MANIFEST.mf deleted file mode 100644 index 9d885be5..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/webapp/META-INF/MANIFEST.mf +++ /dev/null @@ -1 +0,0 @@ -Manifest-Version: 1.0 diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/webapp/META-INF/webapp-classloading.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/webapp/META-INF/webapp-classloading.xml deleted file mode 100644 index b212826c..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/webapp/META-INF/webapp-classloading.xml +++ /dev/null @@ -1,35 +0,0 @@ - - - - - - - - - false - - - Carbon,CXF3 - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/webapp/WEB-INF/beans.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/webapp/WEB-INF/beans.xml deleted file mode 100644 index ef5176c5..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/webapp/WEB-INF/beans.xml +++ /dev/null @@ -1,34 +0,0 @@ - - - - - - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/webapp/WEB-INF/web.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/webapp/WEB-INF/web.xml deleted file mode 100644 index baf0a570..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint/src/main/webapp/WEB-INF/web.xml +++ /dev/null @@ -1,69 +0,0 @@ - - - - - - WSO2 Open Banking - Push Auth API - WSO2 Open Banking - Push Auth API - - - contextConfigLocation - WEB-INF/beans.xml - - - - HttpHeaderSecurityFilter - org.apache.catalina.filters.HttpHeaderSecurityFilter - - hstsEnabled - false - - - - - HttpHeaderSecurityFilter - * - - - - - org.springframework.web.context.ContextLoaderListener - - - - - CXFServlet - - org.apache.cxf.transport.servlet.CXFServlet - - 1 - - - - CXFServlet - /* - - - - 60 - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/pom.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/pom.xml deleted file mode 100644 index 3f5c2a2c..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/pom.xml +++ /dev/null @@ -1,190 +0,0 @@ - - - - - - open-banking-accelerator - com.wso2.openbanking.accelerator - 3.2.11-SNAPSHOT - ../../../pom.xml - - 4.0.0 - - com.wso2.openbanking.authentication.webapp - war - WSO2 Open Banking - Authentication Webapp - - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.identity - provided - - - com.wso2.openbanking.accelerator - com.wso2.openbanking.accelerator.consent.extensions - provided - - - javax.servlet - jstl - - - org.testng - testng - test - - - org.mockito - mockito-all - test - - - org.jacoco - org.jacoco.agent - runtime - test - - - org.springframework - spring-test - test - - - org.springframework - spring-core - provided - - - org.wso2.orbit.org.owasp.encoder - encoder - ${encoder.wso2.version} - - - - - - maven-war-plugin - ${maven-war-plugin.version} - - - - - src/main/resources/ - - - ob#authenticationendpoint - - - - org.apache.maven.plugins - maven-compiler-plugin - - - org.jacoco - jacoco-maven-plugin - - - **/*Constants.class - - - - - default-prepare-agent - - prepare-agent - - - - default-prepare-agent-integration - - prepare-agent-integration - - - - default-report - - report - - - - default-report-integration - - report-integration - - - - default-check - - check - - - - - **/*OBDefaultAuthServletImpl.class - - - - BUNDLE - - - INSTRUCTION - COVEREDRATIO - - - - - - - - - - - com.github.spotbugs - spotbugs-maven-plugin - - Max - Low - true - false - ${project.build.directory}/spotbugs - ${project.basedir}/src/main/resources/findbugs-exclude.xml - ${project.basedir}/src/main/resources/findbugs-include.xml - false - - - com.h3xstream.findsecbugs - findsecbugs-plugin - ${com.h3xstream.findsecbugs.version} - - - - - - analyze-compile - compile - - check - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/java/com/wso2/openbanking/accelerator/authentication/webapp/OBConsentConfirmServlet.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/java/com/wso2/openbanking/accelerator/authentication/webapp/OBConsentConfirmServlet.java deleted file mode 100644 index 4b4edf01..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/java/com/wso2/openbanking/accelerator/authentication/webapp/OBConsentConfirmServlet.java +++ /dev/null @@ -1,155 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.authentication.webapp; - -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.consent.extensions.authservlet.model.OBAuthServletInterface; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import org.apache.commons.lang.StringUtils; -import org.apache.http.HttpResponse; -import org.apache.http.client.methods.HttpPatch; -import org.apache.http.entity.ContentType; -import org.apache.http.entity.StringEntity; -import org.apache.http.impl.client.CloseableHttpClient; -import org.apache.http.impl.client.HttpClientBuilder; -import org.json.JSONObject; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import java.io.IOException; -import java.lang.reflect.InvocationTargetException; -import java.net.HttpURLConnection; -import java.util.HashMap; -import java.util.Map; - -import javax.servlet.ServletContext; -import javax.servlet.http.Cookie; -import javax.servlet.http.HttpServlet; -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; -import javax.servlet.http.HttpSession; - -/** - * The servlet responsible for the confirm page in auth web flow. - */ -public class OBConsentConfirmServlet extends HttpServlet { - - static OBAuthServletInterface obAuthServletTK; - private static final long serialVersionUID = 6106269597832678046L; - private static Logger log = LoggerFactory.getLogger(OBConsentConfirmServlet.class); - - @SuppressFBWarnings("COOKIE_USAGE") - // Suppressed content - browserCookies.put(cookie.getName(), cookie.getValue()) - // Suppression reason - False Positive : The cookie values are only read and here. No sensitive info is added to - // the cookie in this step. - // Suppressed warning count - 1 - public void doPost(HttpServletRequest request, HttpServletResponse response) throws IOException { - - setAuthExtension(); - - HttpSession session = request.getSession(); - Map metadata = new HashMap<>(); - Map browserCookies = new HashMap<>(); - JSONObject consentData = new JSONObject(); - - //retrieve commonAuthId to be stored for co-relation of consent Id and access token issued - Cookie[] cookies = request.getCookies(); - for (Cookie cookie : cookies) { - browserCookies.put(cookie.getName(), cookie.getValue()); - } - consentData.put("cookies", browserCookies); - - // Add authorisationId if available - String authorisationId = request.getParameter("authorisationId"); - if (StringUtils.isNotEmpty(authorisationId)) { - metadata.put("authorisationId", authorisationId); - } - - consentData.put("type", request.getParameter("type")); - consentData.put("approval", request.getParameter("consent")); - consentData.put("userId", session.getAttribute("username")); - - // add TK data - if (obAuthServletTK != null) { - Map updatedMetadata = obAuthServletTK.updateConsentMetaData(request); - if (updatedMetadata != null) { - updatedMetadata.forEach(metadata::put); - } - - Map updatedConsentData = obAuthServletTK.updateConsentData(request); - if (updatedConsentData != null) { - updatedConsentData.forEach(consentData::put); - } - } - - consentData.put("metadata", metadata); - - String redirectURL = persistConsentData( - consentData, request.getParameter("sessionDataKeyConsent"), getServletContext()); - - // Invoke authorize flow - if (redirectURL != null) { - response.sendRedirect(redirectURL); - - } else { - session.invalidate(); - response.sendRedirect("retry.do?status=Error&statusMsg=Error while persisting consent"); - } - - } - - @Generated(message = "Contains the tested code of HTTPClient") - String persistConsentData(JSONObject consentData, String sessionDataKey, ServletContext servletContext) { - - String persistenceBaseURL = servletContext.getInitParameter("persistenceBaseURL"); - String persistenceUrl = persistenceBaseURL + "/" + sessionDataKey; - - try (CloseableHttpClient client = HttpClientBuilder.create().build()) { - HttpPatch dataRequest = new HttpPatch(persistenceUrl); - dataRequest.addHeader("accept", "application/json"); - dataRequest.addHeader("Authorization", "Basic " + OBConsentServlet.getConsentApiCredentials()); - StringEntity body = new StringEntity(consentData.toString(), ContentType.APPLICATION_JSON); - dataRequest.setEntity(body); - HttpResponse dataResponse = client.execute(dataRequest); - - if (dataResponse.getStatusLine().getStatusCode() != HttpURLConnection.HTTP_MOVED_TEMP) { - return null; - } else { - return dataResponse.getLastHeader("Location").getValue(); - } - } catch (IOException e) { - log.error("Exception while calling persistence endpoint", e); - return null; - } - } - /** - * Retrieve the config. - */ - void setAuthExtension() { - try { - obAuthServletTK = (OBAuthServletInterface) Class.forName(OpenBankingConfigParser.getInstance() - .getAuthServletExtension()).getDeclaredConstructor().newInstance(); - } catch (InstantiationException | IllegalAccessException | - InvocationTargetException | NoSuchMethodException | ClassNotFoundException e) { - log.error("Webapp extension not found", e); - } - } - -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/java/com/wso2/openbanking/accelerator/authentication/webapp/OBConsentServlet.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/java/com/wso2/openbanking/accelerator/authentication/webapp/OBConsentServlet.java deleted file mode 100644 index 0bbafe9d..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/java/com/wso2/openbanking/accelerator/authentication/webapp/OBConsentServlet.java +++ /dev/null @@ -1,290 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.authentication.webapp; - -import com.wso2.openbanking.accelerator.authentication.webapp.util.Constants; -import com.wso2.openbanking.accelerator.common.config.OpenBankingConfigParser; -import com.wso2.openbanking.accelerator.common.util.Generated; -import com.wso2.openbanking.accelerator.consent.extensions.authservlet.impl.ConsentMgrAuthServletImpl; -import com.wso2.openbanking.accelerator.consent.extensions.authservlet.impl.ISDefaultAuthServletImpl; -import com.wso2.openbanking.accelerator.consent.extensions.authservlet.model.OBAuthServletInterface; -import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; -import org.apache.commons.io.IOUtils; -import org.apache.commons.lang.StringUtils; -import org.apache.http.HttpResponse; -import org.apache.http.client.methods.HttpGet; -import org.apache.http.impl.client.CloseableHttpClient; -import org.apache.http.impl.client.HttpClientBuilder; -import org.json.JSONObject; -import org.owasp.encoder.Encode; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import java.io.IOException; -import java.io.InputStream; -import java.io.UnsupportedEncodingException; -import java.lang.reflect.InvocationTargetException; -import java.net.HttpURLConnection; -import java.net.URI; -import java.net.URISyntaxException; -import java.net.URLEncoder; -import java.nio.charset.StandardCharsets; -import java.util.Base64; -import java.util.Locale; -import java.util.Map; -import java.util.Properties; -import java.util.ResourceBundle; - -import javax.servlet.RequestDispatcher; -import javax.servlet.ServletContext; -import javax.servlet.ServletException; -import javax.servlet.http.HttpServlet; -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; -import javax.servlet.http.HttpSession; - -import static com.wso2.openbanking.accelerator.consent.extensions.authservlet.impl.util.Utils.i18n; - -/** - * The servlet responsible for displaying the consent details in the auth UI flow. - */ -public class OBConsentServlet extends HttpServlet { - - static OBAuthServletInterface obAuthServletTK; - private static final long serialVersionUID = 6106269076132678046L; - private static Logger log = LoggerFactory.getLogger(OBConsentServlet.class); - private static final String BUNDLE = "com.wso2.openbanking.authentication.webapp.i18n"; - - @SuppressFBWarnings({"REQUESTDISPATCHER_FILE_DISCLOSURE", "TRUST_BOUNDARY_VIOLATION"}) - // Suppressed content - obAuthServlet.getJSPPath() - // Suppression reason - False Positive : JSP path is hard coded and does not accept any user inputs, therefore it - // can be trusted - // Suppressed content - Encode.forJava(sessionDataKey) - // Suppression reason - False positive : sessionDataKey is encoded for Java which escapes untrusted characters - // Suppressed warning count - 2 - @Override - public void doGet(HttpServletRequest originalRequest, HttpServletResponse response) - throws IOException, ServletException { - HttpServletRequest request = originalRequest; - setAuthExtension(); - - // get consent data - String sessionDataKey = request.getParameter("sessionDataKeyConsent"); - HttpResponse consentDataResponse = getConsentDataWithKey(sessionDataKey, getServletContext()); - JSONObject dataSet = new JSONObject(); - log.debug("HTTP response for consent retrieval" + consentDataResponse.toString()); - try { - if (consentDataResponse.getStatusLine().getStatusCode() == HttpURLConnection.HTTP_MOVED_TEMP && - consentDataResponse.getLastHeader("Location") != null) { - response.sendRedirect(consentDataResponse.getLastHeader("Location").getValue()); - return; - } else { - String retrievalResponse = IOUtils.toString(consentDataResponse.getEntity().getContent(), - String.valueOf(StandardCharsets.UTF_8)); - JSONObject data = new JSONObject(retrievalResponse); - String errorResponse = getErrorResponseForRedirectURL(data); - if (data.has(Constants.REDIRECT_URI) && StringUtils.isNotEmpty(errorResponse)) { - URI errorURI = new URI(data.get(Constants.REDIRECT_URI).toString().concat(errorResponse)); - response.sendRedirect(errorURI.toString()); - return; - } else { - dataSet = createConsentDataset(data, consentDataResponse.getStatusLine().getStatusCode()); - } - } - } catch (IOException e) { - dataSet.put(Constants.IS_ERROR, "Exception occurred while retrieving consent data"); - } catch (URISyntaxException e) { - dataSet.put(Constants.IS_ERROR, "Error while constructing URI for redirection"); - - } - if (dataSet.has(Constants.IS_ERROR)) { - String isError = (String) dataSet.get(Constants.IS_ERROR); - request.getSession().invalidate(); - response.sendRedirect("retry.do?status=Error&statusMsg=" + isError); - return; - } - - // set variables to session - HttpSession session = request.getSession(); - - session.setAttribute(Constants.SESSION_DATA_KEY_CONSENT, Encode.forJava(sessionDataKey)); - session.setAttribute("displayScopes", - Boolean.parseBoolean(getServletContext().getInitParameter("displayScopes"))); - - // set strings to request - ResourceBundle resourceBundle = getResourceBundle(request.getLocale()); - - originalRequest.setAttribute("privacyDescription", i18n(resourceBundle, - "privacy.policy.privacy.short.description.approving")); - originalRequest.setAttribute("privacyGeneral", i18n(resourceBundle, "privacy.policy.general")); - - // bottom.jsp - originalRequest.setAttribute("ok", i18n(resourceBundle, "ok")); - originalRequest.setAttribute("requestedScopes", i18n(resourceBundle, "requested.scopes")); - - originalRequest.setAttribute("app", dataSet.getString("application")); - - // Get servlet extension - OBAuthServletInterface obAuthServlet; - if (Constants.DEFAULT.equals(dataSet.getString("type"))) { - // get default auth servlet extension - obAuthServlet = new ISDefaultAuthServletImpl(); - } else if (Constants.CONSENT_MGT.equals(dataSet.getString("type"))) { - // get consent manager auth servlet extension - obAuthServlet = new ConsentMgrAuthServletImpl(); - } else { - // get auth servlet toolkit implementation - if (obAuthServletTK == null) { - request.getSession().invalidate(); - response.sendRedirect("retry.do?status=Error&statusMsg=Error while processing request"); - log.error("Unable to find OB auth servlet extension implementation. Returning error."); - return; - } - obAuthServlet = obAuthServletTK; - } - - Map updatedValues; - - updatedValues = obAuthServlet.updateRequestAttribute(request, dataSet, resourceBundle); - updatedValues.forEach(originalRequest::setAttribute); - - // update session - updatedValues = obAuthServlet.updateSessionAttribute(request, dataSet, resourceBundle); - updatedValues.forEach(originalRequest.getSession()::setAttribute); - - // dispatch - RequestDispatcher dispatcher = this.getServletContext().getRequestDispatcher(obAuthServlet.getJSPPath()); - dispatcher.forward(originalRequest, response); - - } - - HttpResponse getConsentDataWithKey(String sessionDataKeyConsent, ServletContext servletContext) throws IOException { - - String retrievalBaseURL = servletContext.getInitParameter("retrievalBaseURL"); - String retrieveUrl = (retrievalBaseURL.endsWith("/")) ? retrievalBaseURL + sessionDataKeyConsent : - retrievalBaseURL + "/" + sessionDataKeyConsent; - - CloseableHttpClient client = HttpClientBuilder.create().build(); - HttpGet dataRequest = new HttpGet(retrieveUrl); - dataRequest.addHeader("Authorization", "Basic " + getConsentApiCredentials()); - HttpResponse dataResponse = client.execute(dataRequest); - - return dataResponse; - - } - - JSONObject createConsentDataset(JSONObject consentResponse, int statusCode) throws IOException { - - JSONObject errorObject = new JSONObject(); - if (statusCode != HttpURLConnection.HTTP_OK) { - if (statusCode == HttpURLConnection.HTTP_UNAUTHORIZED) { - if (consentResponse.has("description")) { - errorObject.put(Constants.IS_ERROR, consentResponse.get("description")); - } - } else { - errorObject.put(Constants.IS_ERROR, "Retrieving consent data failed"); - } - return errorObject; - } else { - return consentResponse; - } - } - - /** - * Retrieve the config. - */ - void setAuthExtension() { - - try { - obAuthServletTK = (OBAuthServletInterface) Class.forName(OpenBankingConfigParser.getInstance(). - getAuthServletExtension()).getDeclaredConstructor().newInstance(); - } catch (InstantiationException | IllegalAccessException | - InvocationTargetException | NoSuchMethodException | ClassNotFoundException e) { - log.error("Webapp extension not found", e); - } - } - - @Generated(message = "Encapsulated method for unit test") - ResourceBundle getResourceBundle(Locale locale) { - - return ResourceBundle.getBundle(BUNDLE, locale); - } - - /** - * @param data error response received from consent data retrieval endpoint - * @return formatted error response to be send to call back uri - */ - String getErrorResponseForRedirectURL(JSONObject data) { - - String errorResponse = ""; - try { - if (data.has(Constants.ERROR)) { - errorResponse = errorResponse.concat(Constants.ERROR_URI_FRAGMENT) - .concat(URLEncoder.encode(data.get(Constants.ERROR).toString(), - StandardCharsets.UTF_8.toString())); - } - if (data.has(Constants.ERROR_DESCRIPTION)) { - errorResponse = errorResponse.concat(Constants.ERROR_DESCRIPTION_PARAMETER) - .concat(URLEncoder.encode(data.get(Constants.ERROR_DESCRIPTION).toString(), - StandardCharsets.UTF_8.toString())); - } - if (data.has(Constants.STATE)) { - errorResponse = errorResponse.concat(Constants.STATE_PARAMETER) - .concat(URLEncoder.encode(data.get(Constants.STATE).toString(), - StandardCharsets.UTF_8.toString())); - } - - } catch (UnsupportedEncodingException e) { - log.error("Error while building error response", e); - } - return errorResponse; - } - - /** - * Retrieve admin credentials in Base64 format from webapp properties or OB configs. - */ - static String getConsentApiCredentials () { - String username, password; - try { - InputStream configurations = OBConsentConfirmServlet.class.getClassLoader() - .getResourceAsStream(Constants.CONFIG_FILE_NAME); - Properties configurationProperties = new Properties(); - configurationProperties.load(configurations); - Boolean isConfiguredInWebapp = Boolean.parseBoolean( - configurationProperties.getProperty(Constants.LOCATION_OF_CREDENTIALS)); - if (!isConfiguredInWebapp) { - username = (String) OpenBankingConfigParser.getInstance().getConfiguration() - .get(Constants.USERNAME_IN_OB_CONFIGS); - password = (String) OpenBankingConfigParser.getInstance().getConfiguration() - .get(Constants.PASSWORD_IN_OB_CONFIGS); - } else { - username = configurationProperties.getProperty(Constants.USERNAME_IN_WEBAPP_CONFIGS); - password = configurationProperties.getProperty(Constants.PASSWORD_IN_WEBAPP_CONFIGS); - } - } catch (IOException | NullPointerException e) { - log.error("Error occurred while reading the webapp properties file. Therefore using OB configurations."); - username = (String) OpenBankingConfigParser.getInstance().getConfiguration() - .get(Constants.USERNAME_IN_OB_CONFIGS); - password = (String) OpenBankingConfigParser.getInstance().getConfiguration() - .get(Constants.PASSWORD_IN_OB_CONFIGS); - } - return Base64.getEncoder().encodeToString((username + ":" + password).getBytes(StandardCharsets.UTF_8)); - } - -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/java/com/wso2/openbanking/accelerator/authentication/webapp/util/Constants.java b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/java/com/wso2/openbanking/accelerator/authentication/webapp/util/Constants.java deleted file mode 100644 index 8eb1fe56..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/java/com/wso2/openbanking/accelerator/authentication/webapp/util/Constants.java +++ /dev/null @@ -1,49 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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 com.wso2.openbanking.accelerator.authentication.webapp.util; - -/** - * Constants required for auth webapp. - */ -public class Constants { - - public static final String IS_ERROR = "isError"; - public static final String REQUESTED_CLAIMS = "requestedClaims"; - public static final String MANDATORY_CLAIMS = "mandatoryClaims"; - public static final String CLAIM_SEPARATOR = ","; - public static final String USER_CLAIMS_CONSENT_ONLY = "userClaimsConsentOnly"; - public static final String SESSION_DATA_KEY_CONSENT = "sessionDataKeyConsent"; - public static final String DEFAULT = "default"; - public static final String CONSENT_MGT = "consentmgt"; - public static final String COMMONAUTH_ID = "commonAuthId"; - public static final String OIDC_SCOPES = "OIDCScopes"; - public static final String REDIRECT_URI = "redirect_uri"; - public static final String ERROR = "error"; - public static final String ERROR_DESCRIPTION = "error_description"; - public static final String STATE = "state"; - public static final String ERROR_URI_FRAGMENT = "#error="; - public static final String ERROR_DESCRIPTION_PARAMETER = "&error_description="; - public static final String STATE_PARAMETER = "&state="; - public static final String CONFIG_FILE_NAME = "configurations.properties"; - public static final String LOCATION_OF_CREDENTIALS = "ConsentAPICredentials.IsConfiguredInWebapp"; - public static final String USERNAME_IN_WEBAPP_CONFIGS = "ConsentAPICredentials.Username"; - public static final String PASSWORD_IN_WEBAPP_CONFIGS = "ConsentAPICredentials.Password"; - public static final String USERNAME_IN_OB_CONFIGS = "Consent.ConsentAPICredentials.Username"; - public static final String PASSWORD_IN_OB_CONFIGS = "Consent.ConsentAPICredentials.Password"; -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/resources/com/wso2/openbanking/authentication/webapp/i18n.properties b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/resources/com/wso2/openbanking/authentication/webapp/i18n.properties deleted file mode 100644 index ad04e6c8..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/resources/com/wso2/openbanking/authentication/webapp/i18n.properties +++ /dev/null @@ -1,125 +0,0 @@ - # Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - # - # WSO2 LLC. 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. - -login=Sign In -username=User Name -password=Password -login.fail.message=Login failed! Please recheck the username and password and try again. -recaptcha.fail.message=reCaptcha validation is required for user. -account.confirmation.pending=Account is unverified. An account activation link has been sent to your registered email address, please check your inbox. -password.reset.pending=Password reset is required. A password reset link has been sent to your registered email address, please check your inbox. -account.resend.email.success=Email sent successfully. -account.resend.email.fail=Email sent fail. -user.tenant.domain.mismatch.message=Application you are trying to access does not allow users from your organization. -remember.me=Remember me on this computer -signin.to.authenticate1=Please sign in to authenticate to -signin.to.authenticate2=as -profile=Profile : -cancel=Cancel -approve=Approve -deny=Deny -approve.always=Approve Always -request.access.scope=requests access to -request.access.profile=requests access to your profile information -saml.sso=SAML 2.0 based Single Sign-On -tenantListLabel=Tenant -select.tenant.dropdown.display.name=Select Tenant -super.tenant.display.name=Super Tenant -super.tenant=carbon.super -domain.unknown=domain.unknown -confirm.password=Confirm Password -email=Email -continue=Continue -forgot.username.password=Forgot -forgot.username.password.or=or -forgot.password=Password -forgot.username=Username -no.account=Don't have an account? -have.account=Already have an account? -register.now=Register Now -register=Register -no.confirmation.mail=Not received confirmation email? -resend.mail=Re-Send -openid=Open ID -openid.user.claims=OpenID User Claims -username.or.password.invalid=Username or Password is Invalid -create.an.account=Create an account -unauthorized.to.login=You are not authorized to login -domain.cannot.be.identified=Domain cannot be identified! Please retry. -wso2.open.banking=WSO2 Open Banking -open.banking=Open Banking -domain=Domain -submit=Submit -inc=Inc -all.rights.reserved=All rights reserved -verification=Verification -touch.your.u2f.device=Touch your U2F device to Proceed -authentication.error=Authentication Error! -something.went.wrong.during.authentication=Something went wrong during the authentication process.Please try signing in again. -attention=Attention -provide.mandatory.details=Provide Mandatory Details -requested.claims.recommendation= application,that you are trying to login to needs following information filled in the user profile. You can fill those below and proceed with the authentication. But it is advised to fill these information in your Identity Provider profile in order to avoid this step every time you login -logged.out=You have successfully logged out. -authorize=Authorize -invalid.request=Invalid Request -oauth.processing.error.msg=OAuth Processing Error Message -openid.connect.logout=OpenID connect logout -do.you.want.to.logout=Do you want to logout? -yes=Yes -no=No -openid2.profile=OpenID2.0 Profile -claim.uri=Claim URI -claim.value=Claim Value -internal.error.occurred=Internal Error Occurred -information=Information -user.details.submitted=User details successfully submitted -close=Close -other.login.options=Other login options -sign.in.with=Sign In With -domain.name=Domain Name -go=Go -please.select.recaptcha=Please select recaptcha -error.when.processing.authentication.request=Error when processing authentication request! -please.try.login.again=Please try login again! -you.are.redirected.back.to=You are now redirected back to, -if.the.redirection.fails.please.click=If the redirection fails, please click the -post=POST -enter.required.fields.to.complete.registration=Enter required fields to complete registration -first.name=First Name -last.name=Last Name -password.mismatch=Passwords did not match. Please try again -user.exists=User already exist -unknown.error=Unknown error occurred -authentication.failed.please.retry=Authentication Failed! Please Retry -user.consents=User Consents -mandatory.claims.recommendation=Mandatory claims are marked with an asterisk -mandatory.claims.warning.msg.1=You need to provide consent for -mandatory.claims.warning.msg.2=all the mandatory claims -mandatory.claims.warning.msg.3=in order to proceed -privacy.policy.cookies=Cookie Policy -privacy.policy.cookies.short.description=After a successful sign in, we use a cookie in your browser to track your session. You can refer our -privacy.policy.general=Privacy Policy -privacy.policy.privacy.short.description=By signing in, you agree to our -privacy.policy.privacy.short.description.approving=By approving, you agree to our -privacy.policy.for.more.details=for more details. -under.construction=This page is under construction -by.selecting.following.attributes=By selecting following attributes I agree to share them with the above service provider. -select.all=Select All -requested.scopes=Requested scopes -requested.attributes=Requested attributes -please.select.approve.always=Please select either "Approve Always" or "Approve" to provide consent to requested scopes to continue -ok=Ok -mandatory.claims=Mandatory claims diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/resources/configurations.properties b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/resources/configurations.properties deleted file mode 100644 index c118c4de..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/resources/configurations.properties +++ /dev/null @@ -1,19 +0,0 @@ -# Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). -# -# WSO2 LLC. 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. - -ConsentAPICredentials.IsConfiguredInWebapp = false -ConsentAPICredentials.Username = null -ConsentAPICredentials.Password = null diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/resources/findbugs-exclude.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/resources/findbugs-exclude.xml deleted file mode 100644 index d7392112..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/resources/findbugs-exclude.xml +++ /dev/null @@ -1,45 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/resources/findbugs-include.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/resources/findbugs-include.xml deleted file mode 100644 index 8932a22e..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/resources/findbugs-include.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/WEB-INF/web.xml b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/WEB-INF/web.xml deleted file mode 100644 index 59a41b2f..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/WEB-INF/web.xml +++ /dev/null @@ -1,104 +0,0 @@ - - - - - - - OBConsentServlet - com.wso2.openbanking.accelerator.authentication.webapp.OBConsentServlet - - - - OBConsentServlet - /oauth2_authz.do - - - - OBConsentServlet - /oauth2_consent.do - - - - retrievalBaseURL - https://localhost:9446/api/openbanking/consent/authorize/retrieve - - - - displayScopes - true - - - - - - OBConsentConfirmServlet - com.wso2.openbanking.accelerator.authentication.webapp.OBConsentConfirmServlet - - - - OBConsentConfirmServlet - /oauth2_authz_confirm.do - - - - persistenceBaseURL - https://localhost:9446/api/openbanking/consent/authorize/persist - - - - - - cookie_policy.do - /cookie_policy.jsp - - - - cookie_policy.do - /cookie_policy.do - - - - - - privacy_policy.do - /privacy_policy.jsp - - - - privacy_policy.do - /privacy_policy.do - - - - - - java.lang.Throwable - /generic-exception-response.jsp - - - - - - oauth2_authz_consent.do - /oauth2_authz_consent.do - - - - retry.do - /generic-exception-response.jsp - - - - oauth2_authz_consent.do - /oauth2_authz_displayconsent.jsp - - - - retry.do - /retry.do - - - \ No newline at end of file diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/assets/img/glyphicons-halflings-white.png b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/assets/img/glyphicons-halflings-white.png deleted file mode 100644 index 3bf6484a29d8da269f9bc874b25493a45fae3bae..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 8777 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 B7G?kd diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/assets/img/glyphicons-halflings.png b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/assets/img/glyphicons-halflings.png deleted file mode 100644 index 79bc568c21395d5a5ceab62fadb04457094b2ac7..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 13826 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 diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/assets/js/html5.js b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/assets/js/html5.js deleted file mode 100644 index 560aa942..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/assets/js/html5.js +++ /dev/null @@ -1,3 +0,0 @@ -/*! HTML5 Shiv vpre3.6 | @afarkas @jdalton @jon_neal @rem | MIT/GPL2 Licensed - Uncompressed source: https://github.com/aFarkas/html5shiv */ -(function(a,b){function h(a,b){var c=a.createElement("p"),d=a.getElementsByTagName("head")[0]||a.documentElement;return c.innerHTML="x",d.insertBefore(c.lastChild,d.firstChild)}function i(){var a=l.elements;return typeof a=="string"?a.split(" "):a}function j(a){var b={},c=a.createElement,f=a.createDocumentFragment,g=f();a.createElement=function(a){if(!l.shivMethods)return c(a);var f;return b[a]?f=b[a].cloneNode():e.test(a)?f=(b[a]=c(a)).cloneNode():f=c(a),f.canHaveChildren&&!d.test(a)?g.appendChild(f):f},a.createDocumentFragment=Function("h,f","return function(){var n=f.cloneNode(),c=n.createElement;h.shivMethods&&("+i().join().replace(/\w+/g,function(a){return c(a),g.createElement(a),'c("'+a+'")'})+");return n}")(l,g)}function k(a){var b;return a.documentShived?a:(l.shivCSS&&!f&&(b=!!h(a,"article,aside,details,figcaption,figure,footer,header,hgroup,nav,section{display:block}audio{display:none}canvas,video{display:inline-block;*display:inline;*zoom:1}[hidden]{display:none}audio[controls]{display:inline-block;*display:inline;*zoom:1}mark{background:#FF0;color:#000}")),g||(b=!j(a)),b&&(a.documentShived=b),a)}var c=a.html5||{},d=/^<|^(?:button|form|map|select|textarea|object|iframe|option|optgroup)$/i,e=/^<|^(?:a|b|button|code|div|fieldset|form|h1|h2|h3|h4|h5|h6|i|iframe|img|input|label|li|link|ol|option|p|param|q|script|select|span|strong|style|table|tbody|td|textarea|tfoot|th|thead|tr|ul)$/i,f,g;(function(){var c=b.createElement("a");c.innerHTML="",f="hidden"in c,f&&typeof injectElementWithStyles=="function"&&injectElementWithStyles("#modernizr{}",function(b){b.hidden=!0,f=(a.getComputedStyle?getComputedStyle(b,null):b.currentStyle).display=="none"}),g=c.childNodes.length==1||function(){try{b.createElement("a")}catch(a){return!0}var c=b.createDocumentFragment();return typeof c.cloneNode=="undefined"||typeof c.createDocumentFragment=="undefined"||typeof c.createElement=="undefined"}()})();var l={elements:c.elements||"abbr article aside audio bdi canvas data datalist details figcaption figure footer header hgroup mark meter nav output progress section summary time video",shivCSS:c.shivCSS!==!1,shivMethods:c.shivMethods!==!1,type:"default",shivDocument:k};a.html5=l,k(b)})(this,document) \ No newline at end of file diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/cookie_policy.jsp b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/cookie_policy.jsp deleted file mode 100644 index e141abad..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/cookie_policy.jsp +++ /dev/null @@ -1,240 +0,0 @@ -<%@ page import="static com.wso2.openbanking.accelerator.consent.extensions.authservlet.impl.util.Utils.i18n" %><%-- - ~ Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - ~ - ~ WSO2 LLC. 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. - --%> - -<%@include file="includes/localize.jsp" %> - - - - - - - - - -

-
-
- - - -
-
-
- -
- - -
-
-
-
-
-
-

- <%=i18n(resourceBundle, "wso2.open.banking")%> - <%=i18n(resourceBundle, "privacy.policy.cookies")%> -

-
- -
-
-

About WSO2 Open Banking Solution

-

WSO2 Open Banking Solution (referred to as “WSO2 Open Banking” within this policy) is a comprehensive Open Banking Solution that is supporting PSD2 compliance along with more value added features.

-
- -
- -

WSO2 Open Banking uses cookies so that it can provide the best user experience for you and identify you for security purposes. If you disable cookies, some of the services will (most probably) be inaccessible to you.

-
- -
-

How does WSO2 Open Banking process cookies?

-

WSO2 Open Banking stores and retrieves information on your browser using cookies. This information is used to provide a better experience. Some cookies serve the primary purposes of allowing a user to log in to the system, maintaining sessions, and keeping track of activities you do within the login session.

-

The primary purpose of some cookies used in WSO2 Open Banking is to personally identify you as required by the functionality of WSO2 Open Banking. However the cookie lifetime ends once your session ends i.e., after you log-out, or after the session expiry time has elapsed.

-

Some cookies are simply used to give you a more personalised web experience and these cookies can not be used to personally identify you or your activities.

-

This cookie policy is part of the WSO2 Open Banking Privacy Policy.

-
- -
- -

A browser cookie is a small piece of data that is stored on your device to help websites and mobile apps remember things about you. Other technologies, including web storage and identifiers associated with your device, may be used for similar purposes. In this policy, we use the term “cookies” to discuss all of these technologies.

-
- -
-

What does WSO2 Open Banking use cookies for?

-

Cookies are used for two purposes in WSO2 Open Banking.

-
    -
  1. To identify you and provide security (as this is the main function of WSO2 IS).
  2. -
  3. To provide a satisfying user experience.
  4. -
-
- -
-

WSO2 Open Banking uses cookies for the following purposes listed below.

-

Preferences

-

WSO2 Open Banking uses these cookies to remember your settings and preferences, and to auto-fill the form fields to make your interactions with the site easier.

-

These cookies can not be used to personally identify you.

-

Security

-
    -
  • WSO2 Open Banking uses selected cookies to identify and prevent security risks. - For example, WSO2 Open Banking may use these cookies to store your session information in order to prevent others from changing your password without your username and password.
  • -
  • WSO2 Open Banking uses session cookies to maintain your active session.
  • -
  • WSO2 Open Banking may use temporary cookies when performing multi-factor authentication and federated authentication.
  • -
  • WSO2 Open Banking may use permanent cookies to detect that you have previously used the same device to log in. This is to to calculate the “risk level” associated with your current login attempt. This is primarily to protect you and your account from possible attack.
  • -
-

Performance

-

WSO2 Open Banking may use cookies to allow “Remember Me” functionalities.

-
- -
-

Analytics

-

WSO2 Open Banking as a product does not use cookies for analytical purposes.

-
- -
-

Third party cookies

-

Using WSO2 Open Banking may cause some third-party cookies to be set in your browser. WSO2 Open Banking has no control over how any of them operate. The third-party cookies that may be set include:

-
    -
  • Any social login sites. For example, third-party cookies may be set when WSO2 Open Banking is configured to use “social” or “federated” login, and you opt to login with your “Social Account”.
  • -
  • Any third party federated login.
  • -
-

WSO2 strongly advises you to refer the respective cookie policy of such sites carefully as WSO2 has no knowledge or use on these cookies.

-
- -
-

What type of cookies does WSO2 Open Banking use?

-

WSO2 Open Banking uses persistent cookies and session cookies. A persistent cookie helps WSO2 Open Banking to recognize you as an existing user so that it is easier to return to WSO2 or interact with WSO2 Open Banking without signing in again. After you sign in, a persistent cookie stays in your browser and will be read by WSO2 Open Banking when you return to WSO2 Open Banking.

-

A session cookie is a cookie that is erased when the user closes the web browser. The session cookie is stored in temporary memory and is not retained after the browser is closed. Session cookies do not collect information from the user's computer.

-
- -
-

How do I control my cookies?

-

Most browsers allow you to control cookies through their settings preferences. However, if you limit the given ability for websites to set cookies, you may worsen your overall user experience since it will no longer be personalized to you. It may also stop you from saving customized settings like login information.

-

Most likely, disabling cookies will make you unable to use authentication and authorization functionalities offered by WSO2 Open Banking.

-

If you have any questions or concerns regarding the use of cookies, please contact the entity or individuals (or their data protection officer, if applicable) of the organization running this WSO2 Open Banking instance.

-
- -
-

What are the cookies used?

- - - - - - - - - - - - - - - - - - - - - - - - - - - - -
-

Cookie Name

-
-

Purpose

-
-

Retention

-
-

JSESSIONID

-
-

To keep your session data in order to give you a good user experience.

-
-

Session

-
-

MSG##########

-
-

To keep some messages that are shown to you in order to give you a good user experience.

-

The “##########” reference in this coookie represents a random number e.g., MSG324935932.

-
-

Session

-
-

requestedURI

-
-

The URI you are accessing.

-
-

Session

-
-

current-breadcrumb

-
-

To keep your active page in session in order to give you a good user experience.

-
-

Session

-
-
- -
-

Disclaimer

-
    -
  1. This cookie policy is only for the illustrative purposes of the product WSO2 Open Banking. The content in the policy is technically correct at the time of the product shipment. The organization which runs this WSO2 Open Banking instance has full authority and responsibility with regard to the effective Cookie Policy.

  2. -
  3. WSO2, its employees, partners, and affiliates do not have access to and do not require, store, process or control any of the data, including personal data contained in WSO2 Open Banking. All data, including personal data is controlled and processed by the entity or individual running WSO2 Open Banking. WSO2, its employees partners and affiliates are not a data processor or a data controller within the meaning of any data privacy regulations. WSO2 does not provide any warranties or undertake any responsibility or liability in connection with the lawfulness or the manner and purposes for which WSO2 Open Banking is used by such entities or persons.
  4. -
-
-
- -
-
- - -
- - - - - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/Roboto.css b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/Roboto.css deleted file mode 100644 index 558ee634..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/Roboto.css +++ /dev/null @@ -1,26 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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. - */ - -@font-face { - font-family: 'Roboto'; - src: url('../fonts/Roboto/Roboto-Black-webfont.woff') format('woff'), - url('../fonts/Roboto/Roboto-Black-webfont.ttf') format('truetype'), - url('../fonts/Roboto/Roboto-Black-webfont.svg') format('svg'); - font-weight: 100; - font-style: normal; -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/custom-common.css b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/custom-common.css deleted file mode 100644 index c6eb9257..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/custom-common.css +++ /dev/null @@ -1,597 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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. - */ - -html { - position: relative; - height: 100%; - overflow-y: auto; - overflow-x: hidden; - width: 100%; -} - -body { - background: #efefef; - font-family: "Open Sans", "Helvetica", "Arial", sans-serif; -} - -body.sticky-footer { - padding-bottom: 40px; -} - -header { - background: transparent; - min-height: 150px; - position: relative; -} - -header .brand a { - min-height: 24px; -} - -header .brand img.logo { - height: 24px; -} - -header .brand h1 { - margin: 0 0 0 5px; - display: inline-block; - line-height: 1; -} - -header .brand h1 em { - font-weight: 500; - font-size: 17px; - margin: 0; - color: #ffffff; - padding: 3px 0 0 0; - font-style: normal; - text-transform: uppercase; -} - -header .brand img.logo { - height: 45px; -} - -header .brand { - margin-top: 35px; - margin-left: 25px; - width: 181px; -} - -footer { - position: absolute; - bottom: 0; - width: 100%; - min-height: 40px; - overflow: hidden; - color: rgba(0, 0, 0, 0.87); - background: #efefef; - text-align: center; -} - -footer .icon { - font-size: 37px; - vertical-align: middle; - color: rgba(0, 0, 0, 0.87); -} - -footer > .container, -footer > .container-fluid { - padding-right: 15px; - padding-left: 15px; -} - -footer > .container p, -footer > .container-fluid p { - line-height: 40px; - margin-bottom: 0; -} - -footer a, -footer a:hover { - text-decoration: none; - color: #cbcbcb; -} - -body.sticky-footer footer { - position: absolute; - bottom: 0; - z-index: 1000; -} - -.form-control { - border-radius: 0; - background-color: #1A1F28; - color: #efefef; - border: 1px solid #1A1F28; -} - -.form-control:focus { - border-color: #00B4FF; - outline: 0; - -webkit-box-shadow: inset 0 1px 1px rgba(0, 0, 0, 0.075), 0 0 8px #00B4FF; - box-shadow: inset 0 1px 1px rgba(0, 0, 0, 0.075), 0 0 8px #00B4FF; -} - -.col-centered { - float: none; - margin: 0 auto; -} - -.boarder-bottom-blue { - border-bottom: 2px solid #006596; -} - -.white { - color: #FFF; -} - -.blue-bg { - background-color: #3a9ecf !important; -} - -.green-bg { - background-color: #87ad1c !important; -} - -.uppercase { - text-transform: uppercase; -} - -.brand-container { - padding-top: 26px; -} - -@media (min-width: 1200px) { - .pull-right-lg { - float: right !important; - } -} - -.well { - background-color: #2f3e54; - -webkit-border-radius: 0; - -moz-border-radius: 0; - border-radius: 0; - border: none; - padding-bottom: 10px; - padding-left: initial; -} - -.data-container { - margin: 3rem 0; - background-image: linear-gradient(to bottom, #1a1f28 0%, #2e3b41 100%); - background-image: url(../images/login-back.svg), linear-gradient(to bottom, #1a1f28 0%, #2e3b41 100%); - background-repeat: no-repeat; - background-position: left bottom; - background-size: contain; - border: 1px solid #000; - border-radius: 10px; - color: '#fff'; -} - -.data-container.error { - border-left: 3px solid #00B4FF; -} - -.data-container form h3 { - margin-top: 10px; -} - -.input-group-addon { - background: #2f3e54; - color: #fff; - border: none; - border-radius: 0 !important; -} - -.input-group-lg > .form-control, -.input-group-lg > .input-group-addon, -.input-group-lg > .input-group-btn > .btn { - border-radius: 0 !important; -} - -@media (min-width: 991px) { - .data-container h3 { - font-size: 1.8em; - } -} - -.font-large { - font-size: 16px; -} - -.wr-input-control { - margin-bottom: 20px; -} - -.wr-login { - padding-top: 50px; - -} - -.wr-login input[type=text], -.wr-login input[type=password] { - border: 1px #d2d2d2 solid; - width: 100%; - padding: 6px 10px; - z-index: 1; - -webkit-appearance: none; - line-height: 30px; - border-radius: 0px; -} - -.btn-primary, -.btn-primary:focus, -.btn-primary:active { - color: #fff; - background-color: #87ad1c; - border: 1px solid #18184c; - transition: all .2s ease-in-out; - border-radius: 0; -} - -.btn-primary:hover { - background-color: #87ad1c; - color: #fff; - border: 1px solid #18184c; -} - -.btn-secondary, -.btn-secondary:focus, -.btn-secondary:active { - color: #fff; - background-color: transparent; - transition: all .2s ease-in-out; - border-radius: 0; - font-size: 17px; - font-weight: 600; -} - -.btn-secondary:hover { - background-color: transparent; - color: #fff; -} - -.wr-btn { - font-weight: normal; - font-size: 13px; - color: #fff; - background: #5d81d2; - padding: 10px 10px; - display: inline-block; - border: none; -} - -button.grey-bg:hover { - background-color: #3A9ECF; -} - -.wr-btn:hover { - text-decoration: none; - color: #ffffff; - background-color: #6b94f1; -} - -button.grey-bg { - background-color: #222222; -} - -button.font-extra-large { - font-size: 20px; -} - -img.idp-image { - padding: 3px 2px; -} - - -.padding-left { - padding-left: 10px; -} - -.padding-right { - padding-right: 10px; -} - -.padding-top { - padding-top: 10px; -} - -.padding-bottom { - padding-bottom: 10px; -} - -.padding { - padding: 10px; - margin-bottom: 0px; -} - -.padding-none { - padding: 0px !important; -} - -.padding-left-double { - padding-left: 20px; -} - -.padding-right-double { - padding-right: 20px; -} - -.padding-top-double { - padding-top: 20px; -} - -.padding-bottom-double { - padding-bottom: 20px; -} - -.padding-double { - padding: 20px; -} - -.margin-left { - margin-left: 10px; -} - -.margin-right { - margin-right: 10px; -} - -.margin-top { - margin-top: 10px; -} - -.margin-bottom { - margin-bottom: 10px; -} - -.margin { - margin: 10px; -} - -.margin-none { - margin: 0px !important; -} - -.margin-left-double { - margin-left: 20px; -} - -.margin-right-double { - margin-right: 20px; -} - -.margin-top-double { - margin-top: 20px; -} - -.margin-bottom-double { - margin-bottom: 20px; -} - -.margin-double { - margin: 20px; -} - -.font-small { - font-size: 12px; -} - -.font-medium { - font-size: 16px; -} - -.font-large { - font-size: 1.3em; -} - -.font-extra-large { - font-size: 20px !important; -} - -.error-alert { - background-color: #FFE7E8; -} - -.form-group.required .control-label:after { - content: " *"; - color: red; -} - -@media (min-width: 991px) { - .login-form-wrapper { - padding-top: 5%; - } -} - -.login-form .scope { - font-size: 1.5em; -} - -a, -a:hover, -a:active, -a:focus, -.data-container a, -.login a { - color: #00b4ff; - text-decoration: none; -} - -.policy-info-message { - margin-bottom: 0; - margin-top: 10px; -} - -.static-page { - padding-right: 40px; - padding-left: 40px; -} - -.table-of-contents ul { - padding-left: 20px; - font-size: 14px; -} - -.table-of-contents h4 { - color: #fff; -} - -.table-of-contents ul a { - color: #b7c0cd; -} - -.table-of-contents ul a:hover { - color: #fff; -} - -.table-of-contents ul li.sub { - margin-left: 25px; -} - -.policies-wrapper section { - margin-bottom: 50px; -} - -.policies-wrapper section h2, -.policies-wrapper section h3 { - color: #fff; -} - -.login-logo { - margin: auto; - width: 20%; -} - -h3.ui.header { - font-weight: 300; - color: #efefef; - font-family: -apple-system, BlinkMacSystemFont, Segoe WPC, Segoe UI, HelveticaNeue-Light, Ubuntu, Droid Sans, sans-serif, font-wso2, 'Helvetica Neue', Arial, Helvetica, sans-serif; -} - -h4.ui.header { - font-weight: 280; - color: #efefef; - font-family: -apple-system, BlinkMacSystemFont, Segoe WPC, Segoe UI, HelveticaNeue-Light, Ubuntu, Droid Sans, sans-serif, font-wso2, 'Helvetica Neue', Arial, Helvetica, sans-serif; -} - -div.ui.body { - color: #efefef; - font-size: 12px; - margin-top: 1em; -} - -h5.ui.body { - color: #efefef; - font-size: 1.2em; -} - -div.ui.form { - margin: auto; - color: #efefef; -} - -.ui.primary.button, .ui.primary.button:hover { - background: #18184c; - color: #ffffff; - font-family: -apple-system, BlinkMacSystemFont, Segoe WPC, Segoe UI, HelveticaNeue-Light, Ubuntu, Droid Sans, sans-serif, font-wso2, 'Helvetica Neue', Arial, Helvetica, sans-serif; - font-size: 17px; - font-weight: 500; -} - -.ui.primary.button:hover { - background: #070749; -} - -div.well.policy-info-message { - margin-bottom: 1em; -} - -select.ui.select { - opacity: 0.8; - background-image: linear-gradient(to right, #2a363d 0%, #25405d 100%); -} - -.well { - background-color: transparent; - padding-left: initial; -} - -div.ui.box { - padding-left: 40px; - padding-right: 40px; - padding-top: 20px; -} - -div.ui.subheading { - font-size: 1.2em; -} - -.h4 { - font-size: 1.7em; -} - -div.ui.form.select { - padding-right: 25px; - padding-left: 25px; -} - -div.ui.body.text { - padding-left: unset; -} - -div.ui.form.row { - padding-left: 25px; - padding-right: 25px; -} - -div.section.heading { - color: #159cfa; - padding-top: inherit; -} - -.privacy-policy-product-title { - background: #fff; - padding-left: 15em; - padding-bottom: 1em; - padding-top: 1em; -} - -.ui.segment.toc { - background: #fff; - box-shadow: 0 1px 2px 0 rgba(34,36,38,.15); - padding-right: 1.5rem; - padding-left: 1.5rem; - border-radius: 3px; - border: 1px solid rgba(34,36,38,.15); -} - -ui.segment.toc.h4 { - color: #000; -} - -li.sub { - list-style: none; -} - -li.sub::before { - content: "\2192"; - margin-left: -1em; -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/localstyles-ie7.css b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/localstyles-ie7.css deleted file mode 100644 index 04f8b280..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/localstyles-ie7.css +++ /dev/null @@ -1,21 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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. - */ - -.form-horizontal .controls{ - margin-left:0; -} \ No newline at end of file diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/localstyles.css b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/localstyles.css deleted file mode 100644 index 1e225726..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/localstyles.css +++ /dev/null @@ -1,282 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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. - */ - -body, p, div, a, span, h1, h2, h3, h4 { - font-family: 'helvetica neue', helvetica, arial, 'lucida grande', sans-serif; -} - -body { - background: #999 url(../images/body-back.png) no-repeat center 10px; -} - -.header-strip { - background: #5e5e5e; - height: 10px; -} - -.header-back { - background: transparent url(../images/repeat.jpg) repeat-x left top; - height: 85px; -} - -.header-text { - background: #000; - color: #cccccc; - font-size: 13px; - text-align: center; -} - -.header-text strong { - color: #fff; - font-size: 14px; -} - -.logo { - background: transparent url(../images/logo.png) no-repeat left top; - width: 424px; - height: 85px; - display: block; - cursor: pointer; -} - -.content-section { - text-align: left; - padding: 10px 0 0 0; -} - -.content-section form.well { - text-align: left; -} - -.btn-primary { - background-color: #316a90; - *background-color: #316a90; - background-image: -ms-linear-gradient(top, #316a90, #4583ac); - background-image: -webkit-gradient(linear, 0 0, 0 100%, from(#316a90), to(#4583ac)); - background-image: -webkit-linear-gradient(top, #316a90, #4583ac); - background-image: -o-linear-gradient(top, #316a90, #4583ac); - background-image: -moz-linear-gradient(top, #316a90, #4583ac); - background-image: linear-gradient(top, #316a90, #4583ac); - background-repeat: repeat-x; - border-color: #0055cc #0055cc #003580; - border-color: rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.25); - filter: progid:dximagetransform.microsoft.gradient(startColorstr='#316a90', endColorstr='#4583ac', GradientType=0); - filter: progid:dximagetransform.microsoft.gradient(enabled=false); -} - -.login-header { - background: #e86d1f; - color: #fff; - padding: 5px; - - -webkit-border-top-left-radius: 4px; - border-top-left-radius: 4px; - -moz-border-radius-topleft: 4px; - - -webkit-border-top-right-radius: 4px; - border-top-right-radius: 4px; - -moz-border-radius-topright: 4px; - - margin-top: 10px; -} - -.form-horizontal .control-label { - width: 80px; -} - -.form-horizontal .controls { - margin-left: 100px; -} - -.form-horizontal .form-actions { - padding-left: 100px; - padding-bottom: 0; - margin-bottom: 0; -} - -h1 { - margin-top: 20px; - color: #fff; - font-weight: normal; - font-size: 30px; -} - -.different-login-container { - background: #F5F5F5; - padding: 10px; - -webkit-box-shadow: 7px 7px 5px 0px rgba(50, 50, 50, 0.75); - -moz-box-shadow: 7px 7px 5px 0px rgba(50, 50, 50, 0.75); - box-shadow: 7px 7px 5px 0px rgba(50, 50, 50, 0.75); -} - -.main-login-container { - background: transparent url(../images/container-back.png) repeat left top; - padding: 20px 10px; - margin-bottom: 20px; - -webkit-box-shadow: 7px 7px 5px 0px rgba(50, 50, 50, 0.75); - -moz-box-shadow: 7px 7px 5px 0px rgba(50, 50, 50, 0.75); - box-shadow: 7px 7px 5px 0px rgba(50, 50, 50, 0.75); -} - -.vertical-slitter { - border-right: solid 1px #fff; -} - -.form-actions { - border-top: none; - background: transparent; -} - -.btn-primary.active { - color: rgba(255, 255, 255, 0.75); -} - -.btn-primary { - color: #ffffff; - text-shadow: 0 -1px 0 rgba(0, 0, 0, 0.25); - background: #45484d; /* Old browsers */ - background: -moz-linear-gradient(top, #45484d 0%, #000000 100%); /* FF3.6+ */ - background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, #45484d), color-stop(100%, #000000)); /* Chrome,Safari4+ */ - background: -webkit-linear-gradient(top, #45484d 0%, #000000 100%); /* Chrome10+,Safari5.1+ */ - background: -o-linear-gradient(top, #45484d 0%, #000000 100%); /* Opera 11.10+ */ - background: -ms-linear-gradient(top, #45484d 0%, #000000 100%); /* IE10+ */ - background: linear-gradient(to bottom, #45484d 0%, #000000 100%); /* W3C */ - filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#45484d', endColorstr='#000000', GradientType=0); /* IE6-9 */ - padding: 5px 10px; - -} - -.btn-primary:hover, -.btn-primary:focus, -.btn-primary:active, -.btn-primary.active, -.btn-primary.disabled, -.btn-primary[disabled] { - color: #ffffff; - - background: #45484d; /* Old browsers */ - background: -moz-linear-gradient(top, #45484d 0%, #000000 100%); /* FF3.6+ */ - background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, #45484d), color-stop(100%, #000000)); /* Chrome,Safari4+ */ - background: -webkit-linear-gradient(top, #45484d 0%, #000000 100%); /* Chrome10+,Safari5.1+ */ - background: -o-linear-gradient(top, #45484d 0%, #000000 100%); /* Opera 11.10+ */ - background: -ms-linear-gradient(top, #45484d 0%, #000000 100%); /* IE10+ */ - background: linear-gradient(to bottom, #45484d 0%, #000000 100%); /* W3C */ - filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#45484d', endColorstr='#000000', GradientType=0); /* IE6-9 */ - -} - -.btn-primary:active, -.btn-primary.active { - background-color: #003399 \9; -} - -.btn-primary-deny { - color: #ffffff; - text-shadow: 0 -1px 0 rgba(0, 0, 0, 0.25); - background: #C94135; /* Old browsers */ - background: -moz-linear-gradient(top, #C94135 0%, #4E0505 100%); /* FF3.6+ */ - background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, #C94135), color-stop(100%, #4E0505)); /* Chrome,Safari4+ */ - background: -webkit-linear-gradient(top, #C94135 0%, #4E0505 100%); /* Chrome10+,Safari5.1+ */ - background: -o-linear-gradient(top, #C94135 0%, #4E0505 100%); /* Opera 11.10+ */ - background: -ms-linear-gradient(top, #C94135 0%, #4E0505 100%); /* IE10+ */ - background: linear-gradient(to bottom, #C94135 0%, #4E0505 100%); /* W3C */ - filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#C94135', endColorstr='#4E0505', GradientType=0); /* IE6-9 */ - padding: 5px 10px; - -} - -.btn-primary-deny:hover, -.btn-primary-deny:focus, -.btn-primary-deny:active, -.btn-primary-deny.active, -.btn-primary-deny.disabled, -.btn-primary-deny[disabled] { - color: #ffffff; - - background: #C94135; /* Old browsers */ - background: -moz-linear-gradient(top, #C94135 0%, #4E0505 100%); /* FF3.6+ */ - background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, #C94135), color-stop(100%, #4E0505)); /* Chrome,Safari4+ */ - background: -webkit-linear-gradient(top, #C94135 0%, #4E0505 100%); /* Chrome10+,Safari5.1+ */ - background: -o-linear-gradient(top, #C94135 0%, #4E0505 100%); /* Opera 11.10+ */ - background: -ms-linear-gradient(top, #C94135 0%, #4E0505 100%); /* IE10+ */ - background: linear-gradient(to bottom, #C94135 0%, #4E0505 100%); /* W3C */ - filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#C94135', endColorstr='#4E0505', GradientType=0); /* IE6-9 */ - -} - -.btn-primary-deny:active, -.btn-primary-deny.active { - background-color: #CA1909 \9; -} - -.marL30 { - margin-left: 30% !important; -} - -.marL32 { - margin-left: 32% !important; -} - -h2 { - color: #ddd; - font-weight: normal; -} - -.different-login-container a.main-link { - display: block; - background: transparent url(../images/icon-default.png) no-repeat left top; - width: 75px; - height: 76px; - float: left; - margin-right: 10px; - padding-left: 74px; -} - -#claimed_id { - background: #fff url(../images/openid-input.gif) no-repeat left 4px; - padding-left: 20px; -} - -.slidePopper { - position: absolute; - margin-top: 70px; - padding: 10px; - background: #fff; - border: solid 1px #ccc; - border-radius: 5px; - z-index: 6; -} - -.slidePopper-cancel { - cursor: pointer; -} - -.overlay { - background: #ccc; - opacity: 0.5; - width: 500px; - height: 500px; - position: absolute; - top: 0; - left: 0; - z-index: 5; -} - -.go-btn { - margin-top: -8px; -} diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/openid-provider.css b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/openid-provider.css deleted file mode 100644 index e5678434..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/css/openid-provider.css +++ /dev/null @@ -1,195 +0,0 @@ -/** - * Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - * - * WSO2 LLC. 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. - */ - -.card-box-top{ - background-image:url(../images/card-box02.jpg); - background-repeat:repeat-x; - background-position:0 0; -} -.card-box-left{ - background-image:url(../images/card-box08.jpg); - background-repeat:repeat-y; - background-position:0 0; - width:15px; -} -.card-box-right{ - background-image:url(../images/card-box04.jpg); - background-repeat:repeat-y; - background-position:0 0; - width:18px; -} -.card-box-bottom{ - background-image:url(../images/card-box06.jpg); - background-repeat:repeat-x; - background-position:0 0; -} -.card-box-mid{ - background-color:#ffffff; -} -.user-pic{ - margin-bottom:10px; -} -.card-box{ - width:400px; -} -.openid-box-top{ - background-image:url(../images/openid-box-back.gif); - background-position:0 0; - background-repeat:no-repeat; - width:812px; - height:123px; -} -.openid-box-05{ - background-image:url(../images/openid-box-05.gif); -} -.openid-box-07{ - background-image:url(../images/openid-box-07.gif); -} - -.openid-box-06{ - background-image:url(../images/openid-box-06.gif); - background-position:0 0; - background-repeat:repeat-x; -} -.openid-box-04{ - background-image:url(../images/openid-box-04.gif); - background-position:0 0; - background-repeat:repeat-y; - width:2px; -} -.openid-box-08{ - background-image:url(../images/openid-box-08.gif); - background-position:0 0; - background-repeat:repeat-y; - width:2px; -} -.openid-box-back{ - background-color:#e3e7ea; - padding-bottom:50px; - padding-left:50px; - font-size:20px; -} -.openid-box-05,.openid-box-07{ - background-position:0 0; - background-repeat:no-repeat; - width:19px; - height:31px; -} -.openid-box{ - width:812px; -} -.openid-box-username{ - color:#569643; -} - -.userClaimsTbl { - margin:0px;padding:0px; - width:100%; - border:1px solid #000000; - - -moz-border-radius-bottomleft:0px; - -webkit-border-bottom-left-radius:0px; - border-bottom-left-radius:0px; - - -moz-border-radius-bottomright:0px; - -webkit-border-bottom-right-radius:0px; - border-bottom-right-radius:0px; - - -moz-border-radius-topright:0px; - -webkit-border-top-right-radius:0px; - border-top-right-radius:0px; - - -moz-border-radius-topleft:0px; - -webkit-border-top-left-radius:0px; - border-top-left-radius:0px; -}.userClaimsTbl table{ - border-collapse: collapse; - border-spacing: 0; - width:100%; - height:100%; - margin:0px;padding:0px; -}.userClaimsTbl tr:last-child td:last-child { - -moz-border-radius-bottomright:0px; - -webkit-border-bottom-right-radius:0px; - border-bottom-right-radius:0px; -} -.userClaimsTbl table tr:first-child td:first-child { - -moz-border-radius-topleft:0px; - -webkit-border-top-left-radius:0px; - border-top-left-radius:0px; -} -.userClaimsTbl table tr:first-child td:last-child { - -moz-border-radius-topright:0px; - -webkit-border-top-right-radius:0px; - border-top-right-radius:0px; -}.userClaimsTbl tr:last-child td:first-child{ - -moz-border-radius-bottomleft:0px; - -webkit-border-bottom-left-radius:0px; - border-bottom-left-radius:0px; -}.userClaimsTbl tr:hover td{ - background-color:#ffffff; - - -} -.userClaimsTbl td{ - vertical-align:middle; - - background-color:#e5e5e5; - - border:1px solid #000000; - border-width:0px 1px 1px 0px; - text-align:left; - padding:7px; - font-size:12px; - font-family:Arial; - font-weight:normal; - color:#000000; -}.userClaimsTbl tr:last-child td{ - border-width:0px 1px 0px 0px; -}.userClaimsTbl tr td:last-child{ - border-width:0px 0px 1px 0px; -}.userClaimsTbl tr:last-child td:last-child{ - border-width:0px 0px 0px 0px; -} -.userClaimsTbl tr:first-child td{ - background:-o-linear-gradient(bottom, #cccccc 5%, #cccccc 100%); background:-webkit-gradient( linear, left top, left bottom, color-stop(0.05, #cccccc), color-stop(1, #cccccc) ); - background:-moz-linear-gradient( center top, #cccccc 5%, #cccccc 100% ); - filter:progid:DXImageTransform.Microsoft.gradient(startColorstr="#cccccc", endColorstr="#cccccc"); background: -o-linear-gradient(top,#cccccc,cccccc); - - background-color:#cccccc; - border:0px solid #000000; - text-align:center; - border-width:0px 0px 1px 1px; - font-size:14px; - font-family:Arial; - font-weight:bold; - color:#000000; -} -.userClaimsTbl tr:first-child:hover td{ - background:-o-linear-gradient(bottom, #cccccc 5%, #cccccc 100%); background:-webkit-gradient( linear, left top, left bottom, color-stop(0.05, #cccccc), color-stop(1, #cccccc) ); - background:-moz-linear-gradient( center top, #cccccc 5%, #cccccc 100% ); - filter:progid:DXImageTransform.Microsoft.gradient(startColorstr="#cccccc", endColorstr="#cccccc"); background: -o-linear-gradient(top,#cccccc,cccccc); - - background-color:#cccccc; -} -.userClaimsTbl tr:first-child td:first-child{ - border-width:0px 0px 1px 0px; -} -.userClaimsTbl tr:first-child td:last-child{ - border-width:0px 0px 1px 1px; -} \ No newline at end of file diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/default_consent.jsp b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/default_consent.jsp deleted file mode 100644 index ff38a299..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/default_consent.jsp +++ /dev/null @@ -1,181 +0,0 @@ -<%-- - ~ Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - ~ - ~ WSO2 LLC. 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. - --%> - -<%@ page contentType="text/html;charset=UTF-8" language="java" %> -<%@ taglib prefix = "fmt" uri = "http://java.sun.com/jsp/jstl/fmt" %> -<%@ taglib prefix = "c" uri = "http://java.sun.com/jsp/jstl/core" %> - - - -
- - - -
- - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/default_displayconsent.jsp b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/default_displayconsent.jsp deleted file mode 100644 index 2048c280..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/default_displayconsent.jsp +++ /dev/null @@ -1,143 +0,0 @@ -<%-- - ~ Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - ~ - ~ WSO2 LLC. 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. - --%> -<%@ page import="java.util.List" %> -<%@ page import="java.util.Map" %> -<%@ page import="org.json.JSONArray" %> -<%@ page import="org.json.JSONObject" %> - -<%@ taglib prefix = "fmt" uri = "http://java.sun.com/jsp/jstl/fmt" %> -<%@ taglib prefix = "c" uri = "http://java.sun.com/jsp/jstl/core" %> - - -<% - session.setAttribute("configParamsMap", request.getAttribute("data_requested")); - Map> consentData = (Map>) request.getAttribute("data_requested"); -%> -
-
-
-
- diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/fonts/Roboto/Roboto-Bold-webfont.svg b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/fonts/Roboto/Roboto-Bold-webfont.svg deleted file mode 100644 index 43b5ed22..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/fonts/Roboto/Roboto-Bold-webfont.svg +++ /dev/null @@ -1,593 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/fonts/Roboto/Roboto-Bold-webfont.ttf b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/fonts/Roboto/Roboto-Bold-webfont.ttf deleted file mode 100644 index 1da72769a88d2f810714c4ecc15ccbaf19a6c842..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 45008 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 ZW+xi~HayM8mb8SX;9)TN?yK90{sW5|l%@ax diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/fonts/Roboto/Roboto-Bold-webfont.woff b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/fonts/Roboto/Roboto-Bold-webfont.woff deleted file mode 100644 index 0c6994871e33563d2dbef857ce8bc1520e9cf3c0..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 24808 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%0AM$T$}k{>uvs+ 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 W{a8|@-gUZL3;+NC0aAPRlmGysvh&yg diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/generic-exception-response.jsp b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/generic-exception-response.jsp deleted file mode 100644 index 5d384318..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/generic-exception-response.jsp +++ /dev/null @@ -1,81 +0,0 @@ -<%-- - ~ Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - ~ - ~ WSO2 LLC. 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. - --%> - - - -<%@ taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt" %> -<%@ page import="org.owasp.encoder.Encode" %> -<% - String stat = request.getParameter("status"); - String statusMessage = request.getParameter("statusMsg"); - if (stat == null || statusMessage == null) { - stat = "Authentication Error !"; - statusMessage = "Something went wrong during the authentication process. Please try signing in again."; - } - session.invalidate(); -%> - - - - - - - -
-
-
- -
-
-
- - - - - \ No newline at end of file diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/U2F.png b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/U2F.png deleted file mode 100644 index c9d8f3a2706a62e387e19c9b3eeb1d35d168f67d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 33071 zcmaHSbyQp5mN2fx-90!2C%6@NC~m>s-L<$BcWLoL3B}!A3lw*4ad-Lny?JlG`D4~( ztt9vCY?s_~?m0VJRapiNg%|}23JOh5R!SWT3L5AAjg5rx4q3n9AbbBJa{s8~uHj_q z?ge(WfD$)%GPR(Ta{yaes9S)|eOyK?grJ~c{cSXL+;x-`1Gsn6QFs3~pCg{if9lz>Xkk`~rBvc9es8otV!X1;c2AaiO_ z5lSI%fp-QD7Vcn5ZwGrvHvw;9>VNSSc)$J!%uY@DFBEq>Ve0=i&%SXuvV&mZ9lTMDR4N&j2c z`$(AD+TGn*fSujT%Zts6i_OW^ik$-l0q}u-mH#pH2>isW#MM# zYUAu~o(jX2Fj*mPNe3DWipd>FJAJ<174r%HC(8@Wwxq}_eEdC?c=3VZ;X(j(p zS^-H{3$VMBtEQ8a{eLn*)!ND3$<5lynNm`Nmy$u-#?jo#%Z>4$@cfsxQWmZ@o)+fP zu1*e=|MIVZ&Ho~w(-g>U0pd4f<>cq*WaZ{H8wj-I0)aWH+22)U z|0iYsM+*H1^q!ahT>f`Kybu1nq%0iY>&W%JU_#8{KSMzU$jM2GYkDsqcOYh3EV++ve}&eQ9r(p^2V=%*{H6-$lYTFIq~L?$lP zOT`q*%ENt`r9L(#pyZ{rH?mhw@w{~RK1)~mjohT%T;pd+-+tO@)%m1!cl2~Be7EX$ zsQGfEjTiWL%=mL7>l3d-0_tilBdD?YZ6U|38Uz^U*y|iBC~8H|+6jJo$OKtVxD1mb zPSj43wynKAdM}g7tSms<^=rsNEoOBn4IB~{vUVig42t1~(>2^N@L5P!r%05~+x${Z zwswa0z57W;xrf$oZyI`w9jwX>Efe@McLdxF&R#b-w*%I<6G%wlChjI&3P7iPLwDmB zI#1adT7C25L=U`Ma6{Gm-gO{HeVEW4O>TVP2JcYQxJw2OujaAtK&iTPRFTt28g#&@~6p)Qv!R|&ef@R z3t6&OX4RZ$2uNfrfrRolyZ)q~*e{FF=>f;hf0R_EL2?EqWp90wYe)qI?P|s>rmX*z zfWzW1G>b07teCE#?~qWyF*Fqluci;}YRQxCH*w*&e}88J;&8Bw7)iEc;TN$<=-P*q zUzv0i+~!W(fH|VG)n`_eZmsqrMne5!x4cETbW7v*+z1M!u$6XN(F1d0pZL2y3)4%@ z*Z0+$FN3XMUJ!Jo9aoGbkLo4Qvx%?77yZt|aRAknx_a>&3qgqK4o{4CA}jMKDO}5$ zs<&v6O>#f0Wk`6*TI;Fs<&*i~3F0!9O~lAw$1^yElUEiUTfdcblQ^d*^*r-@Dt;n3D(f(2B$0pzm{{w0}>$m|3X5 zyk#9R63uH_`C6pmv9W|Mv(k-VxPG!f;PVfSdC?%z;49-|vG5yFz5VaT0kpMt@41PT z^teTo6w@8=tjLKAjEUMxvV=x7n-s@E=LZ+~%Va;q_3bp@Jt^5E>-l@siu}ra}jLxc&od^8$H>ojvPDY)BgZ__+gFN9fJV03)$5tecpz+3JLt}ODtuNn3hGOrJ z5^6;@IFgH}R?mb62l$4uJh7JMcid7#=*H3Ih?7;@6%>1d=J@MMt<3gD&ytk!l@-wM zM!eO=kO9>nju9gx-@Og)trY|_mgjkO5sh84eVG-N3k|msPFFqA8>$_O96XLpX?J|2 z)iRxzgQvo7*6XVs9LALL+53=67YKR4y_KXxO9rcxS&a?VwJz0V>6NqH$i`5 zd!m?ks6*cho?}OxqH?Ml7Jzj1T7Npdz*>Jrx%k3&D<+gi?cPo^99c%;RZ);{dAwvEd}HVZvlADgO)Fvm!DM;P>jU&Xs?lO-)%lK}Kx7yYc(?^e2K2 z`YV%qPjqgp^>Yy;Om5FXCd}#YzHtx~tUj3Ex>m;rbfG&)KQuNt;4WM2-ybM^%3QTg zFt*+RuPpO&o;c;~R4XWD#)VC&%(RBA%s2kB8IO6%o3%L2Wdrvfw|!ssT4-#HH`qxn z?{d|n{hiVIyr&VM^uxog00fYBp+yXX_TL*b4xxBG2bWi(lv5K71dBb>s8bKOe0j{9 zdsBVQO0J>C6)Eg3Ejq;uNW9$m=0-e86T^-+y{ba!dGh(JtF#MmlIALl{0;nKXIs3E zH50l~ch>Iyb}dN%INsddRyv@h(;{`nBanw~zvheX|EG^4*yCuP&6DW5$JgsrG|le2 z7cqC<2Jt8HnUmYi&NE>s#b>6P1(`!-G?ry)W*ieGqVp^1C|?tEiB>|1%`A{Rk=dc+1&uj<{P;<8cdI zCvA$a47}^TV(%_^9w{KUL6^>qTgjjfOFZ36${|FGUCjTz!NCB1^P?PKR4iQB3IGUN z3;eC!awEJ1SqeI}Y+QVmr7tojl#+z|KT$;JVgYX zH#FtT?*($#nq#tZFZve-R{7#TA?5dr<&zPgy%hHDK1N)Ol|qI3I&)>T0YM<-jn`XY z=R3o@++kOf*M$wjPZQxDRY3xygWCjM_jFkd-?fxRl}+iT_5yjUE4`pAv&X}vW($2o zG1&@z&H7Fko5Qq5tkRGKJ_u&6V|_O$mlY*}gTdn+*){TAK#P%?)~Tct6viK9Fa(}H z3B7LSk0jE(IeM<@kyL!PbF*b>$HvWK|K$heB}{6&{9AA4{3ZM=NcHy==4n^34dRGb zh3&mQTO6j~C0pxSLe}i}&2_&n*ru+(^EkTLUA74-S!NsM$T*fud0eBt=5MdsMGHqC znHcRDq73hPpQ%sE2R1)_svae%N48A|iKm+gxZsBCy8G$k1Y`07;)2V0RW^IuTYOqa8BaxtIJW=nVg>`P^y_d zORE}32K9VZ?c04gzPf!1=z)6Jb!LOEz_n69v!ND?%i-(M(` zS}S=Se)y}oamM0IX4_pOpDEFbA_$M>%&}?Zsv?^I%R99jjR97H2+0^CiAD06I|y0G zNjK?urAdFEBMc%mfBS~xD2m7%bbKfCG4ojgvYPzGQPAr&xWn&m6m^?GLOOVSYN~JU zI@O1ftV}p5G#O|tl*xji`0BM0-GCJ_;#(v=imnnNXUh(%Lr`N)m4yvn3=2cKhi8E; z^Hrn7hewszdipZn*K75-o#(Aq+S|{5KX;UB#11RMfgqOyXHyKeUHvv26~*RRT~1KRJ^`x2Qr4P>UA^(ow-lj2L_bqYW2!=sX%OIA5$(BGI!pfa_s(8q2iqO zBbp%BFLb@?HNU+gN}+}qsfW4)q0$DEmIRMC_V(c%TCVGD#`HrH9M-%enxCH(G02oo`IzmU`Z9Z&{Z$7tj})4ke1r+E`# z`O;`}1z2o$`SxWS*2kD->nRLGqxgv4a~F$Yr%8dbk4hIRrDETNdEZuU7^*Y%NgjeZ zR~j>q?q0CkaM+?~8v-CkvGMwbO`9dBss3yBfT|%Fm0w*as9eabyq@_@P%2xyUd?P< z~C1I1SV8e57OSa>E>d@NO~ok{a9W9ft3X*`H0?DaSpL0%l8@nsDLAY_rw8Z zQNh&nb0-+{v7KmgauQ@wW43=T#yfO1C3x1rlK6JhFdDYNq$pyaz{i3I{%Lt*yVzh? zwwUuea5Z)H($XKV1F7IJ+$FT>&C*jq3#SY3ceh>)sfANf?Nm)$rU;;#kz(M8C^RbH z!r++-VZaaNDE(@XvTInuh=AD^!_aKc$V;HXT8QOE&6EH4p%0SlyFxFoG^$AhO&3f_ zFe!`q5Zdtfa|KaZyG8{hu|9h@fVoJDNF_Ryvbr)PtLX|Fc6Y+L?=zb7WJ}G+n@a=e z^{S3e@xHAKMxopz{@QYtRy}QLE-MFvKW0N|hJf4#cf>IvJM*(n=AWDX?KR}+(Ng3= ziQwS5(<*??oXnG`bgZNu7Q7RtdM|gp7g&)aN-?WwGhiAfmbBtUy@QzI2q4tjr<&2cb z@Hox5B7M0_fu!6mUZedIZStBe4W*@WP@vJV3UzfT{UI=zS{rZti!2HK$Npl2n*{v4 z<%#II0>)Vu79&+m1FZOh!zcB_d_@9R6w+4s_|#_chR+d8QuMvvu#s>nQS&$zF48=f zIw@1!ZTNVbxHh=aVoYip^`&330?LJeNE{ap^Pb)n-)8PrcT}TB{0enu6?VdbSdgs_ ziBn;Ts6U7-=2_aDe0&6OVyT7n>5@a3#Y>W=upy?7DGEsUA^7igrfD z1pFnMulE5s1D$&&Fd8WH5?@_R$wM~>d3}RYGsTsIFWSjIyf47I%_aelk2fSHC(Ar# zw$TvJOXGY-aHumr%5yHK#!!3s)uiCYrD9Yr!SA2Q)y&N)XYRhP;@C}JE_v#i)BTH) z@SB%#QJEXg8NHFz1^j*47^?ufbz;bfurWHS9Sk@(&>9*ECIl%P6$5jI{_nP9_o1mj`jz1)B?C?Q;P`*b##%|ah^ zEM5}{9SuyeoNu+$RH3rL)*3k%$)hfrnNGcylY2F79|YDki0j6Vo+Pa#GM^J6{VS~v zcHq824{4H$UN}?CMv8kt?PFexVHzNm=#v1Im)bVD6rD3Q2AniZ_fVI~E+^a`u~s8D zDsI2dv>n4kV)i?(2M(oh(;wnsHA%__ zAP0Cr-IkMSm98|ZFebJ1URa|_i3&V{kO0g`#f(Cs==;zPu#NcPL!GM-K=FaL^HXbeRM);zXfO7c(!s!a?a>3p#8c6}8}%`_3<(5hqwf5rN`eTWPC*pU7eBpXG? z)3CWWqnAuDLuNMv)L%jM#%HnrStkZUJ8{@=esH%11ihth;+LVkbze@!+8pEs1wz2W zft+JZ#hwpzf&rFoGAY<85xN2@XT`{rGRxx?S8n~b-2EvTh3$QyMmoTcKbaZY2n-%T z^V8D=ABs#ml*$IOxeMB?ZaG17VvG=qw(Tql;}!_|(cq8G=PNJdryK4DVLiy2<0 zWm-9l%1W&~2U#!CA=ElrBIHq{7?hG^gl}*PjAvMX*9b^p)Q;rjzCiW-WJKtLq$F#{ zd{%I@m(>tkesz*}$of2E=Hn%8JI(v+J~4PS;PnPtyxU`HRKf5i5=ZS@s@B(+q&y!kF^GEVWsJCQ;&~qR{$A z4gIUMrIe!MRc{ow0WI%_x-&j@6VLv^`wf4)GzooQ0rojn5D%x2-&;6FM_K4KHR;~= zihiE*>x4wuv3!zheFd!qhFd#xOuT$V<2o&O*@)dYfv)H&7gKvuUMIqZt9^!{3?&s6 z)NfDRC^$xEL073LDV0IF?2m4cIAW_(fwOkQ?Ye`Eo)$_%zS?lgaR^Qu4M6ixa#Lg~ zO1x$pER~WgPMsYMUnznm$5u$;j=Xw^@(&@N-L*;`Y~PB;U5-eY5Hy^Ska9TVpZD!D z^m>xWSKLTa0}M(%;F~HJsKTKjpk>38mt^!x9wmJP)!@s_@h7soVwj_8lJepRXYy%T zWM~egd+ru|ZW9xMMj5Pofqj{TuG(w(0@p$e{U*nV8%qtLiIWt_@HKVpZ^d=HguF4i z%yYlC`|a3pt+NSuEi%?mOE&blzZP+`Vz#66;U;}l;C-LP+<2}{#bPdX!mISk*TQQ} zF7vT5Cc&vq_4{RoiR1AGP1*T1-0~}<5lOIFe$(>D2=g)erDLO`F>Vs`q=(kdZw>N73c_R(zVb7i`CL)@xd#%rXJF9; zjHeSHZ7xA)o=geM4rU8gi5CRD`b?kX-p**|5TH7-Xk|X5>3nurxpe^YG>m5P;;uUM z%!!ApFZafcryu`8LY_&a8y5#tfY;QjTeW7Kd{cP8jaPMX1X6SQ$#TXJ^E2b#(>uGbcrZZmnOY0Z^kt$}WJeQI3?5CtSp{y#i zNmb-55A8^CKX88)!Y`~96heMfAZjhS(A@k^@S41Bv%W*u$f4$7Ym3sDGjaZbr!>sK zn~6K6Kwr7r&i;_nz;mf2qzW+SOYkH%YLoy$R#Zw9T1RMz<+}05{j}PjjuFh%A7Xn- zYJZ6+DM7fnvw6xZ&q*uxJ3yj2B$ zDa=iw^V22{R?#3Z`8rJ-6e+)0H;T1@ywKRac>0f*R=3O*m-)jdhsCQ8pv7C*&RMWfc1biNn( zn9?|e3O^*jY41YM3Ot3>CqWp>fEGvSL`Tbh>6!9$n4l9Jo;V7q-S)Kb8cD_=gu+^0 z)!hq5Aqmhgd&g7m-JEpQMDY@C9PEQ0o9pe=W#dSA2GjW7@B_D?*Ts|lA%*lh;MTF` z_p-8&@yD$L-Q;AR9pVCi;t##&3CsyasabGkbbDqgMsL7=?;Q2ovqAjad22Cso*Dqf+a#MJ)@kI(qzOKc03 zj=X2dV2_#O@@}l*gdV#e_iY!PJREMhJ%Vj^_n$QK5)%rC z!oSyP5^BQRYi{eyW4{u3x}?J<>h?7o*=ZVQ$SUzl<7L;C z0PsA+*8YcDd_1diyjCF1+&Cu|0By?X%5RYbxcyj+T(g#y72+SH@nlz9l!L+E7 z!8`aIjsJ0!I9j*d140hSLP8!M`D~%a-6e zN$VpyUFK3%kb-N+H^WqjSr{2%)lEpz2Vh_}SBKp*iln_FvsmR7DcLA!K%QEgAY=zw z9cg9eW(tvv?UjAj<;v?l`_Ooccs105|2JWeP4S-2(07T>q%IJ{Km?~CIT%Wiaaw!$ z_ERrDafp^Rn_zau>YtyoCNA)6(e(E z*2AbtCa4kS(@MzQbr%jmES!*!t*h4O3gCO8Qye;gQ5ER-vJNMrx}TPeZdg2adg|A~ zU2;v>UzrgP0)P5SG7LU+)Yfg|Y{mU{GKvFk$^6YmH0ffXpomPH9@nTbZo;3h(%n%c zYa9CsywH4FJyMI=WIZlcE4GEbmoAam!5u-E%f+2(k+`OoZNV2n18WRal=71@nN<%+ zNn=Lbz^2uv+gCxLnMc~iY_M4DhXEPIJ&NljCBfEke*GTK_4>SbY-$OKX}e#^*_nI_ zT+zx;W6AVmGp_q=I@f0t46Qs8->hdTeC5mC1;{G?2&>?l(#j*3(-N*sd~JPJ?mDaNqzT+pnx@J<{%>G-x5<7iuxnwZ$Tj!h%wojt6@e5Gln? z0~|`nQwIF3GhJ8?Ht?w4E_EqEun_8SEbD12sZsM?P`W#%J?q@1vkPT8xoee9nw%Vu z;({U?HX{z`3U!-YLixT>C6Y51(Sn=1@aI)A^Sad#eVYhqrXBDKts@@qm88JdxhRc3 zg<<0FZxFn=GnCLfeqjcWgE$6IX*IJ^fig8oPW7fT!kcP=O|=uO{gQWAq|@9^#mF-t zXqy^fJi9X%5nPn{Aj0gD75anl%j;LpDmu4h$|+xrOnUa1&XjOy7V{&fv&>VKRX*G9 z#_Yh!wd3MG0(B+k5s?Gv?Swvs1d4($*iEFV@@2dSN70QK_z@!)OD(Vc{Zbnl2K5aJ z+UdZpM(~br^WXiPP5q5_5?+U8>wrgd@g@JzX=;Ebkt(Yh-iYbg@C#KLU<$r07G`+mXI&tF(R4GPK@wCOAj19}YHvf+`xhb(Dv29k-;j zeMId#Z;G22;zb$TlU@&+xUi*Fyggo$$z96k!KR07ppS9mTNMq~qKiJ&1(Au|EwT4v z1YOhcdmcAqF7SWDGtDOY;M&upyCRU8#pWM>ae@3Gibl+OHWR^Ot+T#G(*UaO&iHkk z18`Ix7%G0Yc1%uF$2M(AN9)G?V9pT=DTFJL)SlzS&PB+<_ZT^9gxG=R*KO+@J?Asd z`VpuWXccX9BW#eX@UHmJ7CXt-?_TeEODA%q#hIa-`qse-qVeI#^xu?@uf2Lu97*D% zuQY;AzMVG;J4GB6_@U?f?CUPJc@;2wc19@)XPGx1E<+`dg6w^mv4$uo2G=1g7P`|% z@WD1neqTbeevrio^CHh%;k=8wMK`BEp!kc?XzM_k&}b z#SVr5(EQ`1*HbnjTQ^4wqx)2h;x6EMzwFWozPpB_`Q9ISf^@4py{PQLBO}pjShRY| z{k?D=&8-ejA$aSA!D|h6hz33bd(ul{nFTonXXw9xZOu4z+;%8Ys)GS24_3CefdNkP-?6RdIZR;y`tkLm#v@*$_dNx{IZmCr&ggGa za))o$&oiPP`B{;|{I2)F@w)>Vm#lWyKE+aic7!ooi#5rLxb_csaPN<8C6XrD z6D_uRJl9@q?()AAjW5^NshFEAFvfVCeUjgN6P4P`Ax82KT-)#jOa$97_oxybQLtSR zD1z2opl_7h^~okaJx85pW<25^1%PVQt597+8!4)726;k5Q#7+wG^0ejN+sLERXd7L zg_!os^~jTgXF~k`e=wyiREWblOi9^yy{Al>lyW8xe#6%E{xXx1amaG^F;OnRRL2D5 zDeoS2;wg8&p8vFF5qn>&9>%yXMNN1gM1F+I(rYW*#-Q&y#x#5y>!Y*NHB2#aE&NHyzAtl+c1`M0-OBOk4>T!(dDEv-)y3Tp3_50p;) zI8HG!I%qZv2N7)ADe=~e>e%fp4MjlU{RM(;J(_n6ZqX$u;}STT_ySz&a4L9QiEz^U zyFYQi2sb`~5jQ#ypH~;={{u_5Th&;Ny^s2kESnxKgU%$sn`uxHMYkJs_PS6~2oWhF z&J^{61}fW!P3fLC{rhilMJg=bV{??%^zt+?m;)`76sD3|y|3i6iy0LD{*dZ?kd!e- zwu$|urKFiWG%9Xrv8rwbrMCQ<3NI3g@$cIE9F8mu3K1cBh5bS6;=rrBx8ayCdHR{i ze%++swJL#I&4}yJm6f}pbG2NtRuY$|+|M`4mAdCn@J_4*sc`zbM(Se@$GKU; z?H7!~i6NP59cnbPN>!oqYI9NQ+D{SzR19u9l+Zt1uzHItzDo*dxE}LsOcS{8VB>z* zuBlymih-X3GS$YQiZpc$92wLZ1}-PJEb<2UaBSuH}kjq$sfB2zY@7D z4G}cYc%1; z8(Z(aWJW z-!pJchJVrCmv2^7+awmaQ^$N)kC%{!M2E7cJlYVe5|_EsQNAV-dl7yd^RY0wFB}>F`e^{A6&zYp#7?9fQYMzkB0b!_9U%uW z0An$De68gU72e}87n(pw--a=B#HR)DajFHkeo(3egJwV{=(dA%QcUnu^m@E`{YYI=c^5Sh$_)8A!C&C0T#(c zM9+9x-03IYM@m{X13%qX!pjc)!3=Ka!$)0lDj#*>1l>Vc2HF@`M>9`-?X<@GWP|ES zhtE1VXbnW)zIEE^X2Q1TlWp^nzn|eU_sF0$&0F;Q3EVR1k1Nw%Es;wh9f5aK!UV7o z@l_EWkQ@I-Nru~0Fg2adV%|QJncV1JFR1P3or*|NLJ^`E#e+aK+1wo-pOD{J8O2VW z&z~eLb+bjpdv9m%^?Wd&^s1$~cBnW_4}V>;H*^b0=~=+RT-Ftw#6la5}D^e zHYfI_(Zc5A48E>Y>wfk6?XqJQ``x3Bmlp*K$C7!pVgJCA^@I7N5NN3`6)L)}awv!I z3uq{O|16$vNAlCe!Pc?0BLCGsdLoF4N=BLCl7BMmAS){iPx~^P$1csk%;2a8ivYmO ziO&Wca%Yw`xV>cA$-==61YdS%$@Pj-2sx|bb1>;O&wse*%4axc+|ERO&0C6d+m^nT zsGJwN7ZR0wc#K3RbH-@}lw9z5bJpnbFmg3>?$3X(>10Bsp^Ld}0MOGrgpytL6iG%} zd?_NxjJX0@oUWbd>K8=phrn_raa~laLShM}${}qt8KzJrau?LunVCqs^BAQ3CJHr` zt3RcklHiBHh{q4x`Oc&`?86B$W|!o|ftLAoLi^)uAoo8eA4*&ORI;Cx6^X?SJo-t6 z{k@H=(aPbMc^0xc7miz9guTxp%%he(k~LYin7e*%#t)CAa7?;84TTd$ML_iIn$l>d0TlYY;UxXMD#^0@88G8Sb!B~R- z%lYjJZXfw+(rcXxzQRZ8^?pdDXQ9)xEX)6Wt`n%gqZ5Dh!%ukCb&S>2uFG{3@4lCy z(;9zEjnFOogI5Yqc`SP?()w#cQ*Ut#0effLX)8&-@1aedcbQ?3L#MuS%X$7?yFO<@ z9pvHln`rQv|3Qru;l1TiD%Ps+3aj(PZ|xE9_UoD1^WQarhV|J-O*o%jEcf{h{7!hB zb3hppe|~+dJ1=1m!n-|LLfw&WeyH7tuLt)bcyBzdquFWIeC~XB>3kv&r^vd%qWXhR zWk<|HxN$S-)%~E(g|JmSL>}oBc%{#In?GWhbvBjF(RDMQvxVl=Y~<)>G6j9VY9)#> zWMyd%t?}_RyUuyt=b*1`Js|QtHi2B2X~FNdwIW9zoPujgFO&eMV z#OS3~F#9{h+jN~BM&ToB;BLoe|Jn`T=X=8EvX}2$=O)~`cXq9K{@y!S&F;VD;AE(n zoRmz%%zU$}pPQ_v$d{^p&Z-f-RQ7tI+mZO{v(U;iZ5TJ z$;ip-H)#-5pv%DfXDlNjE5}UZC=&@=YfHCez3c1~zx#a>7GS67cKp{j8L6bDxo@`t zwfMDucjRW@rlL%C?3^hcbiJv*wvdIb4Aq?vp?%IL96+hJ+j{6H{MaU+qvq|DT2E3 zpgL7f^Rg-Oa|sYPn^5i3FA-KSVoLx^R1olF%roUahm@k1baAJfr0X`3Ery6e9@?>= zr>t0oa+Woe&4Wx%@!fD46Zw>;KpQnBu=z*X{M5~XNsW&subl4rVhl9HL)iVx92I^^ zQguHTu=NDR?fGM*WKNIPk)GyLEUcZ3R?@I3M}QglZkZd$a9_G2SFRFMD`cs2|J#xE z6$9h#Gx+axo-$gn?}wjb8(8tB%$bpSnuaBqy0bAgYe%MqdtQ6whUfhVV7(Z~`OBZt z-lQYXHlC3L+ouB2v*r&*f6maf0?J}Sl{Qjd!QQ+hCo=;Kj182E`)4e&$5)H1{K>F` zVaU50)?5Rt8!NXOaHd0f(mcbqZ;oeE0rdfcg;YM{L!wsqZ75-s(bt{~F{@=`lwi#H z+ZMHC9Y3h0NVP?d>#4PdDOL%rFgd-uU;FkqW_nYJ0>!=i{Vr3z2h;B_i+SbH}EDC(wBhM&}fs@R?XqWpg-*Rw@DE6$V#mlV*5(@{Gaf|@{v-CxOfPt zSCF;~XR_cTy=fIt`q_9`=+_eWA_Sb5$<*Gaq8QhM#%F**VBj2*`N8w@t~k-2yjs6Y zIY0iQP=T#Q)){e7mF44Vmdiet*YPS?Ty>UT<8fu<761F*xlb}h8I8{9e02_d%H|nY zg7Z;zQ#VK5*wxje2xoe)PWV93F-*tfA#=n>J@)zh+#yu|BanWd=~{zZMy4<|t^mUn zt(3SA-%8lD3C^Cw;Ps!Ji$B(^gV#PUtvvgFhCMQb#Q}XJj=>7-9Xf-BJK?xPwn|mm z!SCy?5uay6UbAMQn1d9Z<6kJZhHah*MBQn3>rd-mW6gRR$ElI}FB^K!*3nFP^u=xQk`3phcK6a%VyP4|SD zSqywT5;Ea86beqYemgjLgb7T(;(rHWgAazvI{D040$Ruu33BqE6{XV2&$vo; z6K8XCheYdq`g*f+<^FDDo$R!?Gq+i`a;`^>dK}X#@jzaZ9(zg?NzF%l7yaVY^GAM} z*$a>p5A5BxgFJ5ts}7YmzjEVHhFYwt666gQ5959`KwPgV@V;bhe)`cCnoU-ws_Aa$ zrDT-UjWkd6Gky{{ z-ssxcqQ7l@|EGs1XxT|<8c`mR*spm#tl5+i^(8DV(+(+PJ%=$p-kWghT(broj7bDM z*V1q1`*qQ1G|Zmx*MuAHE*Q8qm`fZSEVp^F``>)`LMdc!3mVu*H{%@I({u8jJqfis zH+|aYI}5cc@X>-(+#3vr$H=Jj9}--*d~DSX9}2iz!`OIx#i;sBJLeB+tA8K~Ys|>b zD=1sJn(Bh`TK6fNJ#3x4tT$?4{^3ow>VKu=Swtob2v~hPgl-#Vewy^|`C;58ek#lw zF~7fc+SY}pB6LyQd}EO!#@xU3bUokHIbg#Bl_+$HwCZ=}74~^|sj{vv0NSgj>)0m#Y*43{8Xjome!OnFG1`?^IOKpurs;gQ zlF{dv{oPa>S!J{#Q+te&_6br%G5?6RE-HWEaVBk{WqYVsfl!-{;d;0l*V*KH*V`vK zz7e%iqt`2th>JI0bGg!6f#0Ml(OSZ+#Id9Q>%#$}eON;*fw4a_I+?)XsRKpI+W743 z_b`;HCR69Tzg7i!Qy)O%(Zbxv!rKP~vj_;ze>SOT-17-ez8{?m51myU_VLL4K&~Ke z?wS(5ljxRVu{3q_%k?`2O}az$RutH~Sj=eo3-yPq^2Wz?0+Dip(?GW;bvu2Qq<*71 z-%KHXGnt(2Ek6O%u zVBQsSDCMk!%FfPDN1o^nF9&54_UwmRIE63%LC3T>Hd!ve*|982^rb2TebW-rJ7^H}5Pl$XE8NQk{)2PH#!F^!23I3yCmz zz?FZE)=taA)BU>*2uhR%%1Cim<8hs~*V@*0<|mj47ap4{=82zo9aZd)j+4=NO=|*M z^xvKX-A-k`>yZSXJc2X=-bC1ql~%ix&1=5^8d?^LA-e!ie=U9VII|beUvW+6vi7QZ z0f@p_q2$Jwqb=SGzMfa*#q5SYM)*4Z>}S4{nVn42U(UD&9lgE^n190z!uh!p-}wph z{XjF-VgX=4#R{O+O&Emvns#E#MdY*ai}+*9u;$_KA``Q+kguMzE-lK2^Dgo+HX~j| z>I>@Y@ydA!zcwOMGe&~6%5Z28Mw}0{Jlt9Ft}P$p&VFC8XrS~0Mjzo1eRZtLRp84{ zzaiS6oPEMQQ4HPIyt%sXOj33T?YaJWV51TZKsNP1+r0d>TFC!a*-;8Y_*02+FH(sT zQC{#-AzVH@O8J3sl0-Fmhaf^pAwO5=;w_D*uQK!@S;|d@PFhAkKX>fA{j4i4VBLIc z?dstmAAh|EGDw~ggJ9u@7wOZ?&>W&x#st|#At9gQ z^`Gu@7Z<`}4q3T+l%5d)^?!Y0oq|zpcijE!-0DK+-)90tcH?b}*q~Xp2O_fyuiXT? z+=(f80l|VXOL#Lhv-LdY@Ji?pg9srvhdVBo@c@8j0*?H=+pO0*36hHp>EbXy`X>)o z<5ScfwJ$CFqU-@pmIWOIVfpEyV%Sha{q;UQoR;Q_LIg69>P?$kmA<_ZdXNWz$JJx{&8nq}tmD{E3u%~rHCex0eP9ib_ z`09XMd^@*E+zLVIa&kY8cD!P);ig^0xrIVztm~^;-kULseXnp{CZE_t7V_$d$$j>a z{4eWN!l!~>M<>ola;i684&IKz-q)Kf2Az&Dueb3lTWD?bBL{wolRaG>XLR=|^hOh% zDUH{Uu{Pu1brk)=IdHJ2i)Z_B2b;B3w;w(xEL(>}4mBl^7h zl~SLJ=e8Npv=Hja!d_PKU32x4dgGM2fr(5`<5?>*+H1 zgOIl2j&~uqAv#+Q=D;6~CW!<0XAX(i_mAF(L87?IX5Fa~IVIyONEU$z^Gx@TSN`Nd zQ}E{Ug>*|IuH8kE%=>het*!<|BbZJ7dbwU_dM8yBGa5)?x>0_$zj4&9TfF)Ykfw?k zzDKB;-o^o^!dEDAUAPXc;tJ10`|X(C{^DqlS~+!^1WYT30$mT<({JZARNxyLf2AD{ zgVB8Hi+yB%9Vg46Q-o?8Es=N*4i9VxUXv%12{sBsOPs%VXUL2HaGaVMJAwO|;`emx z0C8N?TdD?O-;dDe*N>|fRo$5vY`ZntFEOyFi6oi@kOv%dVDXa4{IVK-F#)n?!)9b0 zXZqDHzaGEuNWDLPF>odoz&{jI{7Tl=*w&nd;(4~t9OAy+2tQ_jBlfrDz@A6&0>P&n z77_jZY~SF?T)kK4-C%r9cm?WYz-Y)9()eQ^lJ%Fd-oS$&WM4}lK=1dnE+!kDFCn*Q z1cW(^&n#N7;zWHz6hYL}>knD&P}{Xj?LOa71#&}VC;~L`K?P&9q9h}V*qg&=pO=2H zlrQ4>|C#<06)+|d@zmURt+pTC0%$Ap%JJGbDQfdN30ZDDF`yxwkA?zypGO(gS#}F3 zP9eGeSRNua-}|y?dhs#7B;I|cmj-g9xPI}aV`x(YW3g4(NcLEmlfKv7w_T8WN8Pku z9ZJVxb(B!(ZZ-D&uS$?~%jX6gbNK0FmMf~sh{V9_K29v6Ht+Kuitd1oHWh|t5!(oO zXqC;xz~esWi-Gy}t%$dGviVl5k2#cOKI6R(O=~D;-b0`ng8B5$%aWSzo<5n-qx2`& z`*XGC%h81%8Ifn`%;O)}A--oFci97PnY)wc#F%E-NAo{I5`(tQfE@wy+(btH8~1~J zecM!swpTCfHLjJ0I}dxIk;o4NvOr0f)+ z3cgPd6Hb>Zk!+e#JY{0xeh&}7JMT;EHuPFE8?+K)d8|Afy9(NbJqxq@(V-_|6*w96 zeAUIh@mCP%myO!KkyHBxg7=x<25eZCIMo+*v~%_^ns7NfnYn7c%LMs?ERjxe1rVS9 zlcA>si-VwNkDSe61G29#BzIk|p%T!Wy{H5-C19>HW$9H_4c?1w+5SozuP;`@tzB0Q zzNcs81n28v=!kcoY=Va_K0@{J z`z&`&87#Q)SW4toDKFgA+6l{Evhdnd^4D}DPPcHFIk&@j7bag@xtE7jo0sd%&Chq7}E71#>TK7WKqSRmk9(R=fQat8ix#&6RX&3^@Jfy z`1mXH5V}cxTlewC=A5SAW%=1FC99i&r_ZjgA2y)udq!44`~L!HKbOGX#z!K1mn2VE zxn{T4?wB0c9qX!lTLB^Gh2TPCid8*l*Vm;lc|ncIdA&fYsw#Am%ow>?#SCHWrq>Pk z5u%qP!ZEi(7SLEvKKEngpOwca1Z7t?7_#Ti^TI^1pdISe&_9uw1f#D&Sk*}D6`e&>a7-Ugcw;@9qc8urZQsr*0t(C$bxIT_dR zqA`<%9}rikPlzYVq^U_tl+?5#OU%MTc5)UB1DEU1+7Mxq#|E@t5f!9`wj+~@Jn5lE z_^dZ3^+b`%#|oZ^>kK%V#B1_XSN&}5N zW-`sfWa?C0>8M-qQwi>|W}^O`JvqMp!<%c@Fne+C-F>Wh-T&~&zSj)E;$ofjN%nNl zHOb4@P+&)T9snKC}1F!V#?~8KlZ#TJhMGNXT9=0B@!3?j$cOvqmmMGZyLCj7>c_r@EaG&m{BQq?y-973a1kFMQ&nXn~W@BsTBdo_;g5lXW2O)j_~U zI2g`@oF35g6Wglvov^SOWUQ(yzy?3;dD zw>bq@Hoq5sXFYyzj(%_Ziy<6=TtaW|kC5|x8iMc+2SpYvCAH6*3OkNiW2P01or){{ z;X`}jj!QR*SLT=Ac`^J_@^>bsal3iz6nx^kIvj|9*1xkW_|$r3PUie!q%{bWVe8f{ z@Q!!f5wdm>QD*=0!d)$OhNs0w9v?%v^wLX&Y{pFqvwa*#IRE@zrcbxY-}(Ob-d51E8&`lz_6M5l-+c3pE`N&kI(3{Yc>DGp zff&YlRIa=3Iyijf2#oc54wvYOXJl$}N&@P-4s^ZzvP)&mJQm{FwA>FSx_MxB`;P5! z`)#*MUl>>N-V%N{7hQP1y`JeKkI#)a-e?5ZoB*bb+xTecjT<*g{#U*xdzr4j`YPDA zZCk)i_}(mZ_-`9FtaU(mM?ii5UbHul$B0uHbRCw$$*wqgl;n6(W-QMOPAnDWe+32A zEg6L*v{g)6TQsaQjI<_>W@FMYk3%=AKJ)0_psBFd!osGhu`s!P$C`{ad!kK~ z)?Rw?#YM&DaBh&fS(QCqz0o>3(m`>9TJW_iNr~JZ4Zb!)Y8LV0w-jva4Vf^4gcqrX zVZ4i&L+d@wngKf-*{ZPvaLJK8wcZ6Ou9{H5AF8wmeaVUH1sr2TzKldj*yFS;?xRrD znVHL(<%R4Qmlx$-a^Bq6mc_7rT(xREDPTpSrSLiIe)V80kCms@*FYt?lmausq~m9< znYtL20+l`vtP7R}pp6jL_!fqQ&CFDlh+reIdCWVWdhhbpblmu`>({e-CnntZKpsTM zNwWKWpWtfTRS}3cA_5ztUE`M(~WNs?S$J)FTD(R-+hlvK8dA)%GJ7=hl?({K)4$04LE%0Fnse{ zcf-L~4@gZLZ|n0SWOCM9-ts14dGd80xc@|byJbKikX*dE++%S~|JdAo!( z_2Vah4Bxx&0Y{31fTM6fId$SS=3tM^-hKPwu5W!?xFTb{UZCCc?|d!J1kbgAI7aB6 zd%h#2GhUSBfMdQMFM3Wq#AG-A_y^zr0sQEphlQ`hDQ($2VnN2H+)YM{VEzX$aL(Uy z_uco}#on<(knxKdBhW8@-K9q3c!O9-oLT-`JU#~xS}lrA8+mpnhpmCPzU9qQOq>M~ zXWf75sb@qO_!L$=-W(or+ikZQ&BUh8eB6xTefQsQ-P~+q4XR)sz5eyrz?D~CA@hQh zit{_pA#xOK z?TbZwN0wHj6XKaf&xyHr$N$VK>*2#k;731tG^oQh0IP75;AQTJ#=+RcWtY~P*%5OM zCWvvcpU|@r!!F3#%IuYFf~Ti`{4~7s^2-uK;;+dW&W|45DT`rh%I1fC>E&07j=nvB{By*8!nW z+2xmA7J&1k#w>dH;YVe0ihI^+3CGOBUAKO15crJ8FxhPP?%k?92THd3EnL1Mgcn#M z*+UOMWRoMI<&&^b=5!Toiuu|)`w#4wjCHnrXD8?$92B^7*Y-dU;CJ=-nSa~#T)BuAQ19uiB}_q-EZ zwQpP{N3s*!COonvG%pmnAACJ@bbSw~N`mPzpUuq@(pbyk{#%AE2D@LX`E?*@Z@fUK zS5M1zz4pW)Dfc}P?T|trEyjW&6{e;3_W}onaiwW@EwBCANN7~!BO__Y#|YpF$+YIN zt)5HZ)OEtm-P??f#dGRR_53{!+5aV4PwF?Vn4T%tyOuxo&ekSfDvsoBF^jQ#+QADaCesEYbwo-6F+$~FC_Z)KF_k`1;N2(?5Y&63Fpul8;A94*W1c^kud2Co*$ub1m%Q|rIKp=;CR;6WlmVzznI2}R? zu^OvqG$`I>C5<0zQ(&WJna98jjsZoU*!_X6Z);OL&II#-y&;ow-jrJiBrZxx z=QukbSeCqKJ?P0mr!TX?_vULd%hMOt;)OajH6iX@0b0#79zAwc_%+;LAiIDVgq$=B zg1sOTNnC-{^W~j;-kj?hyZjt1xK>Y3g}fiELGVt_sSdrcWP!}Wkqc&dklBvUF}b(Z zmLLc<7EsI&^5F*dyjeNd>jIv$!UDyY4DsXb`6YaB79=dp`Q9EN=W#wR=gx&X!Yq(k zU`T>1Ng)B^+T}7AK0wn7TZ%k;{%;ggGNLoXZ7v)3RIT4oNRC9c>MX9|qa*7Y{tmy{ z-HW$!mYO%`q?|vX<1a8v_*;W_tUez+aL`5BrZG5rchL!v0ru?K>nPVDX3uJ`iGy94V|*ri4b;QOvwvss??$Pq4BzS|tODjh$8qz*9Gut2$Y35wRPSuLznD_95eyvz+^?u&n)yl`17ImbN9 zxQk4F_0tYk3A-JZTzIa#va7c>VR)!8`7Fo2dAlt&I~C zF{zBH@P47P#&Pbs+oX?dBCwgkRq=lCgC7d5-j-@02z;|riI5XQ@dGkPAgFo>;(MWiF?!5Es zQsm6DRjo-gFPh57KmHNeV1!g>fbif0--o;Ix?5@@3qHc8q0@BPVz(?jy5h%~%M@4hJsE*!6Ls z+^YbtiXq?Ig)oq<>Q|2s1z{~gf~q`-LE>D0nRXGrs_rS4--T&&?fA)A>K zuKs#m|19$v{g;9CrNW%Zimdvf=DAH^JVS>*LQ^g)32=N*Zxu7CFx z0Wky`hU$HeH-c(6@r+r{oM!u9#|>d7OzN!tDTz(8;_13blpH0l(`frY_MecMGi`i>#i#npcPSYP zV$wrYzr~jYOC@BzIAZ7r6WC1iBy5r=1j-W9)m?N&u}Z{`RAPsKy(;J8pkk6~W?4uK zrISBle7#V^CIuxZCljffM>Wp_n_PiOp)Z$8Q}0GP(huY6i)jJCkWk4}>-{7cZRKRo zNePqX4ZtQ;XeumDoouawP*RiBWVJbS^dzm4TC%ZcPIoV;YHvo!ELZzW{qxpi zQujFwwZpC{R z=FRnwnd?mq>TGa6L6Tq`=y3vj@A-Li4fa+zD*3VF$1M2^AOy_K%*@L*1OEd^erKX! zr82iM*vftPi`(-$#(HBa+#Ur-U{|c;|c@mw7j>=X%wBuu6ij3KD>lP*l zuaK(p$9Z9G9F6ssbwhNnecv1q$oKXpCOO>7_G=J66Aaw_}6% zD6nknory_>^QvZ1EnBe0*HTMhv^A5LS(rT4N|BN~x7j^Mjs&)4^Ct1YNcMP#2Y5nT z&<>*58@FLaxlRaX=jUL@&h6rsl%TrlRq)0e-w?`Q_{d*gfb1gm;3s=1uDId~skXRWljPBaCOSBmlzN;pu;D#Hn7Xi>yIYs#1T&a(-j6vVBgs*trWm3n+ z3RjQ}|15x*Bg0$~5a6~CJ9g|a@A6jho}n&;Zu+os(?-e7!i@=roc6%i2tt6EV?^N6 zU@FZNwFYEO!Fd_aT{gue>Nq88G`B~*J%@k%XMYJB*G?XRGhFgw6k8yzd6uhUzoJ{I z%vs>1JG{g3#2Xu*v}$%$$$Y5bU_>avTmTUMLSnq?p7e!%>0-n zt#)Y)26LgPx6ekFIK!;qb4ivwplg#xVFI7j_x4r#B+EU6rM-@Fkl8ANEMx|*X0b#d zcWg#9pFGO*K_s1`yQXIcM~1G;oMuxjR>8?211%ln-?zfb9CHBVd7 z&$FXr1J&X$M~{zeJ`?WnWY6wM@U%IuYB3XSu2RAsh=ag-y>WQ{`4`}U2OktvUQ}LZ zyGzcJ%P*AzSib7rcfb2R;fYw@2!(~jX&T@7#y7~$$>t6x?qNcd5A{c|*W+FErkihs zb?eqhAGng;Lk~SHS^-xD)b{*US6v~Am3Zgpzww2srv60ZcJOa5F7gPK7Kb$pX^Ir9R3;~Q>(bA+_l zfgeBhj3ikS&yYw>qw&n!V&k0H>C($Dk7|SJa(~WPm+beTxI+&5nyxCJZV1s{u>80J~-G5)I4*HV^T+`W%Ip@!nY<5*u z7AA^=68$m4O;m<^q*3<^KPek5$5BjAu6nw~OKMafJBgb(^}@J!%L0HGbEjJ~V0~~} zy#v~vOyXx&O~J(Ec<@rN24I8stft^|uTHI6Wo>*XOzih39p+;D!BlUK$FaFzyTQA{ z0;MZCUcH%Si*fq|&R9y1k*saLa#{JORsDChLJgH7noGg%#>qNU}>!V~rkxDoe(0lKG?IgvjYR4)>&Q=w9YOhl-MqeI7 z`KH6pm-Q=+P9^PaaO@gV(nbG|HK{KGg3_=7RM%BZK8m*HRJF&rGNQVdW}!wYcLS=#^XIB2p=*p8PM3_kGIDki5NusvIgmC7)DUfDKOC^j4 zU>odP)X8*Q!r{n81d7zwwAT z4R`MN6Kq}LAm}2~0h{g1@Tc$P5q^SIw)oVk{4x*kbE@=%^0YK*9zo(zx4S(_J&xppz znw4WV3Z8QP^{C5bX}Fa+?TQsbD>zbJ#Ril5Br(THvgO?{eiH^Srwj_Xx4h}k_5obNf^5| zb6nhrbD#PW5&pc|Ym(X?CI2@SW_3oUsS6`?gTI?K6LGzCUNpBWD6K zz9O6-#eB6fNsd6Mv@AOxT!lwRF0?5X2C%qrV2st((QDrqVd8#y25iQ)74II7kVjG) z0M>RbITJ{d5Kq{`7)emo}jxCK@zm?-1p-H<$)54NwF!yy^y$>Ih4 zIp4>t<`ReE5>8LJ4C|g%SvkzcGd@|+O-KfKha)HtNhU)Rpf!^dr8(bly`gDK7ZJCn z@m_SU`7v>W*PIKE;~Afv5I&4xCKAWv(knQZv1bO4Pmaq%mO@>XFE-Au<8CnQM4Oo8 zq}|gtbG_3v`Pz(Qaj>N*Z3By!+4nh76A-1|kt@jN!Z0{D&GhtY5%S}l^r^C1E@Li% zpFW8#;`YK?fSJ4IVW7>T}sqg zww299+_r6d5lgWl@_osq#(x(EXH7bY8mr45T;w#0zMxg=q48uaZoGT$*s)X9t|gn& zpo0zC$q>yUtKgtMs8l3E$;$Bs533YEDtn1@50b@Xq_&R3ELN{xWftJtV*h-7Uv4Y_ zxSVR_f`ytmW@`>x3~S4lEg_#t>>E0Y#64oyFI3R+X)*>pwm;{2YH3HMWb)~GTwhDdN_0C5EKKcP zDBXLQ8@bag#InzD$l~u5OsN95ry@s-q2PY4-(!_T?k0u7BAKBQh-c$lL6nCvEv^OW zcXfX$WMPZ7NoDp07kKRoj|y^)E(7*KFGxg~%o|9VGH9r9zf|LC zek`pJ=rRzIqiC2irkJn{a@(>ID)CWZ>}dvXN%>HVcv##}>77Sd4i)VnOavK5L_E@6 zP-SKd=J)8e@91B8PJ}Z1rfXip-MHM#Su7|;bUMOx*I2;4iRDEf1q%iaR{~59yLMxV z$b!Z|6vvObQA}hE=}qvWv<&gZ!EgusSQx)zpy6+&!i3v^6~>v_`R}fp9B-l_t$>zz6EZwu zn2KM}Y9Ll$BacQ(%`eUvYUl?xV-=_;uoYVD24?QWbI-j9hYlSKNu`_3ulRmhsRf?| ziVCja$TEvh@KH57y+aUEX#_0{2SL+jv(O!f^SZl)yV0-u30|YVk7cHARI;Mc$4yC& zRxFq~PHQ5q8L4-Llqi$&MuW)|E0@t2iT3qem@s9#KcHt`*t7TYi+4PXD>WT~cMa>% z`9_U{Al6R{z*`y!7Xf_@i%yrt0BwFGTUCPb5F~cFGM!j>96EFujvPH|BZmPC+K|%} zvUQDR9Eq@yNh~PwPP0e?1XC@HDB18aiIgp1ExORU5=Klq<1~`{_Z@&W)2rq8s!D`- zv((eJO3-*RK^xI>V?Y^;8Ve@3ENlQF7!ro$tJWa#$h5~GN%@*;Ae^G~8!Ok$p?K#) zifHs2d?PC35&<})#KjjTh29|{5mF^hhq8O;I3#6>hvIDMD6v84S+G&aKpns0g$!bY z38%F)%okw5h^UYWA7dF3VJs1z>6kO9H9<8#ns(t2+aUAr^*&g~d>!uqus0Yg9xDSV=>HTTFzLPQJ&kggC*A*3ONo_hM232h?vy<>F63H`N|~neyJhjs@SlRYc?^phzj!rs9^3w z-uI+hB6yUd4wxvVy1PXVkzWnBx9MibIJpnbbkXm^VQbYa66hb)CdXYdu9fAIn0KY7HauY-vp+ z%Fa$P*r_#$$g`7zJXF&>QhW+h%?|4XRj3jjDZrU18Sl&#RMMrWy+J{Xi-l{A%Tcp4 zV6>zO^U?M%Ml<2|nH6j1q*d3F0##WgT(mGi6{AX=0?{IY9NjO__p=28sY+ZX%;^er zQ9TwdWFq%yvRgteFo_aZd1fB~+^iHQ zYQn_i^Cd?T(+m|ziqX!Hazrqj4?1%lK|zCxQhp}pU{S!Aw4IT(b3R-8uoWha*hzwQ zr2l8`zIXy)3Y-$g+0oqJFk)y?8Hb1W-P{l%$_bKFc^IDaWTUPAT3MVKza{ zLP2-b7E3IksD#*Orn1@sDeqGW-;J<^Xi;H}`-Fu+P=1gUvq^B7V0P5IJdu7VotxIX zi?wD3IUheN@~+LN()-Y3g+;ij8D^k{#fkBK8ih~nTE+$EQIiy)sBAnI!*3G7KN_&f ze{A~0Fk$i^R-Sp2tZ0So$DRYq)LfkxBcDL2EuDB$nq0#fc&c!MVH1p~u(&7#Q{#+s znAEhTLJSKw;5_bmqRjNe0t5>HWmC@GXn_|zW7%#_&6`6;QcK5J69gB?=#8~V$aV%* ze<>}H9`}oKdnxt$(a3?Xe|9p$rY23mN2=iQ=M z9Z%hvO2EajX*fK{7a)$D&f;IFy!fa<)*32&F(&&_lXihoqVMu+Sxl0rlIes)6^s~b zipLI-5{!!)@YDi}no=-*B-$(H_~r=i#P?4L&jb@QpkUi1sxxWV zusW9_FGiwjXFwh$9tK4NO}!w5kromXW22_XW58Z0^<3&jTHvT4CYVa`{)zVCT)SQv zkp@!;3mOu}|C8SFF$q_Q1wox|1|f6BMnpU+yaR~GDJwt@v6~u_lByZjk-Xo`EXBBv zl2R>cbhMK{f#$S&3kLN4;mR{}u%KV2_4rdRh}4*4W6dB^RSKkPR8|HNhM5rY1@%J3 zVjdP8j>$B{8Rfj61>C+wnb}!uP$R*B{jq~S8e8v$=bwQw)6vT=+!j7Od}yCJ!mt0Y zIezubC>9#keYSO8E*fO4t=Smb2Lvz8&I3qCC%P6ePdjgu)C&%%^zkuG#9 zfx0?@QciR$UbB>@v5;kKxLhn~RHggiV#l|%*qZ`(DT}+R%Osx|nXH^e0Yh*jMp{gw zd7XXD ziuX%(M~0kHtEmin=nto6fPm#Kk|>e@u9AS|0{MYnRmM6r-J3JZgwIVg^Ftc@YQ>6g9MwDnQY9b}AKG$4gK1L2oD=vy! z&v6W2ogs|UT{DSrC7sYG0g+Jw#FVBDh_w7Xk4og}Dw$7MimXXPl6khND&tgwL5n;m7f-8#=JCxh7c!<47G%FBhFiQ@U zFAgeRyQEpA_yLuvvJeYY@p?fPohQjc%&8(6n~{*l`aqN@=ES_E3GogZ zqLMSG3MQa8X83s%H-(B&a{j`Iav*&~N?ULo&eNwum~i{wmcR5UA@Tsj!@#ka(s6h! zPf6Lq1<%bxJ%)tKMr$WQhNHSLsr!*)>_THC5w$({Sov@P%mB#TgMt*_JN`~n3Q29^ z0hGNfr7=$v6;UBVn^e$a;!%r23NWd@SWazF&c2ts^2^@2hIx5r+rK!GlTTEba2|s@ zm$hk=PQb{1ZK~a(_n-tci##FeqJ%t4W6=YUY7iK+%!#0w_q8;~Q=y=NUP#hJVoBlA z+5!YTg`!G^RHHVCdEcU>+p&5d$F@*L*{3MUQtEQ10-jFg@yJrXP_n>l&s(6)pc<#B zkNK_ryF@Hxpi5GDX76U`+&p8A*ZIAtQfCOopIj#9J6?7HDInY$9Azp%+c^ zx)0>aZ^ox4c}S8v$#u4Tzj!dG>?7phXvBv|3WY`HK&{qeByd4RFGG&qU|I_}rn6%N zFc)x;v~X5thL*fD!|YK4ELa|y2QGRsSa^5ZQqrJx{rpjyhcl;~@cRDrnGhzlY_$_= zLeEJ>{H7*)oZzNrxN6cDCP{R(=K~@@ks=;PNkuOOBWiC@N={3^pJb*^yX$sFf(6XF zK$*dqI}??Bpk})wGk+)($hFzmSS7uI+OBadZN`Dhi-JSa%(x^}C6u^`sbLwLDRPQ* z-*5pl8)D7s%qb^~l~iX!m~i_eb3D2%t(>O#ibzFnwrMj^kQ`3|3MmbgI`7MQ;zEV0 zvq@YFu(%vBK{3KL$%IZ$lA!Jns?wM`3zA~CqIW$f`Bf=n!D=%lA#O!c#_3o?>=sYU zqXw)e1yiXZj2{Xq6ZAWc^N?VqhhmPwU@BY_!B3bMK8sRQCgn;Z>v&iasngz=Hvb1F z^Tm@DChULy<>iULVmZ#HVRdI^8Pe^(l#;J-E@E9)0;5e%1@?L3IZ;`FDH9F#eIm_4 zvDkb8aipS<$|&oY(Bh-=3>4R!!KC((rpY5^f;6JuXI_#*#4%z~;mW{jZK@*1EAgu%pP6k$1GC8bE@5DuMna^N zAV+&UbRCC3f8Py1FKfO3dfEelj7yvf2!iz zK0yqnmH$|P@wpv{Jwg0IDlmyZtLo>80NlzAL&plp%1xtyW>%rS+>A+xq5W8bWo1saAr%P`t)* zFaqxb+>cO<=Q&O7AD^sN@>+!nw?8+>edf4fStuqQ%wt_JucM2l+P8`H@e3lx2JFP3 zER6sT0ai(=*hw4^CR#~^JJfXS$}F_W$#K~E|Fd_lF?Ll|9A9VOJ9qjRifPBh7!#yM z;s+nJ8hoL~7mCOSg3l<%2x63=F?<1|q&^VQ#0W|tDUT40Q4k{r#L|&qkU|TJ60o#D zTPEm?&@wa5lsnVuz1y|!*>~=F>@%ItoOA9z=lp-kKb=f7_nx!%I_vDc_TFp1=G8Pd zK2EdES^qk$>uMq!{kpk^8ii-GR6NHKd|($kDJuA8uK9l&%0<@#^?zy|TdIXGP$t+Y zON|#VwDi8QYJi4^hdG$6qt?+@Y&aDB8?6&g5(t#nTzHrxM~`ZqRF@<=>rwRdnYGRvpLrp5~$#& zv)HoT0n~Cy!OD&q6o8@bq4aCcCULtz*MbAWy z!3GVo+U<5Ar({2FBQz`LS%P&p>*U!3O?=iWQavu1H1)_*5ibysqUSDqy^WgP59s=( z&bDWA*>YxZ(hMeotUHn8M9(UEFqxm9=Y~7VZQnLs@km(kdaGDND;opd-RLgW-s0=% zIW*Dfzj%DoUT|i^v(!wDtpCtn*pWk9fi$&REvnb*v~JyC_~?jO{T2WJWOy{RcBu37 z{{8!Ca&mH!1Z7hB$Y_eLy+|(_6f>FMdfat+TiW70E6yQKqxB#`6{ z=53=>Wu|AK1B(TtgP!P1!g}92e#THJet1bGD`Rg4vh9tM`>GGrIKaM~mFtgC!jYfH z)wAC|GLpu{yq@(WYoO=>W%cUS+*RH{^KAmG0U0N1B6ROA5)Rm%@1K`n{Nh2f=jc;Q)wi3y#j(1Jos@J{Nd+j_3C7f3=o#YzX8 z7eKv6b8}6aoSMwSE4!c&sJ1AQmaDi$Xb%f*gYOdJWChdqvVZ*I^w|5A(M^_86u-Jb1#iXqcg0z_10VH zy6dm)ICWl(j*fP&Wy_ZDazI$ZS@DM7TMPKoUx$rNH-CFO(s7BHioB$qDu??q1fF z8Pc=^UGzv1Agu#iVwN_p4VPt1I&?VlUDh%PdI_2S@D5%tblfLdPpND)20M1xx%1&% z14)-%yLQsp*fRWLbXhC3~^9RSWCb4lm34i_ny2$G2X1=z3W%x?H~ACRZ_(BOWXA8P4_jLL5lmzi-j`Ur(dGc8&K~o8<$lt+ z>eCHRv(8~nq3r=E$8)9nc;ga5zd%UH0+0sXglpe={Ho6e_yg-=_yGVb=3?_a^$q2K z5u|L~kKhXcESn!wiscuJ983h6F8&pK0)S=p?wwd};Yu)(%(K)#*$LkOV9Chy@>y~u z@2DIyj2u^XHu-0s0st0gw|QP(cIASUV+MMWbtY>cd3TEJLHv2*N8YnS4$j^}!YBp09(9jdC z53%lm&w&zhn|WcrRJ03M1vU6A>(i`j;CrA%$nN>i82>IDs=$Vq%}w86eU>hjn}}atN`4jj|CQ;+R0$hP z5?DVaf0N+9;k&>g#*ANHMFmc@Rwo?cbAaPO{6$XG`zWETD1u3V z->}}m`Zk6>;26vpKM{ZVHOzrvVx!5rf^`b(CNmTQpo^?p{iX4bD~!JY1d~OPWzXl6 zpG>FAvzm}J3r^4F(GWvmBaj#46(;m;<&HqJJf1HmL~?k4|mByqUgslsauN8mXEi6WCT`7k5{n1Nd8RF z8(|J-ky^o0!1)#`;Q2HNCe=f7D{_3nCs{)9NSV z6%QuK4Q+;^H z%tCXlzp?&Ie*GEAgz)V;(U91vp9_ zCMd)kc?X53Gfy!A=2Y^l?w?xl03(4zN@@Os^>@<)*@LD>&jj>`Ve%KsNx`0ttk(wQ zLZFc|W273{0V7G0KE-;{1QUQ@0t&!d6I5gdX+uE6M)LEC3orl)8c94O;NaZ=0000N5y}kSHq1Y9R>r0D@pqpzz^0uRir8!hdkx zWE6E!@aF-_G93P&&_&^n8@zr5^MfTms{aRmN#!o5@2>4^?e1lM-wN^a^5U{}vUjsE zcd_DfzJD+6ml!pIFd&Mu(mLKRR;PR(ka-*k|6bTk`KmfGhsE~lzG<*g!u`wM$lSwj zU#|LPt~YN!k|yD0Dt^CWOT=~c%&$35aZeAA+hV#pvgGs6X^65ka!f7pLMb$Sf4uKV6zsnv|m`(#yN~iLYv=&haLV!@GE=T`+wL-7jOOaeGp-QX+329zv2EkKMAbC3Mrl8}W zxei>*?S8?u5{Z2ktdp8<4Yf>e`hU@IOz*P1{?REyK_u*P30tDOhSGLPrSablpb97~ zIJ4*t(%thQxvp`&e}48~7}9@>kuns8{S^L8cH(>niA_pk#SY5T#ph4LG{?f40 zVf;Eavf&XFP(Ss?`!t_HMAw)fBEm!S-|A>2P++`Utf-NEvUOl4QZiva zC$sXyW`oxAcdc4ygD9mdr=0H15)o5Hezg3Kee>Tbvp_1K>BxliyRpuY=HEiuvOS7s zru%z{Y-{Ia;^!vtmEmkrS(_UhN9fVqo4SJ!iHQHDw}G25twCq9Se})9b{Xa?@Plm> z{mHqmE+Lxas;Yg}>@v}m4LgCe`{c-A)#z&eD3Z8XVgRBYX7!^Z!Wk$AzQfMzTt>u0)#lK3BqTiMc;h{r6!xzVTW( zZ0}n_W8YQuzuLl867l-A8uB)MVB^C%nXN~U9{n$$3z@Al6P`D#phMr)-4A_#VefOg zxFI{uwa{ZpLR@U&4)ouhpUhx6@7j;xb0veVmF35T0VU)@m@BGSf7O;^}iRz z=m=ycZ$;M#xxafCGL5+yp|Rn3k3(aB&)En2Xa9{wEHxfBRtG0Pf)Y%-_GY?&HZ}f+ zbyTg@X`*vD)}j2r`X=C7Yh)C4ZU#YA4KyeRa~LM9EATAG@113uoWaWF&%Qwu{O=~% zn?-dLCYn)4=XK@MBC5n$Sct4uVRlBn{{?(T*mfMPKs;s!k?`sB? z`HDnmZl==88gf!xl09oU@Q{<_GbP_2v6&ZZ(*F|6oGmvGKj3RIC%+V5@S@M=Pu|XC zEkTinc{)~s@_(g(jhXgr7zILp^$MlM!=IdN#J0N1B0lI|O1|){#l)%JO7|tL+zYY) zN?284!d1QVP?YVrf0WtB@n6NW2^}V ze0+Sc5LBO-woLeA^m)x!y_D^VUjKGxo3mHZRIG2%taoY=(%Qd@_lWTt^^hvcwdcg~ zx3`!FgT2p!yNE^Ah5tn&N4IC<_avpo?CE+6T=R;FiAVODXvZppVQM~*3RE~maQm7|vD-5BQX6yf5C@~OfonZl+8KRg{flz{E?8=)mfe+5n>MHOR2Dj(8KG90Sm()Q z`(E7Ovx~@Ni|8D$xQD}-Bk$=IGPE>|{V&0;{csl%Kp4~q4;Zh1-d0{qA!9M8F zYtl$VXI*1s>xb)#N3JVv{aftm|6(wn<2}e(KTcXs;T5mtv+#~WSs(3KGnsuLADLx? zjl{B0$#Y8N#NWL`@UOuN%7ha#-SUe>uc9x~e?K_?@)5p*8MH12RPaS6!zVU$XTwIF zYWO-AHg8S0!X2@wW} zJSjpy-uDR)4^KE0@9%}~D9O(c|5)#e1l=+)q8xaZEl}Q%5T1ogCgQRD5kSW;eU5VCxnR|018I&@~?Z+)T+fnk@o9{1?#_#~qs>T@-xh#^z=t`o>P$ zRc>w}1ko?imClry(9zI%h4t^9cfa|hUkOt*<5LZCxt{0D%5T@`{>C(-{K5Wx8Fqo2men``c=13VrDZSHQBdu!m@ml-*H(w4d#lN2W*PKLu-mfKQAd1I&HELHCLHi+v z;X2V;l|e~K>LYxc`M*UEX}Rc0MrbOVoRoA?Ls*+f@jK0b@705yn@7nFOY?P#38wY- zY&JaRW$BZtF@UX>OlRL=R1kb;^yZ-Jx^B@3$up}H)F@tgX> zo2)D*CaLOi2a{O5$td2qp81GmrI3HSU`byP*m%f+UVS%dr4MmRalfQy^79?Mh#qgY6g%n;$zDJOL$WzpIkf&s0b`tUYq8HLAU!qo;P*cQbG zJ1!;seE-0B)w%ikdFyZoJpx1{Eh$OQG;j8PZDsi(PXVzov)1+g_cr`>jK$pi7Aozo zM#JyckB%NlGF~aP2y35>t>ORI8Eii;xu-Jt=6Jmvc7YJ+zA;k#p-n`{_)8<@7DvvN z@N~&Q2_n^0Li4DUoRP$HISI)dpQ_IN$5wKM!rIM|pzvfh>Lt2?lC_15K*8YR+?Na} zMVjKQD^}T6o12^d3^coRC|q*BPgp*p^H=IZu%|$aV+YJZaErXE3!yp;x~!M%)#U#h*eDYug`}|K0#4GAMJ@GT{$i z8Uk9S_G!NNs3X)Qtv7Ki(epk#NfvaI7RtCJyX1COs>cc?>xEbh49$Vk(hnQZ|KeAG z?%RC97O`Q{FXMf+9$m03)s)P`d~NNA*2^dOjV7i9&mxIyH#Y8f_%7fPQSd~|Ym=1I z4`#6$#9#mN_YNqC=a_S(y!uLHK-^qfsmsMg)Js{UI)As#g`G6(BA3xeauJtgBd(;m z&^Eb9iYKA594Mj&MsF62C)<>N>s`Eo7V;9W9HB-E<9)>mupbhmdUW|ZkM{TXP41WsoyX;U=95s{^j_pDu?&EiwrW&_6w%ij+n$JEc{hmiYU9IAtkPSjp& z6O~s(E4m%;8k*1YEq>W>IpnTPc<20mx$T&w&6#ivLJy6l9`A+Nr6fQ5 zi=&iE-cm1yBdNT)CFD{#hzsq-5i$evj`*EW-K;_>?Nk-A0Z{{Tx&0Ic+@g_XZA#M5 zno1Qsm;Vx#cxth}IDM>2e(oy~fYT{0HfFCD-cg}c^N`MKD8@nDqe*Ue_4VzZllk$G zAC^eOH~2UDT~&a`+gy!w5E4Y74BG}lY%9Hv@5;!83rR{!&WmRz)z|u-if%} z+J2`*w?isbD|P;Ut4nwbE@ zz~GyxhXdKSB5&NdF>U{*wsY#W`X!GhTUbVHgrc-7K5odoH!kujqCh}jD@8vNMaWnM z{fF$k`CH#@JGN`rdV{}psf5p1SKO`|%wq}V-~(MrtFI-z{A_1(%EG?9I+4YQ0$=C# zk@ROK^`0fXVnO(bSyXG1X@Fve&rhoCp`}kIWq0K|;nLqeoW`IO4y(s|I5t6M8uvP* z+`oILFXTvbYzP;%!o~RZd$wIZocI-0uFkZ-s)r6;xZ;{h8{da`EowKE^1HXxOW_YW z;|Eo$Vo;K9KZ(cdy_`cL*>+dQZ|ap8Sg~<(e)2t7dd;`!c|Fv{+sR~N+NtkoVv=sxBP^q1`1xFx4wPmRio7rk z6D_E`e8m6!oJ`?&b`r%rsCmHv zpGA(Q24l6T+dWOu%7klo71;l0ySLHC3a+laIyyR*T@(xl4uWY7sZt3#P=4~cS7J*) zeG0GPybP$0`L8h9Y`29#LQr7ihvHJ7;{Uvg2n@-9iJ5D$ZHb7rp*uY)oMv|aJNRW}BE53K}w+ta1o#A=&o58{t1cFs)VE@dz^>zl5Wu=J{wo=H2PNkDtMh< z&S}!{W#i;5h*Jsl&DZN8S?(6cO+vAWs%DJogBW`Lj_v~XW|D7y)WX3(8dtUD0nB4 zYrY@h$2^w0el(IB_&Jc0eO53_nInJFEZG35dY-M|dh=Nn$AhO(f zMxuN3fv2r4I~Kx29nFBk-h1%miQhE}q9{7i(F{nW;|}hCf%4RIkhN^l3hlQICd}LW zD78Ii=0-u2%Z#^{au16OVv88@tQUWSOaMWkvPhY&=@2E5bsWrkaruROcEQ7s&xzGU z+M^}`ncea9^b}x}rO&i$G;wC>+f)oR7IgSxMvT{wAm2Yc-5p=sNu_aa7;xsV-uglH zs;-V&;$+XlQB(8Te%S~59^d%{J)Nrj^O)evJwf#wn7F0nVNmGs)Dm+kaVR>ZXWnkGaQuebPabJ>A(qs1v&K#KUL|Yp zC5sO+1CEo`BhJm^4vz`RSiGI<*96zgt$GS?MN?m>@OWub8fJRDtk-n)%{!(ytF<4# zKU9}I)~NuQY_pwI^2PR`GPgkPk{IT|w5}-JM9nH3X=`Ak8(9hqCae^%K%-|s-8o~? z;2%}CoTr10i;cx6+nF8=-T$nE?d7%Z8#3|Bwv`5J`>g5SSzUp#J@ePQY5Z@xoU@4T zwYJ~p?&a$fvpATXInm80M61f;!`XxJdl^(-YZJMm_kSpSWxYlFp*3GKJJh`Wu@8N~-Yh;%7DhBF zVCd-1G-nryts)YLy!aBes6Ya6UBwg&A5t5yfPjFw;2Fv? z-MG=J0@aBf@@jof4*>xIgS{EMJIlL=y-5EL{3$5hYwjX);YO7TNXwkN$Q0P( z2@>@6s}Zk|P)&7C&ilrIW8e1aM!#)>v$Qt+$rQL))lO60{`>PuRHB}%JW-UqZGIo< zyu7(;e?0%W$_Q1i>}2oH_td&Nk-5!%1qL)ocU>J9O=&5`yj{!&a}=8*x$5%$SN)+4 zS0$56r7`YQN@KS^a>AEUpaXGK=Hun5(ptS)cH9sb9LDz~t%+W}4>-0X&uw8N!#@@l zglruX+&v`cYEg_HxDRm;3aVjIK)h@>MdnaK>N5?`ty$nP7nrY+UpjNrl zl*sot#`W&Z%*?8GJ$&wq8s=Z#nm5^(Hjav0evFkr{dvu>+NNo5j>zBEw!jH0HUH?n z2sHNb;qJYuI*-3s+YJI!*`sU$y|FtBe?pt zR=e4LjOz{i`fkfZoQ}+!VZNn}I|h9ppIyFS-vzWF<@>Ejh z8cz;ZR^*@3PYsF)K>9kO8D*!RT{Rhr;?3Bi%}jq^!t2VzaQsHll@{K|6wo{)gL;-x ziP9p7W;uPymDJNlNffvyS5Ray(QIvdNfw8#y3u}s2BUq?dDbU1xu~S(#l_#wIXxI_ z#snOyvyk0mVk#zb3JP7%<>7)_XG$L^-o*8uYlF^{)zL&&HQxsS0JtvpXEv>rHcyLN ze@gDRS4!l(^ZZX~;IZ;d{KE5|-rft=y{Wg|dIiREwwdW*;ZK;aS@zg%#dK35N1Vrj z+I-`x%!t(OI)Xv5KDT%+B3aUeKg!ouQ)vU)dUs>xgI`+-ovjr-C#&qAku zxZ!MA`<3m)DClKC{uK>FcR!1+FN)EV*6&XHk!@u9tyPT?PgXk)IZrp-8H#6rqG#la zM@`toJ1t;8nmk&3CocBoU@YU!P`~xY8ta3tB}xUT3fMsuR+%^Laz50^;~S{q z25l}GOfcfrQ=uD!`iGhWPbZ*d3ohKafl9|6k-XB=bhj!nM)>aYh*VsJ89Nf0NmQ^V*w23UYD{P4a7K|Ez(z z!P~Eg*hxQ8Rtw?gi22jT|>4L{5t=ZypPQJUwEzPgfdp8tz0 zV?Tp!a&*MQ;fWpZJpNaq|+Y=S{`UEFD zvSD?r4+-e=n+|`~e;2X8#?GF3Rqsuu!%UO-^7TpXzxJL^<454gqc7t)_3+jxxVt6Su2}lrns3$B>S8RoL>UlluUxL6vVtyQ#`{$Js zNqd^*&g!^X#kgH=+4pnzE`_3Ta`__{_F1+qb+cFA)PDx;xhUhoM%Qxvueu(urQD=W zv7zgf#6127?p_U8{76fG=FN^LWUO_))^5XnXD>;-h{9ula~r<)vQ(jPLm0SdZoD-I zZht0W@3k|plDLeOC@s!a2%;DtuCA^MR>W$&>BYma4(^5U3VNCFfmxR;KSEw+WUMoX z>YSJPVS1k=m%nHJe!&LOuudGS`p)Fq&T?B@TjAk)^UUi3+0s5rK?i*^D2l4|PJH&G zoHGJrMr0ng3s5xj467Y%dv#@DIXY@~7n9yRtkk9cl|`-gnveelSK(;|TSNjh(T)_i zjJ|u1KRya2M(m(_3{@-9w^eq5b=G2C;X}xH?$Q~V@Ee*bcY(G;876Gy%{tfonAh4> zn3f7oMrW)hKSe%Q4svd zj~_qgCs;)i<_@=QK9C7E1fiIkV?DpP_>%+CJRO2mRaL!TfBNPNrLc=xMaUvxfk>tC zHMr@5$F;D1@{6Rnhup8T<4vFa^hv)8ybpZY`?>}G#;R`k`5zVa_Ae{J%RE4=aCBnK zG)g_;1C5gt*nH~Kbh z4}JCY^ln+L9|uUrCnN}n4h77;CoCLJs$<5hjoiGh#Zn2 zud?HA%-hF8#yE@d zF11DmO16TsNzZv?6iRxni_~QI2MKQF_n_IjwneQB`Lz$^r%h9tR*P_+s<%-c5lG($>Gc?heWmWOw*F@{#mGy zM4@>7j(5d1uXCC|U3bUD|bD+@I9pZUNY}HHD&`Kg$>Gv);JN z49w40 zonEPlrgSB+1n#E2HkWTL58s_W_CMTtR%JhCO)Kijxx6_BWL;7=!yV*kyYDDQ5e;d* zNL{`Xc@3q>PLiYl1@l9AixF@vJw_iI-R9 z7YjUjAa8%Hme$Jx9!eKwq(O-iig$Q;*zy87d4ASw%bWM_-+!PqCERqhq3kqLcspgi z`7|Iq@TQ_7q5t7pO|M~%Q*@m(^f}^75@N=O!I-YfV#{b=L>CsviIjZ#KC$F}KNQ_D z>VbT_AXj!>Gm@aa!$2W_XlbHXM&?GF{-Mgj4cfDfGKPr;P_qke}1f` z`~CU@a){@ll`JZACj$K(3sL1F(P0FwpUQ%edqKbUKJ$&hlZJDa;wj8b7RA+7N*vzl z1nfa)o3&bySA*6SRYf9;E^C+~o$B9su=m^j z*C*-SwaI(J0Y}1PC5A$$fSQ;-^Q)zcDNZ@nRKw{@v^>8MsuA3{(S7v*sI7hA>EX7w`g<^>PA*(SdMGk#U2!YZ09!-Pv1@8-9VJ zIYqzns=aE3!9b+q`EI7=6Y7w0%A}s$1iQ#APufs)DFdp?&f!9ZN>OQ( z(#NxunLhumfm@Y@g=gsfw=`#Wn@|0j zkGS<&E(PqFj~3o8dMCQxp#2WkehjCgZQve(mQmdRic96WK3#IU(Zeke(0u7+=JYV| zR~Wr7o1>i$AtNy#Es;ze;Ac(ySBOl>3%S1Y1`ZCA=w zeFf{!5;R1=-s9bw45xb)mHY0C`H`!1K)?Aqg1&Zni@fV@gfbpJA0I!1Prd9dQxo19anV_*lo=|PpLZM-)c?^uX;*c3i5cUf zwlevn((%SF_K?ZozEsnzHERvF^uJR%m?XEP6>Uo1VNkVgD(10F?z7n|r0U$XpErB7 zzo{B<^rI^dM>|iY;H8AWuSg8r59_}4myK3EiOF7UY;35no|BFVh}6l+Zu33y6L&+=!m(T31(>?e4v8ROVw4uj0I>eLi5Q^D*=?VNIRS+ed$%1{V8$dq^;N zH$w*1^>}#q(GHH(wJW>|F#x`l*n5$AHD)HpoKCx{>Vt$y7|;RdGzDE7+s_?w_jXo& zK+aX-EM~CGlu=#rQ!j=|doZxt!=vgMI?G-DGm6nk8vWJo@ORVx&$NR+$_)mW|=(9(+`(rTnb6Q>8N zZObdf#q&w|IJ^V0QD+(N^AC1cfG5he=w8?Ti0ESB0bha&rEHky@|R~3JrruM<1C_5 zaz%J=UgY)xwGv!f#H2Ye5n4S%LuK_02~`q&oSu&#gQB8{XR{)?&f}BP)4l_o zX;6iuR9<|iqO;s(wvA$!KR3Q@=JdpCW^o|f;?_7S)3YEaCo}-I`xx?e(ZWMeONJdU ztyi~b=eQ&S8flPQLBx}s&&E^;+T$y0C@QLq8tkL^qG}N7nI$sRrvpW7t==r*hS*Rg z^;lU(b@_3#GzO(F_!;-Wa=MsxDx@m9umr!^Lu6wO^}k zOG0|xellGxt91u`tX-~rBO)Tg2K;QSpusvehR|X7gaM2NpW~no+HrhR(k%faB&g65 z?+<3BcrY&$!pmYxXlPL}{=BKB^w|U(KH{WFep_#DAn^zoN~+MzgQ5H2bKeI);tX_y z-z3!hAKkVVH5+@a`TRlAx|S+jdw;&WrE1K?FeQIY4wYL>XBH?1+5S|hHCh5@YV z@<`Qf?F}tybf|$6JHfMicdf(P8CmKBE8=b6c!CpHK;`c(AB+S&(JJ&3ruEokbf&u% zL(2<|vnQSxA0Sh|`>(HeFa>py6mrs#T54t6_ubxH@Qr-noq^?;;WxTOuAZx{%>CaK_a6pO4-&=+agfmlLCZsyO{yEQ+JvJEy{J z;j^o{__oeys(}6Z`x@HU@4iO~y8To!hq<4vICY6kh#5{uD}r%+?(O}att#2}J10|6 z16oq=Xo+DoW@G7MKG!o3Q-Hno1unew(%oGiQLZBJy^!FBOKDN!EBE%<)gA3clo2K7={FyIZkmfgU9+Gu*Zd zcEX;2Ns1FBqDd~U&pjmu?)KLHLPno2_z!ai!$Sqytlq0-AFy%oiGOdcjB(FteFg2K z5*8g@mY=WSceMBCUBK!4CuHM;xynV?`S@IU31l4K1LrH9gvFb0{_LH|2vk$KB@px7 z4E!0433j-%Hu)M5nDOz2VS-B!`o#gw-B#z0ue;06Rjm9RL3FC+#fJM^$PskTV03(f zvVyVnkH%LxzKu;B7pVTYhe*5A#`}WXbwm-F>pR`x!|sn285xC)+57-osl*6G4AAs0;_ z3}Ce47sQV=QVqcZ3BX)DjJ8-*IOm)^0I?9X9n5K29x1AN7__m!RcSjEDtWr!w}6G5 zY?lTuC0#SL-ti4t__Gun%@Qh`aQ@0OQpAo-yn1rh#RjW|Jp1-F9K_4hvl^h_4R$|M zBd&`dD0@Q5RUMp&c*0}iv`B4{-(DcHaoj1nN6l2##fvRf>}z`oc~>TwE9Y%czhrw9LoGG&iGat%R zrGCnKlGtlp?`Z|isfk@X&l`#enfKmoSOo|_AF+^R6&B3BzGH)xrCWqoKV{+vNZAiT z=zV@XD+WxTCEl~oH8zy-qUDaL^Y&8&z*JbP3!jSB9775DWP3*^wiys$|UcIk; zDYsH^MA|OOoCD^2QVPqTyN61l!!rn1)h{Zjlv!$it;=UrOG)p{IFiPIfz1J7cDSQ*ZqbJ&%DR3rLtAsk?Pt z(0dEw`Z^q<7`U?At)F_hn;jnf>A0v0IzZKF-O8IyV63iz$1A2QPj7*MXJIy8Zk2!_ zm{^$aZ_HB#Gqk)lZ+{PdZr(4U=6%({dU5_o@ns1~7b-*94%HEMZ>QXH5{?ADP7dXB zIdu#1>pi(ShAgs)!FRbAty>uCo`8%@7#xYy<^zKhg z#3rRl8cLaP%jhe{iDe#p^y`DUkdXc*n~h*>}s#0SJkN7jCJhT<_p}u;5EKDWj#PrIKF1jN-~)LNhn3`FcMR(7|{F^EtTMj6gP< z{x)r}*Ic7t(uUCpyS&2KR({Tn+qwX*{8QbZ1VRSoVJ5Y5K0~mMfZ-RWYfzH1C&%1Y z=9FI4~v(zheO;*}8M^C9hTe1~!9sCgT z;ikwYzk)U;O}EOJ`BPgt@R}|^#^gQ2`Y@(;=~vm^c7*o$o!gUmE?_fl#u8&4ofDk_ zxUI zs+4O$Ic2G`p}B4u9vc_;Jx#*Dt{$S&D}oN(o$`8bS7n`)m^k?7`v(H~D9YJ=xCAB3 z<;`jkFs!UT|FhgniJHYjS;SKhXA zNHXy)=$pT9)te^t)UeK-eQmfPPLV`<>t2R`Cx#a%0`9EICv|mnyoRA)c6N4)fTfqG z0}dUJKfe^+hSnHagvTP~f;AO>?s6#5+6ligV=o`By2UIC|GvYmrS>rIBYKfJ8A2A$6Z=6=N)-=oTURApK#md`W9Yw>-=w)rb)wIg+SJY-YYBu3Pl+bPDh z(K8Rxa~4sUL4zG?l_40Wkg)HFxKLTP{t*n<@Ab2fX$sReexwOG3rfOKYZ7U2HT;%I z=DVk62AiC=wyZg4Cm?yUm;Dp)b|q|MXd&H?i^pfTP1LKUm$o=!@=a*L{ z_P&KA#LivifC=laBbeE^>p4CveD+{xt%vK~H*7{UDt2?BZwS2TXOBf}-PfmI`kx-} zx+aBcD9wT~NC7%d!&9-1aDv}gZaBhsVJ0AHh|>ufC8W_hX%Fs$vb&-9!H!c}^%kXU z4G2l0=;!R~cqwNE;r0){NLy=L&@)+To=-G9`rX`;=zPxf&G8DN1uxM*>;31|H;gE_2v3A(X8ffQ)eX*i};YcN0W0;63Tt01Q z_qfpRI{gHo54J+e6ld(q-0Pi%gXSL0p>b;r`YH|EZ(Z__2aV`R{!kVS<=Tbx zO=ED}T&=Rf1-D8EfN%mfhTT>ZJm~c9cxSQ?9w0$gC58}?rovI#i@DC>78})GcLCe1 zu*qv5^6>^xGY^1x`Ti3ak8IMZzQhTDBngX%iSH8gF<@_%Fw#pg^_DmcUqk~i@DBFT z2N|v7cl*3tT+Bi=ULTh2;H$!?>%0>izA&qxscbmy9wsnAXjU_M$K2dJfdO^S{~{q{ zVsLOUhd@RyHlmO%J|))YmxC9s*$QKn>g*$+By%&qpQ67$1P0!tQaHAWq1Wm{-jbT+ z=8(7sksWubc<;)R@VgZ3z@^|T(Jzy^s16fqEv_$??N$aUZ$KCL1(iJYQsB`uT|K=a zV1D74tBPp?r_4!4S<5(g&tM4NqWTMAENHKfV4fgPPUD^Dnn8#bA`YN%HZzRAkY2z} zC;RX%0ljyNJrCuVnUftMHxPBQ=V|1#&w(ksc8SkAYWG+Dx?1EIA>N~I+FY$$9p6%T zDPG0H<26@>LBb-JIu!L-*2#X1`iW0G5&WZSm~dp-}ioN$ACdIYjRm8~5JU z)m?_Bzx>!!x34bnG@!>h@X*jWf29+2r$G9_Zvx2z_PW&%cryoVLZS7JeFajedd3Aa z@^t{}l3kR5=j7e87MQVk0AIH`?fQYPkn8)+XLDlH`?=I}m0vE_8*ydx?=c&-*zoK2 zW=NWD&rG=oOS=dE5$jXTrJFVFtR%_-6SXmVFgVncN+s~<&xgbhmjaE(_QX$LgTTmR z-iBKSy(4RE<TCbLF>k2X23|*duEuI=YD@$dpMis& zn&J2B27sP`$;A~(r?0}EP-FaH%H=i@k^_Sk8U_Z1BX(6rFgdV-PeMIBT^?x(MfDE2 zEVM`q(AmXHFnhHu8J8yLIQThDhpwNkq!Xzj~1_Gi2Hf&BwjOQ3ce5b8323} z&>TN7!k_6L?(W1a_;6Ugq!DS_XB580=s0Z%76%d(g?Tgtt5ZKA)!wEx$uZoMKWLACafaPo{2je6rW%33ePK*sYI2rqbVp;~&E0jW-y9gY2+p*D08hy&TF^JJC5QdwfHnfLBX~4Yibh4uwNg;Xwb(CdPe7BvDsarOLYPm9RhpL zVCx8qw}sG@d)(fHeW@#KYP)8}W6W3v&U%Q+OfeoR$q*T<8o-y$Dj4Z<1Fc{LEEQW% z)jhy9(S2f@DqpeOKNe`ehDSY2anKDa&K2VifH16;Ul@jFgc0ZPkRU^+TDf={UY zH-B=6c{out>iwBKc8gC}P{AxCKr-Iuc^HPd#U zH%O#@YaN5evG}#tjSmcuTNGSIlpjBRauCAW!bl&`LwNBJN**)B$ag8%_SqwRM8ja^ zI~*d}M33e#96kaDgoj)|>;20H^CazE*oVC-UW`m`{1f`qP>9u(mdt z!m*WBXg^LMd2`rY#saj~SO9}x(yOT&i*S$pS*f0WXwOs6cteJp*(z5Z#rrD0t{H$v ze1Mv$`yx6aLc(*Eavb!CH??t6>MunuY^N}VO6ltAx}~433f5@k!~J0R;lLph`8Hrz zMZdJnq0l}kymXHV3FjoK<6;|;;_Ji)5y)<~?NQ;(Nj2ziSHf>!?xoZw&N21hzI~i= zvc=;dA%u6)fbbj+b@NH#L&rb84rM89$)&j!i$<>`ehFxWXB* zl1IM+4~8^8=9=p4wc(eMmX^k}4o4Q47VfvJqK+Ne?FKW;3qcc;=bjsj_AFow4Wca0 zRj@^4F4)yy++ez=9oGuOgvaoo`(U_kP%el4Zx&z;05ctQU8G#Hr``(o8&pfst}( zpa+DEv-dvYBJQx;Gb+hEnE|g)+laU=F#-vzdioWNoJuS4KtB)$7(ZKHe*)6A&UTVT zTPWdNC?VMc=qH562P0DM@<9qUG=k34#Z%>0MDRe`Mf|h0$-|D0T$}sjdK1LZ_t|UL7xHKfO!;&fea>D@QRd^z>*!vJ-SoHxuXIRz$ei{lITA z;#CI_>L>t+&tS^o$?8;n(qz3?DHk7K97Y3GwaR@8!pdZ&t@fv}Wy~b^kh7g*S4&3NseN-L)@a&x`y9A^)D$R7Mxiy3!a(sL|6=*^m zKu7GGkBR@BfW$woze{@h^toL2gj6)Mj{S(*M(k)`{_0zsW9o>I#Z-q%HlW! zcp@JIRNm_NsSF*d=D#O!mz?=opGMwl`4mKp{DX(r?Muw>nfF5(`N|~SB zSa8*e6Q0xJB9(~QHy;xi_F=u9uxgth4J^SVZ9=br&N?gZBRs7M2qA)(;@LBnP<_yq zJz!em6wuG8<5Zo%oI@QLG>P%?l$e3Z<1b|v{&6g-_f4Ti6dq{xz!MwiPV|;DL%yl0 zEp6smMcG1<$-N$08nEG?hI}ayT!GmRke+9PuyBsr!$^fzCu`Jq!5+0&1L%w(1|<|u z?`H8&A6+zILlBTqKJWp0UIRso>FiJI3M!|DKk}{^Cku*4Be<9`6DiFQ>^jVFSq!~6 z^`FD1mn7A`I(bF*Tp0cd6J2D&G4^jqzpi<#*Kpn^JguJKSS*)zAw?;%=3x?Z?5EGcncxc8`qEjjlYB35Iw4I@= z*JO1zXFBmgo1q82%@6AKV{lOAn_+sV7k~^ah*?=pbQ6$@bIseMhL7Ot*%0I77iR*&=r!LVtQNTvSkim8-G~9_lrVMe~6;=%<1 z->xYbh5|AvD{-)RxunZ-5E|(;=7R?Amc3nFbr^6EWVM8UAw;`gjP0d}M+DJYnD6ed zYf{vB{fq#&HcL#KO<312HiFr*29?w)UDV4dn-za$wB$Psw||FGdrw)qtMwi$ouKbz zMzj=`T;=4<31(QtdX@Po7(tjA-?zKIxMfg5ChhCcfaNaC$Cwkrn>c9kd}xUuD~R_< zP)0#2d%z5N$s}CHBL^}%D^X4>a1Z;^#ym=4_F4d9U!d+Bfzq9>7{`*U8tt&xZEIu3 zK*)FnN?>xY^-P1$rp~FfxX(sT3@?lO+Jpeprt6Q7=hlD|MO`%U&UAr6!&(yjkauQaJ10Dv+A1P%~eGT)W=@L7LCeM%RvcY^3n04j3BAvnAB)%$z2?dFM44oHd> za6~;qRWU>WL8i zPJN5*y#Sf;*C44h0ml2#K*o551LYWuSa9y#P1e0K;(t%!!>4e^HZIzNN7C+R6t6y6 z!LhBGB!QtJlemn9=gM5>@U2@Sa=frFZ(%2Nd!5q_}uR&9cpzmg%8w^J-$%UOg= zHva0>tJz|G_b?+?yc$LD*JVKN;*&m!!(!L9i7IxI970CSRCpa|3)64#OK62%7`}r=&99@U z7c%8^@fS$*m5wVl=I!`Sz>QN*cLPsf@7;l6$Jn$q8lJS?yJ2VPWMxpI@Fa`Z&gz|B zDA~5f(&B#`@6uqd?eRlpasP5BQ+;g2t9@2I?{H42-)0<|XVz)zMKLN%SS{ALp*GGQ< zm6irW!>f}Lqvf+*;9;5wkT|j~J@}2a|1E@6OQIzU07fyKncbTnXOgcN%?k+v!Cwkv zYya%l?Z zd`_?WIfbk;D+PruqouHObHK4(&J}VcUgE6(Yw0`Ssc!rKzl2U!M@C0O4rOO#R)}NI z+uq6!X^4iVbc~F$3k^GaWtSo$Bqh5|NYYj`|L^bf``@qE^Lp;*e!4m5`@OEu_4&Nl zW#785Ar5Mh6Mb-I;JxqP7yQY}*?9$&9~Fnu`Hz!p!H|3wO5TPp@kzWNx6Z?0Y=c8# zrBQzG$)E7XVrafayh_xmOo(28NBmyWasg{IwCsJaFd*~ z$WrV3gQwuiu;ta$w`O8zZ_HOX$XSg8IX3aqGIvJ2h?WbMhrgrlH*64Zr-ps&!3Dx@ zGm25LR%6Bf2YA49IR5Vfin+|U&BUMc#uzurc>DkzRlHus-)h`%+SWqm^%K)yAEG#N znPAUFIxi`(IEvVw@)A+FI&;qMoiJY>$_4 z@Xor!KS-WW(d%AJ=Dz3689{2Uf5a|w$v0-V&94U^dQ5R0r3xI!HfNcJ$D{Mbkj73KZ3e$BWv5vfxUOH9Cl20LaO>}%``z7VfSlh2jPweV?B-+p zDo40*DDbd9i6YTqi~=)fREiHh=v+m=ErB6-yV2aB4~)&+?ABQI9mC`ya`FK|WcL+qbTG!n zTl^S{_-_09;`pDx8o#LREWBTZhiAf$T1B-_ANW50 z@=wLbUnjFB7M_)HrIZiKuh<(DB-e5-M5id`Fm>R`lPQTOyIR7rDyD&$jY1*0s^-QG zy0mn0=SLn+PO+K(Yn5N+vu46HFvW(TOSk|@VUJNf&GRz$Ne)h0OZwf7FK-AF4;*%^ zC+@GHt1it#O?-}OgHxJC&LY0^(xpqocK-fo6|SN(pH%HA8w%WG|9EzG_OksOg1`b? zvmcTfWlhaz#bUgBykT(i#klYtX5P)VpHKJ)pkGwrU}xWYB4nTK!Pj@$J8qx+rAV}* z%efEU;8o3p(#DNU{AKs7r44MXF5sty_hV0_L#o%GVKUh`2%ljep6$#VE9iqZCpTME z8>&6cv{EfEW%X53Qj$L|ziqpn{&oVNFJ*(~(W>A3m5zS$_>7rVVQC^xAyuwd26(dT zN%Ol{plbPjSRg)t{JviiS5P1K0CXbc0CS4fVf ziHWsK!#E3JYHIoly-)HcNy!iq5fNViS)s=sL&&u~7+gdLn#pg^ZiOl1OE)=;Mc{oM z+)lT*vTAiMSHWu%($ds)t3Ysz7go0GFWTkgd-wFsDM@C0L5Bi3SqVQl8G3trpC$f2 z9pKY&sC>2TEn&RB@e33>8{LO3ZgW*m6@{)GFcI`>aq%dxx7%I{H={^p+qUu9kwcDA zvHr_n;nWZ6n4iN(_r?rs928V0zinIQudn?(djDBMU2+zzadKuRC#s+S{X}|acsM~9 z71hCTSNRLLWoke`K+WKwU17!7%T!jVvdsenOSWvj>;J0gbD#PZByZMKuH2N*8kW{ZG~7NaYsy_?)NWi~LFabrEIf9vtLrPx zo1bxCYRoWm{pr)EMS?CLM^PZ>gd)rd$0p86_Qs#7w`99+fliQd1wsPsBU!SN?u+GK)eSV%56i@U2%I<3m%#q?wkMpO5m6C4L_dg zu$>Z1m**ezO>+)6bClcLm_~92K0NPWD(1W^?$Yp?lROEkIc&sMhVe?F$1mOt zpO0a0RECtN0CV$rv-gV4TS7-;&ys$rB|yznn*GY8F)kS8`W{Bk8;HPp0WGB@EAi>; z*Rw@MQZUG~ru)7Olk9|RFlu&%(j>^W3`rKSx$_9pp z@5t@jQ1p9YWY^f6H$Gud<;W>3^E>{@(AL(bu39BVo}MWBMN8NO`@hc@LzXKVuqC%H z&{~#B74Z(bN6T_3cNq$b_=$P+8T1G(2;SIx>NbiVRY<7w*Ww}tju;IkGcCdI+g zuW<^iS2|Va<~N;h5Nw4`;Pb76Q=B9jx-i!T*W&v5(ds;dl7Y{kKTiW#zB%yKSrTJz z4@u#RD=4hI+um-%U@)Aq@&nl8Z5}osnoI2VNj_7M>^gIdQ|K4-oUii}e2%qsS50}& z?L2&HD@(YStDg17wvX%RZ$yW$howry)JED`>RmoM0LVZ+6LKj$jB`=6e+L>Wl@^y$EpOG@6~ zupi5;BR`Jah_1e+ zg%IxwE^K<hD>TWGI~n=irqZ9T~a!V__i@>q*+w^!<~)EG#S=)zw#L^!fS2Rkcw< zA{~9pZ~U{I?US}Pha5JrIhmtN|e*Xt`>mWg3%?gqu zQf~9@=!u`kcQel3VTP3N(PYF(GA}81TGZc=oR@JZ9~_P(gy`R#Fmu zZtTW&4N$%{5?Z3mJ%4<7zSqm!+ja2Z^m%(^T9E3|ew5xSH~DDNm7xNa)wY3g9_!Mf z)*3l8%Io*vG?qEBl_HhE);8{r;rVxJ{0!6lu7DV7<;r7>828U1z|D*d^B!}RIDQR_ z@-)o~S)DrY4Nu|K@5RXrZZ0mFxo9ybAg-&kVLQLt?nh*Xrj}OkyTCybl@=iP9=T&( z%KwZH;pH-;qqD@t#TRgDyn%$ky8|7MCMOd54mtMM$Z3MZ>?|<5SRYF*8@Jfk)j%z- zzPB+pcdNX7#bd6WCr;e@f!F_jOoY9_W(B$r88Q-~_sE`Iw~i*kZS%2(l+6nXw|;q~ zxAI(^&}Nyrrwx^^OH?TB1weN>`0n1_j^6#8OS!zYOj@Yyn1xm8kW(hj$<_O?z5b$^ zhWgVz>~x>r4^|su+iEJ*uZP!rLHv8nI*n7~h0QKY1M^&|bk#N;VN)(WgIfL?zrHp1 zzMiNHFa$}M>lApQ>K8u56w3J*yy3LcZ7h{)0Pz`7D>pop+_Wk9(AY{^6pMm_Lebo3 z;ixkZX0O?K`p<&Hm3#RPrNYd_L!#j=wtZ|>cA5`x&iNI0SMR(9L`NIsgnPlYA7JM$ zk!FQcm^gY$w!W^s{jPpaEn*8_Q4>y$GJ(bT3=!-vp-UTbSfMRs$rY8pu zl@~aOkSy~8&BrOsh#lo8%~PknLDUx2)~2!(89u>PN6PtGS_6mtzh}uwi&U;xd3ef` zi1W=YusLPA=Zo=(fTs~fiH)5-V3xLjh8gQG=?hgML=eX83E%AIu^$i!r(|Gbv&vCP zyF3SIxGPy|56uo3eERWY^|EElA_0&b0Bz)jNXAh#O=E6VPaF2Hti1KbbPSq-b-8yl zFI~FN86jGFJ)0x(U|E4oo%`xcyto=nxBmoiJk==gewYpQs9tDj=q$>v1zMzi0AfXi z2%Q|OR!|uoC=BmFL42V0+nOb%rx$KJ{Z~`Apga5L;;%V|L`oak!i(%Ub?fGs20=W6 z(r-wDPCtX1dxq5TQ)#rfLhAk>m@eghw5TMLjGuC&3Nq}3A|K-c@cHRgWv5yvjvtV6II&bDPF_NybM|^)vocERR{TtnfpkX$3Na`vG6R{-G&)T| zv`Eh|lqJlR402i~ACTtQAwf7hv{fM5ODB$B^$#7p?Rer_;x!%mVFU{_^Ph>1XgMa?LOdfdJVfSB}3o3c28fRW&Xwlwep3pYjroB zu0>YpRF?!Zv1WACsUgtgumSpOmWj#9y;J6_Djoxm$qp|yRlva1loOH^)tmRk+7%Tc zqx*yUF+a}v5@85lnAJoehD#IK;LVK+|D^BA_7TsI574sktX9}r>rh;7zy3I1{=-8z zL@AmJcZRNEP24AOeX<}4=T!&);HP;zK`p6>HlRlPU?J7a&!xvQ_&CnlqF-uB%OD>uUI6A{d;YKrmWxa_T!1C;l-!Szi7$od-!nQ zx5@6QgE0vSQ_$l(kArS%D8B{uT(6|O{Q23(dCP;wY79n79TlQm&J@@g*7&D#?Jl}z z&~8pqk`>_miuX2~DAsyT_A%HWx`YJB*#hSqViAK4`Xk3zpNaKfom<$6b&dYczrcEI zL2UEp1X3pR1%yBi6j_e0`DY^{?!i5=+Q`sw*eu3p_wGCroa)1QF0+(y{M%bywY}qZ z0#xd&)VqaztJSPm4f>_~_9d*he05Kh&_PbvgP~GKUl`DL?|X6U+?}QAwNM#txA?LQ z@YXW2$wBR~IiSZCop$S%=GfEycjnWsUgd>klut;`=~a5V&~DE^KNs%%Y_WH9eLp`W zD!LFT%6#zn@ncK+tIVS6D*I2LKhyEYi`N+RLPYwmbI(mPhCyu{y+bfeS3G%pQ1F*G zCS!ezlyma$(#(}p?`vv1dLlM4DtEe; z4Lo8Mkvpxft)by`;PHpO5|WZ#bD;8AuFhB^IwmzWH5(Io>Xj=z3mNqRD`6lJ#P}O{ zK0ZFQJ81sr0dT5n0Mk8K2n2Klu2BsX=!-f`E!(r4&P*HT;dtk{UYz;KA8;)d0w$g>Ixv%(8`?6J#{>;Po)QC1s_%E z!vK{g)H?AT>q`RRuWn8-Y^@$!B><7U=f?Z`lathBTtIs5Bii<6NW^9il~z|WsQ zo9}*g;9tb)DMF!BczA{wJZV#)gbxEE;3mk*Yk~psd9!C-UD9E`UIXRMOJ}|~iyyXr zaSsc!$?8rd?`1oER2JbV-u0YB_G$y`hNGV+hP5}ED(e{3a@8P!06rA`^5=|zK${@sT(7K2hzd8b!t?{oW&31pDKxAUXbFu?Fr~f8$jFFADVoOobU$d{ zvj_a=Cq>1?BJ}n3DQ#_SKOa4^y7-a#AI{Vh01Y+JZ-7xOrqO8rH^$Tz6#BjKy<GR@?bdGC&@axEN#Hc1gaI+A2ZQNyykQBGZ%O$FOc8J+t#MQR zFi;RTim~?~Ub5*0_iNQ}9>%}15FlEse0znEx>ZKY+uh&fEMK;Qj$k8yl{I{FJaL-D zr-!MmtJbXx;^yU5v%I8uCPT?PitP2@Lx*v|&rjkuKF}reL%ZV0eMwHtK+k0smzGwb z`Hn)eLQ{Tgm4UT2-@t4r_^nsaodtP#c+^#P_w?8xoFPMyjRo;7GT#9WO*7Q_As)~% z%u5n9NHU;QGpRz}&mO6!EbndFYg24@YrE63{NU%7lG!W-G$co5vfVeJq+Z5xkq6>% zx-l^^j|}zok-!fLN=jdU{|tb_ z$U2wJ6Ja7puNrCgi)WY{YHDi6m==g(_`CPyi7kbjTjfla-})nS!wDaUN>@W;v|O&& zL95{rC>47$9K{E;xBmWHpQ9jwX4{KL=0E7b@3Eld+V41fa%rn;|6g&I{47rfMIly9 zLD2A7lmu<*dC6$VlmCKG`FrPNRiSx&@b#9YQ^{pPzJjQB7lnLbVc`dOFDKCt%bffT zWI(O6qeHdGL{?tj5+>@3K&2z-cjU9Qj;k&BJLr}>h(~#ja-IqhdVgWCG>! zQ53!pF~2Lg>G^?Qcj=Y+iy6;|(V3D=jlK`oA)?MMB`$nd_|8ym2FI7{2JFPyE}n0C zuDs(M5nf->CI`Q>O%~*IrZ-o^0YIu2JHU!neoWu$f?e_hnhaU3ohW(TU>5B(K_NGc zMeOH>(BUa{n5QZ6;;&LZ1}i~1*yMI#klWdG-tEpQcLygYDfJ9<(O9vVTnfz_LtqOT zHOt=>Y^VA3oXVdIbmpsAZ9o4vVNpu;v~sjHut662VVK&=w|yde>Q(&wbAlJ@5HZpZBEs%#v&B`-v0M0`&7V|Q z>s%cF3Y`Ro5*OJ?6HZd8+EAb78a8GoLax|LUOv1uTo-@WX z=1u;G)8nUyS?Nrc*5#KK*N$_5URuBN%>-trLQIe~aKS8?EF&&$JuiC_rofUGMKrdO zaR~_v19P&KTzkiB&p@r~^6xb0t-}}opk=#JlAqt~a0(vU^~c^fDBI90GpjlG6d!!< zq+fMZot>!S3`%m*lR6)RsWsiVT9guR=UgF`Jip$}W^LNBmCqh`J#)Ieb#*f*8xgyL z{;jaPBS&mJAt^WgRuDG}A4|5JghZiB=O}#8lMA4nu8)g#)yVyZAV?Wc#2KiHM%L35 z&`w8zE%yK{?b-jV(Ykyp}o6u8|@I@b$Tx5M(LzwOF-K7;Cs@1eGo6|z$knN%^a zcy?VXTk|1{fPI63s(#Kwak=-Y3G1&*w$J`yTGNXM+)c z&L(D~(Sy`13|m>YasPn{kc`c}?WMX&TznaR_v7v2;_Zd6I9IIT!`bx!X!mqi!rI-p zLuPm%c6WPX8+`%;`3A^#bYHpk{{8!%pp}~iSUvxKbKhH(6*q7}N7GEF>>^0s5uHZ0 zeMxA-*u#2u7PAgZyl#g$N>^}1`SX>W9YTU9dbm4jj$tio1%j=|l%-sbvnzH+h|g4m zWbcKQYBF?Aic!R)aoXz=MTliSw0}o|rxl+C8cV>h?DTsM%F)gO9W}9n_W(@wsP7$r z69x%{%Z4XUpRNtv=AQ8RsDo2_Bm&cgg%ozauMMk1x0^ci=FNS1x5ieHtq0lefO+_> zw^GZ_?eiWRTU#xNH}7*p)JcL=B9}KtS)d>gt%AREhdDPy`K=nwm5NPX>52}P5hE3w zOCgvJzunfB&NBoIZ8)p+6Gm%0t(8aLr7QT4$!?UfS@TU42 z8b-;<%Emc29rT@rVED4(eUsBq>&s8B<7wUYt#jLAl%$XUksshSA8TD|XzcYL)} zEKu2E*$6?p2cI^ZT%`pcLFsg2GwVpCGEnjKWg3xlaq~z!ct^$FnoCrhzY_Mzy&YRr z`sbqTsNJk|eX4=AU9sI=E3zC!ndSD(G4{0|P)&7?z3UlkEBX=hm(3 z(ggycBY!akrfFE`Qi${gBQ}}L=OQw>(rHeB?YCt0UG=!xl$YEx2`ygw{PUY5c@le! z+-vv0ys8Nq1c`KP^JdQka8(ZP{-2L)WW7L4o(X_zs(>KwAi5$8uCu%WhXQknvt%t; zr|if566J3%#)K;l=1(U;OtnRbDiY!4FYFdeV@H#p{}|@l+4MI3{YEx#IZVSQ{HEgqnBSgw!(Y>R2i7eI5HQt# z_wF~m;lr7myJTKrYWM{L?7E?6l}8(EFdE_xaw~Lnb{>Nue1qDZbYExW6#Q-b@2Y0g z=T^Iev7_O`FjBk@Uy7|sk_Sr|hI}9nhpDr$zOGUXVN0i<2 z9CC3!e6er{2ijzQ{>CW`K|bgp>s(w4(fGUEId!rn*T{8QK%?n)|0$>KyLRm&?eQgD zqF(W45UyK<3`wiyh-KP{*NttbGJG+_Oxd4D3)^jLYuf}x%?Bh!I#FdLe|f}%8VLXP zZYYk-#0)@Pv3%Kd3;m!P9)71G4d~I)Pi@Or^na~_)YL_Rio&b>A4uMS`dwLE(v>64 z9lMHio!i#NyXRO=IznQ)=+3ZvL&nfLH7MmUg%OD^D{yPpKM3ZYK89dVoJ(c)VVmNP9To6&u=y;!u_D! zXm!FJEJt8l1bb|WI`_g9aA#9_%x*ET+Zok!g zkTkLG{wkT7Yj%_r9BCAV&9A$(s)`CtV86hcHGu>HL%|q?I33$JZ!Wi<2!i(iE>>V1 zq7W6=@k#r#+57wG*_S>Y_*Z_IU8Y&qdm)v-c#HYQInyv@$s`YFhc{tYC5lxNA(*%U z^sfgWOn%3XJ8hME;9HdF^qiv*XY$_E-fpv{3m?{?z$PHXkTuea3#H<)_}F&MoaX)Z zxHiGt+xyDUOE+1-<=>rZ{gb{xfez{Dau_et6~V{-1)cE#Q6)}ib6&IiG`aDT_%0c| zH#NHv^3-Y2Iq$G>%(SmPWcn!RZpq-*x7KGS9(-Blb3Kpnm0$gw5xFY~I|%r{i*sozgeaycILd?YJAL zAcm*%dz8@?mrFFABgy+yX!8zRr0nHcXQEHK#SGA5D__}ST)h_GZNUqQ1{|wHsV_XO zWYV_nJ|ADhLXrtj(EvF#2V6*gG(b!D-&gC|5wn)m%>_JRTL$`>!oSb<~Z0rCAhF;J8_tzWib1@ThhqWoO zA-V=C1-izw>qV?uyd*s0d)*4MmY{1FA-nc37-JTGYGbZjsriW@OR|rjUv6@8=5Y?C z&PTUT9&ZGSUvS|cj4fM?3=Eo1-k?WLl#r2)=~u5-7@L^9HZ{xAHYOFW)37PG=8RB{ zOlAl|oPKY1TF}a6HNHL%aPum{qxS_1k^S*m(~XB?+h<86^XDySq7Qdf@T^sf$LERu zI9Sa85-~g>fWAmmYZD%94aN%}ocOXg`8OdvQbFFezKl}jn8^s1h6Y&z!k3qV($dmp zmLn>bTsR;3`1q#LdMSdc?i#+(0@%2K)C8@sU&RAt>k9`_A-M7kp`%1+x@?tNh=}z! z&HEoL@P=-2xrB2iL9)P>O^R2bR>gp7b0Ic`vWJzY6$-hSQ<23fD_7h@tr%T69g-9mV3MZ$p)^hrxJ9Dl zq5vzAhT(UstH9OD%E~1~D0@NSeb%6Qx*eh-l?<@Cx8d;m{(B$k6c{%W3A^_TvYN6d z%Z|dasnXd(Gg^mL}80thf7?zW zLhtZM>*(srqB5>pHCZ#;X}^Ft>;->oyU`M(Fl&f zH88K_(eT>Uno0?9e)|4B3IW`}=SBsDg#&6iosS#;_n#5G23cQVRk{dKYROLT_j-DI zk??_MA;Ey}BruDDnJuUm7pP)+h6!8FuT?)!CJK*&{rw2QOq>V@9yA6m-z7XeE{LKB zz&E>mn4jN_uk&oNN48R!aHP+8GMNjK-svliRiO(!!puV2Z!p43)VCM<)d*1%{_jo@FM|LLrJH(YJoflRv z-0A)t(^U^ffUv>Aq)MCOSKX4=1VEQ90m0aEU=`ATHUPQUbo1CkiP-k-YhSh;Uk!-( z?aMoNNDM0F&Ye4YvGq`roi{TzwLwm7<>i*84}bsuCD&RZ{5W%Yxw(M?UcedNLch}k z%SNxVvhsYrsTk7(f|I-C;vqw)uqJp9Uz;QKC&;?|Z>et)z9k+KjAtdDAyD^e)AngA zQKnPXd%tm+FV05oT*E)0hSecMLW*9`x#zZBnP_}us#4(WgcJ=5u}M|`b+=-tHn4^9 zw{QKgf685W`}pj|Zl59feGomK0W3VS{b4`&K@w|0A$B-^3--)&cr6M$5cjKPwRU+& z+oRdgMMxAs{Y10e2Yf0C7+p28)VyFc|B6E;1nb`|G_*Dd{;z^pI_jq}TjX@X^j8X& z#jquJ5Qvi%bY;JReqhTEky7NeAU;&lMTnGofZ}#zh50ue?WDDnLD5(_Yj;6h$q^RO zg)85TO-++=F3&I~Z6e?EQX4+OGYVs1s?4i5{!ibmxk8&H_ADQ&)Ek!=y1yN_*>7kAqJ}DEY#^ zM~IvT3YDUh7BmRhB#x})W@ln|$1Xeg@r9Ozs_Kxenp!$JHz*>H!gzI|Q$48GIO*m-#5X2(f3HBs>nXB}~0FIZA)b#XYnSFlGVX}w^BIE@-Y(B`7 z9d96RJj@l6?hCFuVuNa6YA|w25=utNgMAt_3k}5GfulzoKclW4DX&>vdhgde?UfkSKg)8_ym%r z%)l_qUZG*Hh>%GO6L`gXBpKQWZLvfD)_?G>Xp-4z1!Hcnqx(d`v}|D}at0ZFUE{MQV;_cayGK?o0_3+k51=T@Gd0<RCK;!`mQ6S;S$}$0h_ExrddNuq++w0YSHfRaT#=-UufwsDmA@ zj<@$XI}0QCC2AIm8#H^rT<$bxXA=NY?uS4Dcj#vk3aBXbblhuHVjL>V?Plr6e*JWi165$2&}~Wpm)w z5?+#-O<<+E4X;>6VWuU1U6qEC65Ned|dn_UKiD0F_VORUvK%xIc z&xglZAFq8U)#0p~kc~?M>L?2lTW~JQo?nwj`15brupgH}xQO#sqS!}~4a`nk?N9^Q z!b47=u8{2F+C3%4yZ!uRwrtrVKyDoOvFaBuUV!p8mX)(u2D0Ic#_{iR2sg05jPW1K zDjTG04-65pXyf@2*&wSADsbAMYhdP?Y3Q!zP?OJC4T=P@a>sAQ;B=*L=vh<8npv)H z@r?}zUU~Q*$>u%A-=&O!vMIOyhwIPu#l*$?l9H055TY1!wIM;L+(Fr9(xYzXGWD)F zlLU~8j~3Q$1E%D11?A+dOS#(Ec#R%Kn#!mdj%ok;?VA9$@g{fzS}VSi7s_a0Rer{i zaAl69M{o}A6fl4@QR@W$W;l{9A z%Q5>?aJ{h+Bo3-8k+SvHQBdsdxQeLg@C#m=q7K>gy4RP+nrEq59EWmq|7@Mj3jE8f z`1m|Yd7mekI~AL9KJdBhxY;3%=Do?zM4q*`uk<|C3;gfi$h42UGr4 zt9+P=C4J&k!AEJ{^B4zR9!$KRndy>EGj*tJ@OO^#^nl9n4NkeYXikQ1{{^`&rKFU^ zPWS`6L-bhq!tuGYrUen$+8J#LC0EIQ1Uh`lDLCswq0tCh#btKzU=4_s+g647`GZYa zRt^pv2vGK!d5|%=fIJsr#87qPoUVa1W&I7W4$c?r*=xIfypmH=go)mV#=x1bFg60T z)OQHx%t^x%T7!FfMiMpsJh3-^1Nf48{st^Hnhw1&2(s6`cro!7?ISN?XQlU~DfO3m z-VSMAcEQI2(PB_q)S6UM0mu+#qIe%iN4}7&`_uJ zw6sW+Q!>1P6BGm~o1Vgjbv~*V>)5J%j7QjeYhIyd;f}{NZEaW}>EO&>2rBQM#xVE* zon~yv!Gm^4jAKybkXh$dMmmmf@H9MdW*J?;gv$>h&55rT-{H5u=6!{d=UcylywL)-`9uCH5=A>RbeL4ZPA9$%8!8j95*$9}*c1z3 z>UZ9A!hMN-5r@w$Moy8v{zxPvCmbBn7Yu5hK#RIDZQVyk6BYd-P5VTnt356u#^O>U zbRf`?k(0B7=V)DYbo7Yq>6>D}8yrZd>4FGz8Q~%L2)!ItTuiJM5%r>YUnv>p63*Z| z>t0Em&O%ORef~Y0@ReL#{BI66wYy??$ITXXYp-M1Kx@Bjjho2^#2vK;ud{oQCBf8z zY$wnA(=$G>ql%rJM3pb@6H_!zy1p;C!um2l`E|`OG=xD*)PX1s(<-@IK~E~<#hQf9 zjRxyfXViApb!i{@aQQu|X~Hq9^OMf+oa< z8F80?w+H}?)|%}80#o_*s)Iy$Uyd{;rj2A|_NQ&uE-tY_JV8yT|7i<8Uf4*d>q;!N zjy87-^8F9ltKFJ1@d?0Zu>b$W&&0Mmm%XV%&r9waOT>y@OiawaC@0b~)R}QQIXWt3H=e^LNj{!10D-4T;!KU!my3%V!+1c3x>HV%&QmV)CrrKZseCTtT$@*xf=Uwvzz+}zxINFQ?89yF?Wanu2@CbW{h zislKdpJ36nf=70>I8#(bB^51f(vIK>{c?3&afbEz8;jRvp)OFiUJR z{dW*{(mr>dyvLh%uG$PiDy#UytbEy7~(1tW4WDTKyT@I-oZ z2Ri$^FsF$BgPTzE_wV0ia>*L<;-F*;^ZJr#jn71e7=4~f%=4pI)~IPkGFVhS%@bZ zJbHCKHtmkj_xNkD-lSw58W7CELC&>e(4s)&VQp~m;1-z6JJB~x;@v*L$<&PQ@8&>9 z7JjxP+ta5v6EiZ-y*N6_^8u&xH}Wtx32ZJac8SF7I3%hNYLBa+q_y!Y##xqeG!7bgaDGysX;tT~33Fl@?p%c{!DA^5UQ~DdpUX#3Z?_V)%}xQP&}2#I zY~Hhq&UX9n&poKrJ7jU0iWnC!?=&*CyhRXh!6(BN7>a{S;&buOrKNKf63{3h&yt;x zs+j$XZouaDw*fN|lqf~qT|nk~cX;Hfw^~EI%e`UEnjR9aADEkkSbK`pSYE*2wCqg# zf(0WT5)x8+PG|SI-+Oovp~gkXFcZr|GKp|`QPEah&+tu1dkbg5Y_?eDM6I#w=>o`B z^eDv?x_!eZL7KNSRIe4q;>p0+c(4tkU>;aP;JP4?DjyUtgjt@Iu?R)8Hqf7`YIYlw zF1JT_2Jgz1lF)YYljP5i9f#)?{0M?3Vxq5Wx8o!xzD1mnU#zB1Xku||hfK$pbEzZT zvww=|jxTBs)F`!}pkPQQ9Bvp7%#<|32Z%~C(*F#`@Jw^S_K-3?G8pgE?c2B60*h*f zmuHwytK6-YmIhD*;T@dd`09df=h$k~&|qhl#QN(}J@9xrH*DB&3S;HrD|$!;uqIK8 zFC`mhG#+N=>J{r?%#vUpC3Na$xJg0YD-FwrgHxvOju&cZkz4C3I>N6&UE;urW+w`L zED>7yACypNd^5MLf<^2;O+disJx*UaOx7=eoGXcnW>0ivKyP#J8`8PD(S=AN;V%ow zLM2^QHI0<*$tWr&!o`+oQaUs~|2iQ){tZbqauCQi=x8!6AoErW(V34~IXM+I9-NAG zXI;cGQ3YuqcQTKDu^s7TU#s$}QwWrOa2J;rsW@FmJc3XMs;FpN&JJvmHI~=3H}8Pk zvLIv*I#YebRiwSc7kQ6rw=lrts0wN5V*n2JKrPds9@8Qv2rBsF7d3ulLW6)i8%?`^ zxXu6G#Jt4EXhwl?X}R9W9YZ~G%h1X#jkG4*7<2Pp#^ofjYZkYnJRjiN8pSzu%f8g1 zZV5Beo|bdIG!6>gs&1(i82-jBTax>Gd%baGc^nW9gLc3>b$ubDoAq>cS#~SZs;*tT zhSA8o1it6{<<5QO-8ez~C@WVUc{fvk8@Dk_3vG7EZ$7cKg;Z=$#0tt@?&+~kB10Ks zp|NxHRUEy_hbDsN3EsmHt#0+MJEt@m#+s*oHv&Qr3Bs=-6PmKr8_t|LLyE`E;1qcF zV&z_=7dI#b2_sw#cu-(f1`MgAAhDiOhi<@_0&VG}zI9N_|6wSS1*dA+ro`SrSnGgk zAiFbUKay|mq9$2jO-4NS*9R89d*wO}(VM}O9IOODAdYc(k;C){q2ammR}=SV?H#RZ zyMZAm0?^qf_%q^gG~G=Q_Iu(az{w6sPov;y5V!jN7pYh$!UTb6t%+0zURhFPOW(2M z8fh^Ia^XVPn*k7UB@vFB7S2FKga)%t=ps-GPvm)QJF=Qg^FZ$tU%e_wosFLn)7IXu zcL#zum-4*Ejh3fB7gz=H@$KjC{cdSl+CAKR4HVJS;2;<9I3y z(b?O}1(i>>ssUN&NZZ5#oW3_FUQL&HOyw1V|JuD7t{S-n3KhC~b4tE5u>yb&V=I&(EQWFX;F zh1Ks2k_{-*cu0?NYA<3?SN)~r-h!Cr^z<6g10|tyWTn8%+BN<03#K(|iI4 z{THCQ?U-fm16VDd(s(!HfQm*=zL^l(oC^ZgBMniQMX+)P^S z%?+e~n#|~0D=0YiEo~`Ev|}hNeBR_NZtSgEyy`x&z{nZp!W7k@`()7BZ)RtmN7!B750|g4j`as|xn=jmb${S5z87dYlNiki~V51mO%DDBv>!8MDE zMQ(OIXCxz(9W>D+BtHkk^Y9bRwB8|b5E8=a16F7bW3cUAmv%x*)CT{;H5LN5@`xfU zs$)~6HRhgf1Ok^WC;mV)atV~~2PlR{gVe-zsjX<_NP#5IZlw#jb#Jy@jt6o)Wt;Zw zz9@5ox`K|JrUZ*7H5;ji9OUCS2s|cjTRL>C*rYJ9E>f2gMrV>TaxHqkv^?!CND?=D ze9i;21DgN|ug0@`Ivx271vnNrFMavgi%}-9Zu~2_!Y>dsrGxz0`iK2AYHKHY##c~c z$)Es4m3IEB6Tl)|jY(qWVA|+7?odel2S*A)e7Y}wAqJ=RArn6Lq*t#zw$-PyuQn$kw)uZa4I!OO&a}$H27-?T;u* z7HF~hHejwC|L*VNvgR+c9P9wFNSC^}Yajb2_2cEgzmC@Sc9_T8AOmVdLFz@O%4@-Aq5?R|bgMN%e>6LZAnCX*YwkYpuP-@c7M&2~U}N8|Q9hf?R_ zFK)IGE9gub=FL;UFTWkq%`wJ+m1AtY%*hn?g00d*mG73?h;UBtGT2HVmDoO!45o=IrAFb=rbJV@hozWPeIdlSbayMYcd=x zvAJ*0C{+}DLV6U>OcY9BDX_ZCjs`eY#yGDnj>KTujFZik>rT4d%!hjCZjFrErs>V1s)y{9|TMvX!;QVoPWdt@%BzLqmyi5VH0(OOp$ z^#<>pJbEwM%cKDVUeVtWUuY|nCThUmOxg&6@Hpld7K$N94zhPtza!_|d1CjjTD3|e zU~j3l=CbfZBMGs)ISV)-3Ae5NKa|Q4GCSygSxaHYXlRN`Nh&qv|X-iSFM_; zLp&k(A?{XxQ*sW0A?)c@=Hy+lj>^Mp(?IF4Ts~?_FYpVL=60eJ9uQY3sCw8+rMjR$ zW)z4vq7^e+R`Jp2wv*{}whALCw)%_0YfkudGXwWb= zOn>2t{1BxzAxzIlld7ZDFA)1j)JQl;Cr5)T`9f8N#5(cLQR@2j?z?CHp^-UB(i*0p z4?`F=fg~~}DDsd9t0j7FUfu$-1mj)p%-~g-u14kR*nj8GaM_SQ@H({W)5xi*`*7)= z!z*m_tGI6A@-swf!%M7)jpcRV#@ZE*0KvtXiBRV7S6pgq36yPinL8BCD z^Ity4y#bePxai4PmtqY#jh}u`bg7Jd&0YxfyiE$(%wBSnLblKF`#eFm zgritmcD^hT?%8Yte&->$Co=J}3p027n>TM%di%pLl0||SOjqqayIlk8X*a|sirmY< zE~=!pcf%V*qvyBM9M?%E#MgKfQ4@;X0cGRR}orB9g5P?u&^=n zZ})+bqFL#I`{@88w8Mc33N0g+KviksypY4tN34JdJ31;V6VV{PXd7=lz0!ztGa0>@ z^I?G0aAic3GLeC~wfmQEzy(ubptgH4Cp~}InLQLZhJE!B6iV*>`^LGC{d*8$@cwZB zSy_)RUg7W#Mj@%l2u70J=1?Dj*g1xQO5}k4sC3~9MEiAe1dN*5Q|&6)a#2rv$qDfo zE|T%Tv6|S1JH)!s6-{By1cn-%LJ{_XLM;mB?}n-_yG_87RrkNpGyuUsW?3LdBhY%! z2L6syAj3T3rEXJIEm`G&#Q)>85-aY~r-dq1}{pbXVt;Uqj*iIe8$ z+=K4jo#=}_R>KjG-yHj!aXE5R#G4OqONLJz#}1^^)k2pDk`!~?_) zH8YF5@fjCldRPmPzm|CAa(j@Ch(d4OuvSDrrRS~kL&;eQq{w|N_GNRaNwWewIl zU0gUJYn+g@yzKOByi30rgjN)%^x8VX+icPS?Zzo;QmN@@$OR~*)03$-;0QYkNO26ky}##*HK3748IH{>)8%fkG#5yts!TuuJXj?RBs2t`LBvBdK?a zu;C5k$p#O40Mjw4%K~h3)Agm5hFD--rY;4Sm>(=^ud#MyrA9IWMEo~pG6L9f*K_!{ zV=e3C94=&TabuB7nT)T{KJ^3T`TN53-ctN`7*GVNgFe#td1weweFgW zFuPMIEnpNyetid>@qMa9cuu-W43W;6y2`t3&|d|~2T4?~gLOe8Bwmg-N=_K<4$mBZ1Y}RJ`a~CemD0Ut!j?S&mGG?Ls;5r{;NiG|y@|ETKwo<$i z0!h~iGr^L2E@>R}d<;WJ%kM~EUgHP@fUJ~MD~7ZV^hNDRc{l@n%FfL#9?{*cU0u9xbt^GAq4nU!HRe|s&}9I zv*1{X^P~_zy9!q;+a5S@AYjk>O0wrrYF9j1Y*JPKT#6vb>;xo^cFemA#M(kr7CGU8 z+ig}L2&}tRtZ=XW^zyX^s__&8vvB~im8Zu%^)x~WBU*lej_ZSMb$Z^hEfP7Ip2(&za?)(cb znTo*-<%QRANG2`zSb%E`GB6ydY+D&c58Ywc#9knU8c*l*t^TiJjZF~^yFTS{t{ZZ3 zxd`81BX2lnMN-Fg1z+%I3;q()g8+<46=oQ79D1tyfbg)BNt0OfpHYs+N_37=mQgO{ zhTQG<4Xjf~l-{i-eG@R++gaI4^_go$-a!79S9}5#u%V9Oj9>=&oi5}4p!V>Nch)0n zO5q!ekbMUj=PJUWZfw{PIq~lQ+!8J)AJ#cnlwG1SI)Q=lYb}=+n1QmBvxB`zap=K# zl^smOanYHaft&?bRdh76SXwu#t{8}ukDL%Ad&NfDfaq%8zotZ$nUy%x96Sm6iPVZb zhN@gO@DFI-Pe7&lH*-WzMEvj1D<)^BpD7T2ETU^cWk^^H^sVd4KuU&*!e4;~{dZcg zh;VV{{p?eMyZI9e@~qt8W0Dq7CrvrsUpyfh%X(|dq+(%s z-EE|wFZjQl6TsA2%OIX7$nR$*V`c4Q{B(SQPgo-H{RWf~l^^L7)S{t44*ET>c;94; z_KGb1s`%$f1Te=0IEBk4RgYC;g`|<4SkB;m_(aaK>sIYMG@>e9EygD2`a-jF^gZNK zR(ypM%0HNS!(Nq_-A>5s#E}nilA0Cg}(<$^DQn3=_>&$Yj){tW`XMqVsk8Z*ON!b41il`Jo+B{eP}$CWP_dE{h(1fchlxj6FZXKX#3_uX z#(fbbe)jrRivjEmiZ12cy$E9pU9LA4SGqb@_@i-QBWT|7F#8bHPFh@U42Gk+5^jrM zxA?1^lw`(2+?K-dZ7Y_M&$>ez85gahs&WoRJJk(_n$;*yLE=!V*Zy7W?IzzS9K{!Q zb$Ps!F6E3Ug^h@@DP{wzJTNH4R_6k_E3i8nHwQk}C(R3rouVr|V0iFw{-{Kx{@b;x zlk)Cb{s5INF;gFQGe*oV-0oX0Z!0BmJ@S9wXv_bkT&bxZQJyrS%E3~0oF}s;cAOTI z@I&9_!U{%ud#ykRpD43zPu;>|>FpzI`m@uFqD2vJIZA}6O)qyOBUtr+4|!iZwy$FU zSEs;G&_ISI4_yq@3j8R>^%qOLfJG&_C`t*$-XzZxP`4dIZNE<+-yP4Y$lb0p_V-`3 zUgGw9@|*ioi_h*;y0?KktuJk cK=^sxkZR_2E59$<2{ezv)78&qol`;+0I{z^(EtDd diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/favicon.png b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/favicon.png deleted file mode 100644 index 2ad073a9cded3f809a2a5bae30a339bbd8666c53..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1242 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 diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/login-back.svg b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/login-back.svg deleted file mode 100644 index 8c719e2b..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/login-back.svg +++ /dev/null @@ -1,147 +0,0 @@ - - - - - - - - - - - - - - - - - - - - image/svg+xml - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/login-icon.png b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/login-icon.png deleted file mode 100644 index 6b43c768479d6926c8fd799568a2952b1b94da6b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 9344 zcmeI2+iMhe6vwAxiwOP$#fLulTJSXy#A-x?5Jd!i^Ac+!K8T>DQ0PNnS|6&2&x+T0 z#oJo%tvaDH^ODk4lJD0W@U*Y%aX=A@y);CsV#doV}zx-nEoMeYpzVAMi z?Mni+<=B>StNXGw+1TgU_QBe-lY7{0H2qfDessv$L`y6E#oSuu`!N$L2+oeXua9|} zM@fRF5ev{fYI02)!3Z@5fY`_dmGi5160q| zruVhhHbHvP!6f|kFvU)js}11_Cjmcy@Rf+k7EE*34E#$&#c+UIXe`nWGjbW za3IN9_H%kmbgJov>##U3?RaJaHed93UwgJ0tCBrhqgb8fIWTQw zK`7B`XOz?G^F=`rN;e<%^8+j<6r>3@+9no+4ko+VC&@bPnCOAw(c(a8+m&|LVR77~ zz4h38K0X;jl}GAfKF%M{2SVIg1r$ z*PW;E^Pdl6!$xBk8liN?Ui#9{z|6$Nl!?;Ad*$q)uG#(~8G(xUZ z1{LCC*oH<(vdhCxK?o|jbg>~M0Z?{Lv - - - - - image/svg+xml - - OB-Publisher-logo-Dark-background - - - - - - - - OB-Publisher-logo-Dark-background - - - - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/logo-inverse.svg b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/logo-inverse.svg deleted file mode 100644 index 6e7f1d89..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/logo-inverse.svg +++ /dev/null @@ -1,60 +0,0 @@ - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/logo.png b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/logo.png deleted file mode 100644 index 2de1612c71ffeae3a1ce0cc95cd453fdeff4f7d9..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 27990 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 diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/openid-input.gif b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/openid-input.gif deleted file mode 100644 index cde836c893f64bcfec04b9c817e3371ff122fe19..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 237 zcmVb{bmUKcqz}))c5uC(7v?)v4a2P)ZNa- z@$&T2)z|&~{r~^}A^8LV00000EC2ui01yBW000GQ;3tk`X`bk)Wk@<6#nZYULKH{p zEx|?+kif!I0vIL|#ZMubBmjWH2OtmxIFVa~6JQ7!1CK!f5W#StOTv&C3=E8h2vI1s n+#cd5;2fT3B_0kF0v!+!GARoV78n&7dMN`JIW(4+BOw4gP{MS* diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/repeat.jpg b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/repeat.jpg deleted file mode 100644 index 891eea015ad5e7f82b3ff39074074a7bcb310588..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4368 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 P`L_OAdVVx}{Fna$Kt@nl diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/wso2-open-banking-logo.png b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/images/wso2-open-banking-logo.png deleted file mode 100644 index ae180a4290ff067252cb374ba4c8890418f8f8ad..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4316 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 RBJTzi{~}!5FpyMzgMq<0!SwX?J&5h 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%0AI*60XD!M>^;0)ws$VV#n#M{WR0FV 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%0Av6{^= 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%0A#>Li_ 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 Hi~{~2Im#m_ diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/includes/consent_bottom.jsp b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/includes/consent_bottom.jsp deleted file mode 100644 index 8cbea3e7..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/includes/consent_bottom.jsp +++ /dev/null @@ -1,30 +0,0 @@ -<%-- - ~ Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - ~ - ~ WSO2 LLC. 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. - --%> - -<%@ page contentType="text/html;charset=UTF-8" %> - -
- - - - - - - - - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/includes/consent_top.jsp b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/includes/consent_top.jsp deleted file mode 100644 index 15427d5e..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/includes/consent_top.jsp +++ /dev/null @@ -1,52 +0,0 @@ -<%-- - ~ Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - ~ - ~ WSO2 LLC. 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. - --%> - -<%@ page contentType="text/html;charset=UTF-8" language="java" %> -<%@ page import="org.json.JSONArray" %> -<%@ page import="org.json.JSONObject" %> -<%@ page import="org.owasp.encoder.Encode" %> - -<%@ page import="java.util.List" %> -<%@ page import="java.util.ArrayList" %> -<%@ page import="org.apache.commons.lang.StringUtils" %> -<%@ page import="org.apache.commons.lang.ArrayUtils" %> -<%@ page import="java.util.stream.Stream" %> - -<%@ taglib prefix = "fmt" uri = "http://java.sun.com/jsp/jstl/fmt" %> -<%@ taglib prefix = "c" uri = "http://java.sun.com/jsp/jstl/core" %> - - - - - - - - - - -
-
-
- -
-
-
- - - - - - - - - -
-
-
- -
-
- -
- - diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/ob_default.jsp b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/ob_default.jsp deleted file mode 100644 index 6e7c58c4..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/ob_default.jsp +++ /dev/null @@ -1,89 +0,0 @@ -<%-- - ~ Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - ~ - ~ WSO2 LLC. 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. - --%> - -<%@ taglib prefix = "fmt" uri = "http://java.sun.com/jsp/jstl/fmt" %> -<%@ taglib prefix = "c" uri = "http://java.sun.com/jsp/jstl/core" %> - - -<% - session.setAttribute("configParamsMap", request.getAttribute("data_requested")); -%> -
-
- -
- diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/ob_default_consent.jsp b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/ob_default_consent.jsp deleted file mode 100644 index 6e7c58c4..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/ob_default_consent.jsp +++ /dev/null @@ -1,89 +0,0 @@ -<%-- - ~ Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - ~ - ~ WSO2 LLC. 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. - --%> - -<%@ taglib prefix = "fmt" uri = "http://java.sun.com/jsp/jstl/fmt" %> -<%@ taglib prefix = "c" uri = "http://java.sun.com/jsp/jstl/core" %> - - -<% - session.setAttribute("configParamsMap", request.getAttribute("data_requested")); -%> -
-
- -
- diff --git a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/privacy_policy.jsp b/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/privacy_policy.jsp deleted file mode 100644 index 3e7bcdcc..00000000 --- a/open-banking-accelerator/internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp/src/main/webapp/privacy_policy.jsp +++ /dev/null @@ -1,238 +0,0 @@ -<%-- - ~ Copyright (c) 2023, WSO2 LLC. (https://www.wso2.com). - ~ - ~ WSO2 LLC. 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. - --%> - -<%@ page import="java.util.ResourceBundle" %> -<%@ page import="static com.wso2.openbanking.accelerator.consent.extensions.authservlet.impl.util.Utils.i18n" %> -<%@page contentType="text/html; charset=UTF-8"%> -<% - String BUNDLE = "com.wso2.openbanking.authentication.webapp.i18n"; - ResourceBundle resourceBundle = ResourceBundle.getBundle(BUNDLE, request.getLocale()); -%> - - - - - - - - - -
-
- -
-
- -
- - -
-
-
-
-
-
-

- <%=i18n(resourceBundle, "wso2.open.banking")%> - <%=i18n(resourceBundle, "privacy.policy.general")%> -

-
- -
-
-

About WSO2 Open Banking Solution

-

WSO2 Open Banking Solution (referred to as “WSO2 Open Banking” within this policy) is a comprehensive Open Banking Solution that is supporting PSD2 compliance along with more value added features.

-
- -
-

Privacy Policy

-

This policy describes how WSO2 Open Banking captures your personal information, the purposes of collection, and information about the retention of your personal information.

-

Please note that this policy is for reference only, and is applicable for the software as a product. WSO2 Inc. and its developers have no access to the information held within WSO2 Open Banking. Please see the Disclaimer section for more information

-

Entities, organisations or individuals controlling the use and administration of WSO2 Open Banking should create their own privacy policies setting out the manner in which data is controlled or processed by the respective entity, organisation or individual.

-
- -
-

What is personal information?

-

WSO2 Open Banking considers anything related to you, and by which you may be identified, as your personal information. This includes, but is not limited to:

-
    -
  • Your user name (except in cases where the user name created by your employer is under contract)
  • -
  • Your date of birth/age
  • -
  • IP address used to log in
  • -
  • Your device ID if you use a device (e.g., phone or tablet) to log in
  • -
-

However, WSO2 Open Banking also collects the following information that is not considered personal information, but is used only for statistical purposes. The reason for this is that this information can not be used to track you.

-
    -
  • City/Country from which you originated the TCP/IP connection
  • -
  • Time of the day that you logged in (year, month, week, hour or minute)
  • -
  • Type of device that you used to log in (e.g., phone or tablet)
  • -
  • Operating system and generic browser information
  • -
-
- -
-

Collection of personal information

-

WSO2 Open Banking collects your information only to serve your access requirements. For example: -

    -
  • WSO2 Open Banking uses your IP address to detect any suspicious login attempts to your account.
  • -
  • WSO2 Open Banking uses attributes like your first name, last name, etc., to provide a rich and personalized user experience.
  • -
  • WSO2 Open Banking uses your security questions and answers only to allow account recovery.
  • -
-
- -
-

Tracking Technologies

-

WSO2 Open Banking collects your information by:

-
    -
  • Collecting information from the user profile page where you enter your personal data.
  • -
  • Tracking your IP address with HTTP request, HTTP headers, and TCP/IP.
  • -
  • Tracking your geographic information with the IP address.
  • -
  • Tracking your login history with browser cookies. Please see our cookie policy for more information.
  • -
-
- -
-

Use of personal information

-

WSO2 Open Banking will only use your personal information for the purposes for which it was collected (or for a use identified as consistent with that purpose).

-

WSO2 Open Banking uses your personal information only for the following purposes.

-
    -
  • To provide you with a personalized user experience. WSO2 Open Banking uses your name and uploaded profile pictures for this purpose.
  • -
  • To protect your account from unauthorized access or potential hacking attempts. WSO2 Open Banking uses HTTP or TCP/IP Headers for this purpose.
  • -
      -
    • This includes:
    • -
        -
      • IP address
      • -
      • Browser fingerprinting
      • -
      • Cookies
      • -
      -
    -
  • Derive statistical data for analytical purposes on system performance improvements. WSO2 Open Banking will not keep any personal information after statistical calculations. Therefore, the statistical report has no means of identifying an individual person.
  • -
      -
    • WSO2 Open Banking may use:
    • -
        -
      • IP Address to derive geographic information
      • -
      • Browser fingerprinting to determine the browser technology or/and version
      • -
      -
    -
-
- -
-

Disclosure of personal information

-

WSO2 Open Banking only discloses personal information to the relevant applications (also known as “Payment Service Providers (PSPs)”) that are registered with WSO2 Open Banking. These applications are registered by the identity administrator of your entity or organization. Personal information is disclosed only for the purposes for which it was collected (or for a use identified as consistent with that purpose), as controlled by such Service Providers, unless you have consented otherwise or where it is required by law.

-
- -
- -

Please note that the organisation, entity or individual running WSO2 Open Banking may be compelled to disclose your personal information with or without your consent when it is required by law following due and lawful process.

-
- -
-

Storage of personal information

-
- -
-

Where your personal information is stored

-

WSO2 Open Banking stores your personal information in secured databases. WSO2 Open Banking exercises proper industry accepted security measures to protect the database where your personal information is held. WSO2 Open Banking as a product does not transfer or share your data with any third parties or locations.

-

WSO2 Open Banking may use encryption to keep your personal data with an added level of security.

-
- -
-

How long your personal information is retained

-

WSO2 Open Banking retains your personal data as long as you are an active user of our system. You can update your personal data at any time using the given self-care user portals.

-

WSO2 Open Banking may keep hashed secrets to provide you with an added level of security. This includes:

-
    -
  • Current password
  • -
  • Previously used passwords
  • -
-
- -
-

How to request removal of your personal information

-

You can request the administrator to delete your account. The administrator is the administrator of the tenant you are registered under, or the super-administrator if you do not use the tenant feature.

-

Additionally, you can request to anonymize all traces of your activities that WSO2 Open Banking may have retained in logs, databases or analytical storage.

-
- -
-

More information

-
-
- -
-

Changes to this policy

-

Upgraded versions of WSO2 Open Banking may contain changes to this policy and revisions to this policy will be packaged within such upgrades. Such changes would only apply to users who choose to use upgraded versions.

-

The organization running WSO2 Open Banking may revise the Privacy Policy from time to time. You can find the most recent governing policy with the respective link provided by the organization running WSO2 IS 5.5. The organization will notify any changes to the privacy policy over our oficial public channels.

-
- -
-

Your choices

-

If you are already have a user account within WSO2 Open Banking, you have the right to deactivate your account if you find that this privacy policy is unacceptable to you.

-

If you do not have an account and you do not agree with our privacy policy, you can chose not to create one.

-
- -
-

Contact us

-

Please contact WSO2 if you have any question or concerns regarding this privacy policy.

-

https://wso2.com/contact/

-
- -
-

Disclaimer

-
    -
  1. WSO2, its employees, partners, and affiliates do not have access to and do not require, store, process or control any of the data, including personal data contained in WSO2 Open Banking. All data, including personal data is controlled and processed by the entity or individual running WSO2 Open Banking. WSO2, its employees partners and affiliates are not a data processor or a data controller within the meaning of any data privacy regulations. WSO2 does not provide any warranties or undertake any responsibility or liability in connection with the lawfulness or the manner and purposes for which WSO2 Open Banking is used by such entities or persons.

  2. -
  3. This privacy policy is for the informational purposes of the entity or persons running WOS2 IS 5.5.0 and sets out the processes and functionality contained within WSO2 Open Banking regarding personal data protection. It is the responsibility of entities and persons running WSO2 Open Banking to create and administer its own rules and processes governing users’ personal data, and such rules and processes may change the use, storage and disclosure policies contained herein. Therefore users should consult the entity or persons running WSO2 Open Banking for its own privacy policy for details governing users’ personal data.
  4. -
-
-
- -
-
- - - - - - - - - - - - - - - diff --git a/open-banking-accelerator/pom.xml b/open-banking-accelerator/pom.xml index bbb104ba..92af9849 100644 --- a/open-banking-accelerator/pom.xml +++ b/open-banking-accelerator/pom.xml @@ -33,27 +33,27 @@ 3.2.11-SNAPSHOT - components/com.wso2.openbanking.accelerator.common - components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.dao - components/com.wso2.openbanking.accelerator.gateway - components/com.wso2.openbanking.accelerator.identity - components/com.wso2.openbanking.accelerator.runtime - components/com.wso2.openbanking.accelerator.keymanager - components/consent-management/com.wso2.openbanking.accelerator.consent.extensions - components/com.wso2.openbanking.accelerator.data.publisher - components/consent-management/com.wso2.openbanking.accelerator.consent.mgt.service - components/ob-throttler/com.wso2.openbanking.accelerator.throttler.dao - components/ob-throttler/com.wso2.openbanking.accelerator.throttler.service - components/event-notifications/com.wso2.openbanking.accelerator.event.notifications.service - components/account-metadata/com.wso2.openbanking.accelerator.account.metadata.service - internal-apis/internal-webapps/com.wso2.openbanking.accelerator.ciba.authentication.endpoint - internal-apis/internal-webapps/com.wso2.openbanking.accelerator.consent.endpoint - internal-apis/internal-webapps/com.wso2.openbanking.accelerator.dcr.endpoint - internal-apis/internal-webapps/com.wso2.openbanking.authentication.webapp - internal-apis/internal-webapps/com.wso2.openbanking.accelerator.application.info.endpoint - internal-apis/internal-webapps/com.wso2.openbanking.accelerator.demo.backend - components/com.wso2.openbanking.accelerator.service.activator - internal-apis/internal-webapps/com.wso2.openbanking.accelerator.push.authorization.endpoint - internal-apis/internal-webapps/com.wso2.openbanking.accelerator.event.notifications.endpoint + + + + + + + + + + + + + + + + + + + + + +