mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-02-03 01:25:38 -05:00
New Validation Module for Resource-Deployment extension. (#12953)
This commit is contained in:
@@ -0,0 +1,223 @@
|
||||
/*---------------------------------------------------------------------------------------------
|
||||
* Copyright (c) Microsoft Corporation. All rights reserved.
|
||||
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
||||
*--------------------------------------------------------------------------------------------*/
|
||||
|
||||
import * as azdata from 'azdata';
|
||||
import 'mocha';
|
||||
import * as should from 'should';
|
||||
import * as sinon from 'sinon';
|
||||
import { createValidation, GreaterThanOrEqualsValidation, IntegerValidation, LessThanOrEqualsValidation, RegexValidation, validateInputBoxComponent, Validation, ValidationType, ValidationValueType } from '../../../ui/validation/validations';
|
||||
|
||||
const inputBox = <azdata.InputBoxComponent>{
|
||||
updateProperty(key: string, value: any) {}
|
||||
};
|
||||
let inputBoxStub: sinon.SinonStub;
|
||||
const testValidations = [
|
||||
{
|
||||
type: ValidationType.IsInteger,
|
||||
description: 'field was not an integer'
|
||||
},
|
||||
{
|
||||
type: ValidationType.Regex,
|
||||
description: 'field must contain only alphabetic characters',
|
||||
regex: '^[a-z]+$'
|
||||
},
|
||||
{
|
||||
type: ValidationType.LessThanOrEqualsTo,
|
||||
description: 'field value must be <= field2\'s value',
|
||||
target: 'field2'
|
||||
},
|
||||
{
|
||||
type: ValidationType.GreaterThanOrEqualsTo,
|
||||
description: 'field value must be >= field1\'s value',
|
||||
target: 'field1'
|
||||
}
|
||||
];
|
||||
|
||||
suite('Validation', () => {
|
||||
suite('createValidation and validate input Box', () => {
|
||||
setup(() => {
|
||||
sinon.restore(); //cleanup all previously defined sinon mocks
|
||||
inputBoxStub = sinon.stub(inputBox, 'updateProperty' ).resolves();
|
||||
});
|
||||
testValidations.forEach(testObj => {
|
||||
test(`validationType: ${testObj.type}`, async () => {
|
||||
const validation = createValidation(testObj, async () => undefined, async (_varName: string) => undefined);
|
||||
switch(testObj.type) {
|
||||
case ValidationType.IsInteger: should(validation).be.instanceOf(IntegerValidation); break;
|
||||
case ValidationType.Regex: should(validation).be.instanceOf(RegexValidation); break;
|
||||
case ValidationType.LessThanOrEqualsTo: should(validation).be.instanceOf(LessThanOrEqualsValidation); break;
|
||||
case ValidationType.GreaterThanOrEqualsTo: should(validation).be.instanceOf(GreaterThanOrEqualsValidation); break;
|
||||
default: console.log(`unexpected validation type: ${testObj.type}`); break;
|
||||
}
|
||||
should(await validateInputBoxComponent(inputBox, [validation])).be.false();
|
||||
should(inputBoxStub.calledOnce).be.true();
|
||||
should(inputBoxStub.getCall(0).args[0]).equal('validationErrorMessage');
|
||||
should(inputBoxStub.getCall(0).args[1]).equal(testObj.description);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
suite('IntegerValidation', () => {
|
||||
// all the below test values are arbitrary representative values or sentinel values for integer validation
|
||||
[
|
||||
{ value: '342520596781', expected: true },
|
||||
{ value: 342520596781, expected: true },
|
||||
{ value: '3.14', expected: false },
|
||||
{ value: 3.14, expected: false },
|
||||
{ value: '3.14e2', expected: true },
|
||||
{ value: 3.14e2, expected: true },
|
||||
{ value: undefined, expected: false },
|
||||
{ value: NaN, expected: false },
|
||||
].forEach((testObj) => {
|
||||
const displayTestValue = getDisplayString(testObj.value);
|
||||
test(`testValue:${displayTestValue}`, async () => {
|
||||
const validationDescription = `value: ${displayTestValue} was not an integer`;
|
||||
const validation = new IntegerValidation(
|
||||
{ type: ValidationType.IsInteger, description: validationDescription },
|
||||
async () => testObj.value
|
||||
);
|
||||
await testValidation(validation, testObj, validationDescription);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
suite('RegexValidation', () => {
|
||||
const testRegex = '^[0-9]+$';
|
||||
// tests
|
||||
[
|
||||
{ value: '3425205616179816', expected: true },
|
||||
{ value: 3425205616179816, expected: true },
|
||||
{ value: '3.14', expected: false },
|
||||
{ value: 3.14, expected: false },
|
||||
{ value: '3.14e2', expected: false },
|
||||
{ value: 3.14e2, expected: true }, // value of 3.14e2 literal is 342 which in string matches the testRegex
|
||||
{ value: 'arbitraryString', expected: false },
|
||||
{ value: undefined, expected: false },
|
||||
].forEach(testOb => {
|
||||
const displayTestValue = getDisplayString(testOb.value);
|
||||
test(`regex: /${testRegex}/, testValue:${displayTestValue}, expect result: ${testOb.expected}`, async () => {
|
||||
const validationDescription = `value:${displayTestValue} did not match the regex:/${testRegex}/`;
|
||||
const validation = new RegexValidation(
|
||||
{ type: ValidationType.IsInteger, description: validationDescription, regex: testRegex },
|
||||
async () => testOb.value
|
||||
);
|
||||
await testValidation(validation, testOb, validationDescription);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
suite('LessThanOrEqualsValidation', () => {
|
||||
const targetVariableName = 'comparisonTarget';
|
||||
// tests - when operands are mix of string and number then number comparison is performed
|
||||
[
|
||||
// integer values
|
||||
{ value: '342', targetValue: '42', expected: true },
|
||||
|
||||
{ value: 342, targetValue: '42', expected: false },
|
||||
{ value: '342', targetValue: 42, expected: false },
|
||||
|
||||
{ value: 42, targetValue: '342', expected: true },
|
||||
{ value: '42', targetValue: 342, expected: true },
|
||||
{ value: 42, targetValue: '42', expected: true },
|
||||
|
||||
{ value: 342, targetValue: 42, expected: false },
|
||||
|
||||
// floating pt values
|
||||
{ value: '342.15e-1', targetValue: '42.15e-1', expected: true },
|
||||
{ value: 342.15e-1, targetValue: '42.15e-1', expected: false },
|
||||
{ value: '342.15e-1', targetValue: 42.15e-1, expected: false },
|
||||
{ value: 342.15e-1, targetValue: 42.15e-1, expected: false },
|
||||
|
||||
// equal values
|
||||
{ value: '342.15', targetValue: '342.15', expected: true },
|
||||
{ value: 342.15, targetValue: '342.15', expected: true },
|
||||
{ value: '342.15', targetValue: 342.15, expected: true },
|
||||
{ value: 342.15, targetValue: 342.15, expected: true },
|
||||
|
||||
|
||||
// undefined values - if one operand is undefined result is always false
|
||||
{ value: undefined, targetValue: '42', expected: false },
|
||||
{ value: undefined, targetValue: 42, expected: false },
|
||||
{ value: '42', targetValue: undefined, expected: false },
|
||||
{ value: 42, targetValue: undefined, expected: false },
|
||||
{ value: undefined, targetValue: undefined, expected: false },
|
||||
|
||||
].forEach(testObj => {
|
||||
const displayTestValue = getDisplayString(testObj.value);
|
||||
const displayTargetValue = getDisplayString(testObj.targetValue);
|
||||
test(`testValue:${displayTestValue}, targetValue:${displayTargetValue}`, async () => {
|
||||
const validationDescription = `${displayTestValue} did not test as <= ${displayTargetValue}`;
|
||||
const validation = new LessThanOrEqualsValidation(
|
||||
{ type: ValidationType.IsInteger, description: validationDescription, target: targetVariableName },
|
||||
async () => testObj.value,
|
||||
async (_variableName: string) => testObj.targetValue
|
||||
);
|
||||
await testValidation(validation, testObj, validationDescription);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
suite('GreaterThanOrEqualsValidation', () => {
|
||||
const targetVariableName = 'comparisonTarget';
|
||||
// tests - when operands are mix of string and number then number comparison is performed
|
||||
[
|
||||
// integer values
|
||||
{ value: '342', targetValue: '42', expected: false },
|
||||
{ value: 342, targetValue: '42', expected: true },
|
||||
{ value: '342', targetValue: 42, expected: true },
|
||||
{ value: 342, targetValue: 42, expected: true },
|
||||
|
||||
// floating pt values
|
||||
{ value: '342.15e-1', targetValue: '42.15e-1', expected: false },
|
||||
{ value: 342.15e-1, targetValue: '42.15e-1', expected: true },
|
||||
{ value: '342.15e-1', targetValue: 42.15e-1, expected: true },
|
||||
{ value: 342.15e-1, targetValue: 42.15e-1, expected: true },
|
||||
|
||||
// equal values
|
||||
{ value: '342.15', targetValue: '342.15', expected: true },
|
||||
{ value: 342.15, targetValue: '342.15', expected: true },
|
||||
{ value: '342.15', targetValue: 342.15, expected: true },
|
||||
{ value: 342.15, targetValue: 342.15, expected: true },
|
||||
|
||||
// undefined values - if one operand is undefined result is always false
|
||||
{ value: undefined, targetValue: '42', expected: false },
|
||||
{ value: undefined, targetValue: 42, expected: false },
|
||||
{ value: '42', targetValue: undefined, expected: false },
|
||||
{ value: 42, targetValue: undefined, expected: false },
|
||||
{ value: undefined, targetValue: undefined, expected: false },
|
||||
].forEach(testObj => {
|
||||
const displayTestValue = getDisplayString(testObj.value);
|
||||
const displayTargetValue = getDisplayString(testObj.targetValue);
|
||||
test(`testValue:${displayTestValue}, targetValue:${displayTargetValue}`, async () => {
|
||||
const validationDescription = `${displayTestValue} did not test as >= ${displayTargetValue}`;
|
||||
const validation = new GreaterThanOrEqualsValidation(
|
||||
{ type: ValidationType.IsInteger, description: validationDescription, target: targetVariableName },
|
||||
async () => testObj.value,
|
||||
async (_variableName: string) => testObj.targetValue
|
||||
);
|
||||
await testValidation(validation, testObj, validationDescription);
|
||||
});
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
interface TestObject {
|
||||
value: ValidationValueType;
|
||||
targetValue?: ValidationValueType;
|
||||
expected: boolean;
|
||||
}
|
||||
|
||||
async function testValidation(validation: Validation, test: TestObject, validationDescription: string) {
|
||||
const validationResult = await validation.validate();
|
||||
should(validationResult.valid).be.equal(test.expected, validationDescription);
|
||||
validationResult.valid
|
||||
? should(validationResult.message).be.undefined()
|
||||
: should(validationResult.message).be.equal(validationDescription);
|
||||
}
|
||||
|
||||
function getDisplayString(value: ValidationValueType) {
|
||||
return typeof value === 'string' ? `"${value}"` : value;
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user