From f81e904fe1ba8438080e11b5fa8012da4658a25e Mon Sep 17 00:00:00 2001 From: Github Actions Date: Thu, 15 Jun 2023 14:07:13 +0000 Subject: [PATCH 1/6] Version bump --- box.json | 2 +- changelog.md | 6 +++++- 2 files changed, 6 insertions(+), 2 deletions(-) diff --git a/box.json b/box.json index e80bf25..5eab7d6 100644 --- a/box.json +++ b/box.json @@ -1,7 +1,7 @@ { "name":"ColdBox Validation", "author":"Ortus Solutions ", - "version":"4.3.1", + "version":"4.4.0", "location":"https://downloads.ortussolutions.com/ortussolutions/coldbox-modules/cbvalidation/@build.version@/cbvalidation-@build.version@.zip", "slug":"cbvalidation", "type":"modules", diff --git a/changelog.md b/changelog.md index ae7bdcc..827a369 100644 --- a/changelog.md +++ b/changelog.md @@ -9,6 +9,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [Unreleased] +## [4.3.1] - 2023-06-15 + ### Fixed - Only perform type evaluation if target value is not null or empty string #75 @@ -272,7 +274,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Create first module version -[Unreleased]: https://github.com/coldbox-modules/cbvalidation/compare/v4.3.0...HEAD +[Unreleased]: https://github.com/coldbox-modules/cbvalidation/compare/v4.3.1...HEAD + +[4.3.1]: https://github.com/coldbox-modules/cbvalidation/compare/v4.3.0...v4.3.1 [4.3.0]: https://github.com/coldbox-modules/cbvalidation/compare/v4.2.0...v4.3.0 From 05a58a87a91698bc8483dfe66fb9f78974615ebd Mon Sep 17 00:00:00 2001 From: Dave L Date: Tue, 12 Sep 2023 11:35:02 -0700 Subject: [PATCH 2/6] Update RequiredIf Validator to accept UDF/closure (#79) The closure should return boolean (true/false). True=Key is required. False=key is not required. --- models/validators/RequiredIfValidator.cfc | 25 +++++- .../validators/RequiredIfValidatorTest.cfc | 79 ++++++++++++++++--- 2 files changed, 91 insertions(+), 13 deletions(-) diff --git a/models/validators/RequiredIfValidator.cfc b/models/validators/RequiredIfValidator.cfc index 62ff365..9f17b52 100644 --- a/models/validators/RequiredIfValidator.cfc +++ b/models/validators/RequiredIfValidator.cfc @@ -38,6 +38,7 @@ component struct rules ){ var isRequired = true; + var errorMetadata = {}; // If you passed in simple data, simply check that the target field has a value if ( isSimpleValue( arguments.validationData ) && len( arguments.validationData ) ) { @@ -60,13 +61,27 @@ component .reduce( function( result, key, value ){ return ( arguments.value && arguments.result ); }, true ); - } else { - validationResult.addError( + // If passed a UDF/closure + } else if ( + isCustomFunction( arguments.validationData ) || + isClosure( arguments.validationData ) + ) { + + // Validate against the UDF/closure + var isRequired = arguments.validationData( + isNull( arguments.targetValue ) ? javacast( "null", "" ) : arguments.targetValue, + arguments.target, + errorMetadata + ); + + } else { + + validationResult.addError( validationResult.newError( message = "The target for RequiredIf must be a simple field name or a struct of field to target value pairs.", field = arguments.field, validationType = getName(), - rejectedValue = arguments.validationData, + rejectedValue = isSimpleValue( arguments.validationData ) ? arguments.validationData : "", validationData = arguments.validationData ) ); @@ -94,7 +109,9 @@ component validationData : arguments.validationData }; - validationResult.addError( validationResult.newError( argumentCollection = args ) ); + validationResult.addError( + validationResult.newError( argumentCollection = args ).setErrorMetadata( errorMetadata ) + ); return false; } diff --git a/test-harness/tests/specs/validators/RequiredIfValidatorTest.cfc b/test-harness/tests/specs/validators/RequiredIfValidatorTest.cfc index 501b998..3ca61cd 100644 --- a/test-harness/tests/specs/validators/RequiredIfValidatorTest.cfc +++ b/test-harness/tests/specs/validators/RequiredIfValidatorTest.cfc @@ -106,17 +106,78 @@ component extends="coldbox.system.testing.BaseModelTest" model="cbvalidation.mod expect( model.validate( result, mock, "testField", "", "name" ) ).toBeFalse(); - // expect( - // model.validate( - // result, - // mock, - // "testField", - // "", - // "missing" - // ) - // ).toBeTrue(); + expect( + model.validate( + result, + mock, + "testField", + "", + "missing" + ) + ).toBeTrue(); + } ); + it( "can accept a closure as validationData", function(){ + var mock = createStub() + .$( "getName", "luis" ) + .$( "getRole", "admin" ) + .$( "getMissing", javacast( "null", "" ) ); + var result = createMock( "cbvalidation.models.result.ValidationResult" ).init(); + + expect( + model.validate( + result, + mock, + "testField", + "", + isRequired1 + ) ).toBeFalse(); + + expect( + model.validate( + result, + mock, + "testField", + "", + isRequired2 + ) + ).toBeTrue(); + } ); + it( "can use custom error metadata", function(){ + var mock = createStub() + .$( "getName", "luis" ) + .$( "getRole", "admin" ) + .$( "getMissing", javacast( "null", "" ) ); + var result = createMock( "cbvalidation.models.result.ValidationResult" ).init(); + + expect( + model.validate( + result, + mock, + "testField", + "", + isRequired3 + ) ).toBeFalse(); + + var errorMetadata = result.getErrors()[ 1 ].getErrorMetadata(); + + expect( errorMetaData ).toHaveKey( "customMessage" ); + expect( errorMetaData.customMessage ).toBe( "This is custom data" ); + } ); } ); } + private function isRequired1( value, target, errorMetadata ){ + return true; + } + + private function isRequired2( value, target, errorMetadata ){ + return false; + } + + private function isRequired3( value, target, errorMetadata ){ + arguments.errorMetadata[ "customMessage" ] = "This is custom data"; + return true; + } + } From 8a460abb0e70a20376dbacdee5732e2b444d7f6e Mon Sep 17 00:00:00 2001 From: lmajano Date: Tue, 12 Sep 2023 18:35:52 +0000 Subject: [PATCH 3/6] Apply cfformat changes --- models/validators/RequiredIfValidator.cfc | 39 ++++++------- .../validators/RequiredIfValidatorTest.cfc | 57 +++++-------------- 2 files changed, 31 insertions(+), 65 deletions(-) diff --git a/models/validators/RequiredIfValidator.cfc b/models/validators/RequiredIfValidator.cfc index 9f17b52..fa77e35 100644 --- a/models/validators/RequiredIfValidator.cfc +++ b/models/validators/RequiredIfValidator.cfc @@ -37,8 +37,8 @@ component any validationData, struct rules ){ - var isRequired = true; - var errorMetadata = {}; + var isRequired = true; + var errorMetadata = {}; // If you passed in simple data, simply check that the target field has a value if ( isSimpleValue( arguments.validationData ) && len( arguments.validationData ) ) { @@ -61,22 +61,19 @@ component .reduce( function( result, key, value ){ return ( arguments.value && arguments.result ); }, true ); - // If passed a UDF/closure - } else if ( - isCustomFunction( arguments.validationData ) || - isClosure( arguments.validationData ) - ) { - - // Validate against the UDF/closure - var isRequired = arguments.validationData( - isNull( arguments.targetValue ) ? javacast( "null", "" ) : arguments.targetValue, - arguments.target, - errorMetadata - ); - - } else { - - validationResult.addError( + // If passed a UDF/closure + } else if ( + isCustomFunction( arguments.validationData ) || + isClosure( arguments.validationData ) + ) { + // Validate against the UDF/closure + var isRequired = arguments.validationData( + isNull( arguments.targetValue ) ? javacast( "null", "" ) : arguments.targetValue, + arguments.target, + errorMetadata + ); + } else { + validationResult.addError( validationResult.newError( message = "The target for RequiredIf must be a simple field name or a struct of field to target value pairs.", field = arguments.field, @@ -109,9 +106,9 @@ component validationData : arguments.validationData }; - validationResult.addError( - validationResult.newError( argumentCollection = args ).setErrorMetadata( errorMetadata ) - ); + validationResult.addError( + validationResult.newError( argumentCollection = args ).setErrorMetadata( errorMetadata ) + ); return false; } diff --git a/test-harness/tests/specs/validators/RequiredIfValidatorTest.cfc b/test-harness/tests/specs/validators/RequiredIfValidatorTest.cfc index 3ca61cd..b1ca2bb 100644 --- a/test-harness/tests/specs/validators/RequiredIfValidatorTest.cfc +++ b/test-harness/tests/specs/validators/RequiredIfValidatorTest.cfc @@ -106,78 +106,47 @@ component extends="coldbox.system.testing.BaseModelTest" model="cbvalidation.mod expect( model.validate( result, mock, "testField", "", "name" ) ).toBeFalse(); - expect( - model.validate( - result, - mock, - "testField", - "", - "missing" - ) - ).toBeTrue(); + expect( model.validate( result, mock, "testField", "", "missing" ) ).toBeTrue(); } ); - it( "can accept a closure as validationData", function(){ + it( "can accept a closure as validationData", function(){ var mock = createStub() .$( "getName", "luis" ) .$( "getRole", "admin" ) .$( "getMissing", javacast( "null", "" ) ); var result = createMock( "cbvalidation.models.result.ValidationResult" ).init(); - expect( - model.validate( - result, - mock, - "testField", - "", - isRequired1 - ) ).toBeFalse(); + expect( model.validate( result, mock, "testField", "", isRequired1 ) ).toBeFalse(); - expect( - model.validate( - result, - mock, - "testField", - "", - isRequired2 - ) - ).toBeTrue(); + expect( model.validate( result, mock, "testField", "", isRequired2 ) ).toBeTrue(); } ); - it( "can use custom error metadata", function(){ + it( "can use custom error metadata", function(){ var mock = createStub() .$( "getName", "luis" ) .$( "getRole", "admin" ) .$( "getMissing", javacast( "null", "" ) ); var result = createMock( "cbvalidation.models.result.ValidationResult" ).init(); - expect( - model.validate( - result, - mock, - "testField", - "", - isRequired3 - ) ).toBeFalse(); + expect( model.validate( result, mock, "testField", "", isRequired3 ) ).toBeFalse(); - var errorMetadata = result.getErrors()[ 1 ].getErrorMetadata(); + var errorMetadata = result.getErrors()[ 1 ].getErrorMetadata(); - expect( errorMetaData ).toHaveKey( "customMessage" ); - expect( errorMetaData.customMessage ).toBe( "This is custom data" ); - + expect( errorMetaData ).toHaveKey( "customMessage" ); + expect( errorMetaData.customMessage ).toBe( "This is custom data" ); } ); } ); } - private function isRequired1( value, target, errorMetadata ){ + private function isRequired1( value, target, errorMetadata ){ return true; } - private function isRequired2( value, target, errorMetadata ){ + private function isRequired2( value, target, errorMetadata ){ return false; } - private function isRequired3( value, target, errorMetadata ){ + private function isRequired3( value, target, errorMetadata ){ arguments.errorMetadata[ "customMessage" ] = "This is custom data"; - return true; + return true; } } From a8cbe0b2d2d2385cb3cc69ff79a5ee1ed8c555a9 Mon Sep 17 00:00:00 2001 From: Dave L Date: Thu, 14 Sep 2023 03:07:14 -0700 Subject: [PATCH 4/6] [Breaking Change] Update UDF validator to treat null/empty values as valid (#80) * Update UDF validator to treat null/empty values as valid * Bugfix. UDF validator now ignores nulls --- models/validators/MethodValidator.cfc | 5 ----- models/validators/UDFValidator.cfc | 7 ++++++- test-harness/tests/specs/ValidationIntegrations.cfc | 3 ++- test-harness/tests/specs/validators/UDFValidatorTest.cfc | 2 +- 4 files changed, 9 insertions(+), 8 deletions(-) diff --git a/models/validators/MethodValidator.cfc b/models/validators/MethodValidator.cfc index 0e6b30e..7c2d990 100644 --- a/models/validators/MethodValidator.cfc +++ b/models/validators/MethodValidator.cfc @@ -38,11 +38,6 @@ component extends="BaseValidator" accessors="true" singleton { return true; } - // return true if no data to check, type needs a data element to be checked. - if ( isNull( arguments.targetValue ) || isNullOrEmpty( arguments.targetValue ) ) { - return true; - } - // Validate via method if ( invoke( diff --git a/models/validators/UDFValidator.cfc b/models/validators/UDFValidator.cfc index 14e6833..972a3ae 100644 --- a/models/validators/UDFValidator.cfc +++ b/models/validators/UDFValidator.cfc @@ -34,9 +34,14 @@ component extends="BaseValidator" accessors="true" singleton { ){ var errorMetadata = {}; + // return true if no data to check, type needs a data element to be checked. + if ( isNull( arguments.targetValue ) || isNullOrEmpty( arguments.targetValue ) ) { + return true; + } + // Validate against the UDF/closure var passed = arguments.validationData( - isNull( arguments.targetValue ) ? javacast( "null", "" ) : arguments.targetValue, + arguments.targetValue, arguments.target, errorMetadata ); diff --git a/test-harness/tests/specs/ValidationIntegrations.cfc b/test-harness/tests/specs/ValidationIntegrations.cfc index efefbe0..82062f0 100644 --- a/test-harness/tests/specs/ValidationIntegrations.cfc +++ b/test-harness/tests/specs/ValidationIntegrations.cfc @@ -159,7 +159,8 @@ component extends="coldbox.system.testing.BaseTestCase" appMapping="/root" { params = { username : "luis", email : "lmajano@ortussolutions.com", - password : "luis" + password : "luis", + status : 4 // should not validate }, method = "post" ); diff --git a/test-harness/tests/specs/validators/UDFValidatorTest.cfc b/test-harness/tests/specs/validators/UDFValidatorTest.cfc index c9172f1..bf7ced5 100644 --- a/test-harness/tests/specs/validators/UDFValidatorTest.cfc +++ b/test-harness/tests/specs/validators/UDFValidatorTest.cfc @@ -34,7 +34,7 @@ component extends="coldbox.system.testing.BaseModelTest" model="cbvalidation.mod javacast( "null", "" ), variables.validate3 ); - assertEquals( false, r ); + assertEquals( true, r ); } private function validate( value, target ){ From 66fc337483f463768a346515a9f840ae73fbd34d Mon Sep 17 00:00:00 2001 From: lmajano Date: Thu, 14 Sep 2023 10:08:05 +0000 Subject: [PATCH 5/6] Apply cfformat changes --- models/validators/UDFValidator.cfc | 2 +- test-harness/tests/specs/ValidationIntegrations.cfc | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/models/validators/UDFValidator.cfc b/models/validators/UDFValidator.cfc index 972a3ae..50b0619 100644 --- a/models/validators/UDFValidator.cfc +++ b/models/validators/UDFValidator.cfc @@ -34,7 +34,7 @@ component extends="BaseValidator" accessors="true" singleton { ){ var errorMetadata = {}; - // return true if no data to check, type needs a data element to be checked. + // return true if no data to check, type needs a data element to be checked. if ( isNull( arguments.targetValue ) || isNullOrEmpty( arguments.targetValue ) ) { return true; } diff --git a/test-harness/tests/specs/ValidationIntegrations.cfc b/test-harness/tests/specs/ValidationIntegrations.cfc index 82062f0..bb4274f 100644 --- a/test-harness/tests/specs/ValidationIntegrations.cfc +++ b/test-harness/tests/specs/ValidationIntegrations.cfc @@ -160,7 +160,7 @@ component extends="coldbox.system.testing.BaseTestCase" appMapping="/root" { username : "luis", email : "lmajano@ortussolutions.com", password : "luis", - status : 4 // should not validate + status : 4 // should not validate }, method = "post" ); From a002980eefa81c7db0dad738aac7293ef29c0128 Mon Sep 17 00:00:00 2001 From: Luis Majano Date: Mon, 16 Oct 2023 10:37:05 +0200 Subject: [PATCH 6/6] Actualizar changelog.md --- changelog.md | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/changelog.md b/changelog.md index 827a369..bfa47b2 100644 --- a/changelog.md +++ b/changelog.md @@ -9,6 +9,14 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [Unreleased] +### Added + +- requiredIf accepts a UDF and closure now + +### Fixed + +- UDF validator now treats nulls correctly + ## [4.3.1] - 2023-06-15 ### Fixed