validated-types
is a library for Typescript to create types for validated integer, float, string and array values.
It also allows of creating semantic types and declare variables with semantic types.
- Prerequisites
- Installation
- Usage
- API Documentation
- Feedback
- License
- Typescript >= 4.1.0
npm install --save validated-types
Validated-types is a type value validation library for Typescript. You can validate values of integers, floats, strings and arrays. For numbers, you can for example validate the minimum and maximum value. For strings, you can validate the length and also perform semantic validation to validate if a string should be, for example, a valid URL, IP address or email address.
By using validated types for the parameters of your application's functions, you can rest assured that only proper values are ever passed to your function. You don't have to do any validation work inside your functions, just use the value of already validated parameter. The validation of the value is done only once upon the construction of validated objects, so there is no performance penalty in using the validated values as many times as you need.
You should create the validated integer, float, string and array objects in functions that receive unvalidated input data and then pass the validated values to the rest of the functions in your application.
Your application typically receives unvalidated input data from external sources in following scenarios:
- Reading command line arguments
- Reading environment variables
- Reading standard input
- Reading file(s) from file system
- Reading data from socket (network input)
- End-user input from user interface
You can also create semantic types and semantically typed variables using SemType
. Using semantic types, you
can differentiate between multiple types of the same basic type. For example, If a function accepts two boolean arguments, it is possible
that the calling function gives those two arguments in wrong order, but you never notice it, because it does not generate a compilation error.
Using semantic types, you can differentiate between those two boolean types by giving them two different semantic names.
Semantic name is just any string describing the purpose of type/variable.
The library contains static factory methods in the classes forr creating a new instance of the specific class. There are different factory methods available. You can choose which factory method to use depending on your application need. The factory methods are following:
tryCreate(...)
creates a validated value object or throws an exceptioncreateOrThrow(...)
is same as above, i.e. creates a validated value object or throws an exceptioncreate(...)
creates a validated value object or returnsnull
. Use this method if you don't need the error reasoncreateOrError(...)
creates a validated value object or returns an error object, always returns a pair (is similar to Golang returning an error)
You can validate your integer, float and string type variables with validated-types
.
For example, to create a validated integer which allows values between 1 and 10, you declare:
import { VInt } from 'validated-types';
function useInt(int: VInt<'1,10'>) {
// use int here
console.log(int.value);
}
const int = VInt.tryCreate<'1,10'>('1,10', 5);
const maybeInt = VInt.create<'1,10'>('1,10', 12); // Returns null
useInt(int); // prints to console: 5
useInt(maybeInt ?? VInt.tryCreate('1,10', 10)); // prints to console: 10
You can also register a custom validator:
import { VInt } from 'validated-types';
VInt.registerCustomValidator('isEven', (value) => value % 2 === 0);
const evenNumber = VInt.tryCreate<'custom:isEven'>('custom:isEven', 2);
const maybeEvenNumber = VInt<'custom:isEven'>.create('custom:isEven', 1);
Custom validator must be registered before it is used by any validated types create
, createOrThrow
or createOrError
method.
This means that you should register your custom validators as early as possible in your application code.
If custom validator is not registered, and it is used, an exception will be thrown.
To create a validated float which allows positive values only, you declare:
import { VFloat } from 'validated-types';
function useFloat(float: VFloat<'positive'>) {
// use float here
console.log(float.value);
}
const float = VFloat.tryCreate<'positive'>('positive', 5.25);
const maybeFloat = VFloat<'positive'>.create('positive', -5.25); // returns null
useFloat(float); // prints to console: 5.25
useFloat(maybeFloat ?? VFloat.tryCreate('positive', 1)); // prints to console: 1
To create a validated string which allows URLs with minimum length of one and maximum length of 1024 characters, you declare:
import { VString } from 'validated-types';
function useUrl(url: VString<'1,1024,url'>) {
// use URL here
console.log(url.value);
}
const url = VString.tryCreate<'1,1024,url'>('1,1024,url', 'https://www.mydomain.com');
const maybeUrl = VString<'1,1024,url'>.create('1,1024,url', 'invalid URL'); // Returns null
useUrl(url); // prints to console: https://www.mydomain.com
useUrl(maybeUrl ?? VString.tryCreate('1,1024,url', 'https://google.com')); // prints to console: https://google.com
You can combine up to 5 different string validators together. Multiple validators are given as a tuple which can have 2-5 elements. The first element of the tuple should validate the length of the string, if needed. Rest of the elements in tuple should not validate the length of string anymore. Below example contains 4 validators that validate following string:
- is at least 1 characters long
- is at most 1024 characters long
- is lower case string
- is valid URL
- URL starts with https
- URL ends with .html
import { SpecOf, VString } from 'validated-types';
type Url = VString<['1,1024,lowercase', 'url', 'startsWith,https', 'endsWith,.html']>;
const urlVSpec: VSpecOf<Url> = ['1,1024,lowercase', 'url', 'startsWith,https', 'endsWith,.html'];
function useUrl(url: Url) {
// use URL here
console.log(url.value);
}
const url: Url = VString.tryCreate(urlVSpec, 'https://server.domain.com:8080/index.html');
const maybeUrl: Url | null = VString.create(urlVSpec, 'invalid URL'); // Returns null
useUrl(url); // prints to console: https://server.domain.com:8080/index.html
useUrl(maybeUrl ?? VString.tryCreate(urlVSpec, 'https://server.domain.com:8080/index.html')); // prints to console: https://server.domain.com:8080/index.html
To create a validated array of numbers which allows array length to be from 0 to 10 and requires all array elements to be unique, you declare:
import { VArray } from 'validated-types';
function useArray(array: VArray<'0,10,unique', number>) {
// use array here
console.log(array.value);
}
const array = VArray.tryCreate<'0,10,unique', number>('0,10,unique', [1, 2, 3]);
const maybeArray = VArray<'0,10,unique', number>.create('0,10,unique', [1, 2, 2]);
useArray(array); // prints to console: [1, 2, 3]
useArray(maybeArray ?? VArray.tryCreate('0,10,unique', [3, 4, 5])); // prints to console: [3, 4, 5]
You can also create an array of validated objects, for example below example validates an array of from 1 to max 10 unique email addresses:
import { VArray } from 'validated-types';
type EmailAddresses = VArray<'1,10,unique', VString<'email'>>;
const emailAddressesVSpec: VSpecOf<EmailAddresses> = '1,10,unique';
function sendEmails(emailAddresses: EmailAddresses, subject: string, body: string) {
console.log(emailAddresses.forEach(emailAddress=> emailAddress.value));
}
const emailAddress = VString.tryCreate<'email'>('email', '[email protected]');
const emailAddress2 = VString.tryCreate<'email'>('email', '[email protected]');
const emailAddresses: EmailAddresses = VArray.tryCreate(emailAddressesVSpec, [emailAddress, emailAddress2]);
sendEmails(emailAddresses, 'subj', 'body'); // prints to console: '[email protected]' and '[email protected]'
You can also create validated objects, for example:
type Person = {
firstName: VString<1, 64>;
lastName: VString<1, 64>;
nickNames: VString<1, 64>[];
email: VString<'email'>;
password: VString<'8,1024,strongPassword'>;
age: VInt<0, 255>;
interestPercent: VFloat<'0,100'>;
};
You can assign type aliases to your validated types:
types.ts
import { VFloat, VInt, VString } from 'validated-types';
export type Name = VString<1, 64>;
export type Email = VString<'email'>;
export type Password = VString<'8,1024,strongPassword'>;
export type Age = VInt<0, 255>;
export type Percent = VFloat<'0,100'>;
person.ts
import { Age, Email, Name, Password, Percent } from 'types';
type Person = {
firstName: Name;
lastName: Name;
nickNames: Name[];
email: Email;
password: Password;
age: Age;
interestPercent: Percent;
};
Semantic types and variables let you differentiate between variables of same type. The differentiation is done
by assigning a semantic name to the type. Following example declares two semantic variables of boolean
type with two different semantic names.
import { SemType } from 'validated-types';
type IsRecursiveCall = SemType<boolean, 'isRecursiveCall'>
type IsInternalCall = SemType<boolean, 'isInternalCall'>;
function myFunc(isRecursiveCall: IsRecursiveCall, isInternalCall: IsInternalCall) {
console.log(isRecursiveCall.value);
console.log(isInternalCall.value);
}
const isRecursiveCall = false;
const isInternalCall = true;
// Only this will succeed
myFunc(new SemType({ isRecursiveCall }), new SemType({ isInternalCall }));
// These will fail during compilation
myFunc(new SemType({ isInternalCall }), new SemType({ isRecursiveCall }));
myFunc(true, true);
myFunc(new SemType('isSomethingElse', true), new SemType('isInternalCall', true));
myFunc(new SemType('isRecursiveCall', false), new SemType('isSomethingElse', true));
myFunc(new SemType('isSomethingElse', true), new SemType('isSomethingElse', true));
You can use a validated type as a semantic type also. In the below example, a function takes two semantic parameters
which both have the same base type VString<'1,64'>
.
import { SemType, VString } from 'validated-types';
type Name = VString<'1,64'>;
const nameVSpec: VSpecOf<Name> = '1,64';
type FirstName = SemType<Name, 'firstName'>;
type LastName = SemType<Name, 'lastName'>;
function myFunc(firstName: FirstName, lastName: LastName) {
console.log(firstName.value);
console.log(lastName.value);
}
const firstName: Name = VString.tryCreate(nameVSpec, 'John');
const lastName: Name = VString.tryCreate(nameVSpec, 'Doe');
type Name2 = VString<'1,65'>
const name2VSpec: VSpecOf<Name> = '1,64';
const firstName2: Name2 = VString.tryCreate(name2VSpec, 'John');
const lastName2: Name2 = VString.tryCreate(name2VSpec, 'Doe');
// Only this will succeed where 'firstName' and 'lastName' are given in correct order with correct validated type
myFunc(new SemType({ firstName }), new SemType({ lastName }));
// These below will fail during compilation
myFunc(new SemType({ lastName }), new SemType({ firstName }));
myFunc(new SemType({ firstName: firstName2 }), new SemType({ lastName: lastName2 }));
myFunc('John', 'Doe');
myFunc(firstName, lastName);
import { SemVInt } from 'validated-types';
type HttpPort = SemVInt<'httpPort', '1,65535'>;
const httpPort: HttpPort = SemVInt.tryCreate('httpPort', '1,65535', 8080);
console.log(httpPort.value); // Prints 8080
import { SemVFloat } from 'validated-types';
type LoanInterest = SemVFloat<'loanInterest', '0,100'>;
const loanInterest: LoanInterest = SemVFloat.tryCreate('loanInterest', '0,100', 4.99);
console.log(loanInterest.value); // Prints 4.99
import { SemVString } from 'validated-types';
type LoginUrl = SemVString<'loginUrl', '1,8192,url'>;
const loginUrl: LoginUrl = SemVString.tryCreate('loginUrl', '1,8192,url', "https://server.com");
console.log(loginUrl.value) // Prints https://server.com
class VFloat<VSpec extends string> {
static registerCustomValidator(validatorName: string, validateFunc: (value: number) => boolean): void;
static createOrThrow(validationSpec: string, value: number, varName?: string): VFloat<VSpec> | never;
static tryCreate(validationSpec: string, value: number, varName?: string): VFloat<VSpec> | never;
static create(validationSpec: string, value: number, varName?: string): VFloat<VSpec> | null;
static createOrError(
validationSpec: string,
value: number,
varName?: string
): [VFloat<VSpec>, null] | [null, Error];
get value(): number;
}
static registerCustomValidator(validatorName: string, validateFunc: (value: number) => boolean): void
Registers a custom validator with given name. validateFunc
receives a float value as parameter and returns boolean based on success of validation.
You must register your custom validator before using it in any of the create
functions.
static createOrThrow(validationSpec: string, value: number, varName?: string): VFloat<VSpec> | never
static tryCreate(validationSpec: string, value: number, varName?: string): VFloat<VSpec> | never
Creates a new validated float value object or throws a ValidationError
exception if supplied value is invalid.
validationSpec
is a string of following form '[<minValue>],[<maxValue>] | negative | positive | custom:<custom_validator_name>'
.
validationSpec
examples:
- '0,100'
- '100,'
- ',0'
- 'positive'
- 'negative'
- 'custom:isUsShoeSize'
If minValue
is missing, Number.MIN_VALUE
is used.
If maxValue
is missing, Number.MAX_VALUE
is used.
If varName
is supplied, it is mentioned in possible ValidationError
thrown.
static create(validationSpec: string, value: number, varName?: string): VFloat<VSpec> | null
Same as VFloat.createOrThrow
, but instead of throwing on validation failure, it returns null
.
This method is useful if you don't care about the error message, but just want to know if validation succeeded or not.
static createOrError(validationSpec: string, value: number, varName?: string): [VFloat<VSpec>, null] | [null, Error]
Same as VFloat.createOrThrow
, but instead of throwing on validation failure, it returns a tuple [VFloat, null]
on success and tuple [null, Error]
on validation failure.
This method is useful for Go language style of programming where you get a 2-tuple return value where the last element in tuple contains the possible error.
For example:
const [float, err] = VFloat.createOrError<'1,10'>('1,10', 1);
if (err) {
// handle error here
}
get value(): number
Returns the valid float value.
class VInt<VSpec extends string> {
static registerCustomValidator(validatorName: string, validateFunc: (value: number) => boolean): void;
static createOrThrow(validationSpec: string, value: number, varName?: string): VInt<VSpec> | never;
static tryCreate(validationSpec: string, value: number, varName?: string): VInt<VSpec> | never;
static create(validationSpec: string, value: number, varName?: string): VInt<VSpec> | null;
static createOrError(
validationSpec: string,
value: number,
varName?: string
): [VInt<VSpec>, null] | [null, Error];
get value(): number;
}
static registerCustomValidator(validatorName: string, validateFunc: (value: number) => boolean): void
Registers a custom validator with given name. validateFunc
receives a number as parameter and returns boolean based on success of validation.
You must register your custom validator before using it in any of the create
functions.
static createOrThrow(validationSpec: string, value: number, varName?: string): VInt<VSpec> | never
static tryCreate(validationSpec: string, value: number, varName?: string): VInt<VSpec> | never
Creates a new validated integer value object or throws a ValidationError
exception if supplied value is invalid.
validationSpec
is a string of following form '[<minValue>],[<maxValue>][,<divisibleByValue>] | negative | positive | custom:<custom_validator_name>'
.
validationSpec
examples:
- '0,100'
- '0,100,2'
- '100,'
- ',0'
- ','
- 'positive'
- 'negative'
- 'custom:isEven'
If minValue
is missing, Number.MIN_SAFE_INTEGER
is used.
If maxValue
is missing, Number.MAX_SAFE_INTEGER
is used.
If varName
is supplied, it is mentioned in possible ValidationError
thrown.
static create(validationSpec: string, value: number, varName?: string): VInt<VSpec> | null
Same as VInt.createOrThrow
, but instead of throwing on validation failure, it returns null
.
This method is useful if you don't care about the error message, but just want to know if validation succeeded or not.
static createOrError(validationSpec: string, value: number, varName?: string): [VInt<VSpec>, null] | [null, Error]
Same as VInt.createOrThrow
, but instead of throwing on validation failure, it returns a tuple [VInt, null]
on success and tuple [null, Error]
on validation failure
This method is useful for Go language style of programming where you get a 2-tuple return value where the last element in tuple contains the possible error.
For example:
const [int, err] = VInt.createOrError<'1,10'>('1,10', 1);
if (err) {
// handle error here
}
get value(): number
Returns the valid integer value.
class VString<VSpec extends string> {
static registerCustomValidator(validatorName: string | string[], validateFunc: (value: string) => boolean): void;
static createOrThrow(validationSpec: string | string[], value: number, varName?: string): VString<VSpec> | never;
static tryCreate(validationSpec: string | string[], value: number, varName?: string): VString<VSpec> | never;
static create(validationSpec: string | string[], value: number, varName?: string): VString<VSpec> | null;
static createOrError(
validationSpec: string,
value: number,
varName?: string
): [VString<VSpec>, null] | [null, Error];
get value(): string;
}
static registerCustomValidator(validatorName: string, validateFunc: (value: string) => boolean): void
Registers a custom validator with given name. validateFunc
receives a string value as parameter and returns boolean based on success of validation.
You must register your custom validator before using it in any of the create
functions.
static createOrThrow(validationSpec: string, value: number, varName?: string): VString<VSpec> | never
static tryCreate(validationSpec: string, value: number, varName?: string): VString<VSpec> | never
Creates a new validated integer value object or throws a ValidationError
exception if supplied value is invalid.
validationSpec
is a string of following form '[<minLength>],<maxLength>[,<unknown_length_validator_name>[,<parameter>]] | <known_length_validator_name> | custom:<custom_validator_name>'
.
Possible value for <unknown_length_validator_name>
:
- alpha
- alphanumeric
- ascii
- base32
- base58
- base64
- dataUri
- decimal
- fqdn
- md4
- md5
- sha1
- sha256
- sha384
- sha512
- crc32
- crc32b
- hex
- ipv4Range
- ipv6Range
- json
- lowercase
- magnetUri
- mongoId
- numeric
- octal
- uppercase
- strongPassword
- url
- includes (requires
parameter
) - match (requires a RegExp string
parameter
) - isOneOf (requires JSON string array
parameter
) - isNoneOf (requires JSON string array
parameter
) - startsWith (requires
parameter
) - endsWith (requires
parameter
) - numericRange (requires
parameter
in format:<minValue>-<maxValue>
, e.g. 1-65535)
Possible values for <known_length_validator_name>
:
- boolean
- bic
- btcAddress
- creditCard
- ean
- ethereumAddress
- hsl
- hexColor
- isin
- iban
- ipv4
- ipv6
- iso31661Alpha2
- iso31661Alpha3
- iso8601
- isrc
- issn
- jwt
- latLong
- macAddress
- mimeType
- port
- rgbColor
- semVer
- uuid
- postalCode
- creditCardExpiration
- cvc
- mobileNumber
More information about validators can be found in validator.js documentation If you need a new built-in validator, please open a new issue about that.
validationSpec
examples:
- '0,100'
- ',100'
- '1,1024,url'
- '1,1024,startsWith,https'
- 'email'
- 'custom:isSupplierName'
If minLength
is missing, 0 is used.
If varName
is supplied, it is mentioned in possible ValidationError
thrown.
static create(validationSpec: string, value: number, varName?: string): VString<VSpec> | null
Same as VString.createOrThrow
, but instead of throwing on validation failure, it returns null
.
This method is useful if you don't care about the error message, but just want to know if validation succeeded or not.
static createOrError(validationSpec: string, value: number, varName?: string): [VString<VSpec>, null] | [null, Error]
Same as VString.createOrThrow
, but instead of throwing on validation failure, it returns a tuple [VString, null]
on success and tuple [null, Error]
on validation failure
This method is useful for Go language style of programming where you get a 2-tuple return value where the last element in tuple contains the possible error.
For example:
const [str, err] = VString.createOrError<'1,10'>('1,10', 'abc');
if (err) {
// handle error here
}
get value(): string
Returns the valid string value.
T
is the type of elements in the array.
class VArray<VSpec extends string, T> {
static registerCustomValidator(validatorName: string, validateFunc: (value: T[]) => boolean): void;
static createOrThrow(validationSpec: string, value: T[], varName?: string): VArray<VSpec, T> | never;
static tryCreate(validationSpec: string, value: T[], varName?: string): VArray<VSpec, T> | never;
static create(validationSpec: string, value: T[], varName?: string): VArray<VSpec, T> | null;
static createOrError(
validationSpec: string,
value: T[],
varName?: string
): [VArray<VSpec, T>, null] | [null, Error];
get value(): T[];
}
static registerCustomValidator(validatorName: string, validateFunc: (value: T[]) => boolean): void
Registers a custom validator with given name. validateFunc
receives an array as parameter and returns boolean based on success of validation.
You must register your custom validator before using it in any of the create
functions.
static createOrThrow(validationSpec: string, value: T[], varName?: string): VArray<VSpec, T> | never
static tryCreate(validationSpec: string, value: T[], varName?: string): VArray<VSpec, T> | never
Creates a new validated array value object or throws a ValidationError
exception if supplied value is invalid.
validationSpec
is a string of following form '[<minLength>],<maxLength>[,unique] | custom:<custom_validator_name>'
.
validationSpec
examples:
- '0,100'
- ',100'
- '1,10,unique'
- 'custom:includesSomething'
If minLength
is missing, 0 is used.
If varName
is supplied, it is mentioned in possible ValidationError
thrown.
static create(validationSpec: string, value: T[], varName?: string): VArray<VSpec, T> | null
Same as VFloat.createOrThrow
, but instead of throwing on validation failure, it returns null
.
This method is useful if you don't care about the error message, but just want to know if validation succeeded or not.
static createOrError(validationSpec: string, value: T[], varName?: string): [VArray<VSpec, T>, null] | [null, Error]
Same as VFloat.createOrThrow
, but instead of throwing on validation failure, it returns a tuple [VArray, null]
on success and tuple [null, Error]
on validation failure.
This method is useful for Go language style of programming where you get a 2-tuple return value where the last element in tuple contains the possible error.
For example:
const [array, err] = VArray.createOrError<'1,10', number>('1,10', [1]);
if (err) {
// handle error here
}
get value(): T[]
Returns the validated array.
T
is the type of Semantic variable.
type SemName<N extends string> = N extends `${infer Name}` ? `${Name}` : never;
class SemType<T, N extends string> {
constructor(semanticName: SemName<N>, value: T);
constructor(semType: { [K in SemName<N>]: T });
get value(): T;
}
constructor(semanticName: SemName<N>, value: T)
constructor(semType: { [K in SemName<N>]: T });
Creates a new semantic variable with semantic name semanticName
and with value of type T
.
get value(): T
Get the value of semantic variable.
type SemName<N extends string> = N extends `${infer Name}` ? `${Name}` : never;
class SemVInt<Name extends string, VSpec extends string> {
static tryCreate<Name extends string, VSpec extends string>(
semanticName: SemName<Name>,
validationSpec: IntValidationSpec<VSpec>,
value: number,
varName?: string
): SemVInt<Name, VSpec> | never; // Throws on error
static create<Name extends string, VSpec extends string>(
semanticName: SemName<Name>,
validationSpec: IntValidationSpec<VSpec>,
value: number
): SemVInt<Name, VSpec> | null; // Returns null on error
static createOrError<Name extends string, VSpec extends string>(
semanticName: SemName<Name>,
validationSpec: IntValidationSpec<VSpec>,
value: number,
varName?: string
): [SemVInt<Name, VSpec>, null] | [null, Error] // Returns a pair [null, Error] on error
}
type SemName<N extends string> = N extends `${infer Name}` ? `${Name}` : never;
class SemVFloat<Name extends string, VSpec extends string> {
static tryCreate<Name extends string, VSpec extends string>(
semanticName: SemName<Name>,
validationSpec: FloatValidationSpec<VSpec>,
value: number,
varName?: string
): SemVFloat<Name, VSpec> | never; // Throws on error
static create<Name extends string, VSpec extends string>(
semanticName: SemName<Name>,
validationSpec: FloatValidationSpec<VSpec>,
value: number
): SemVFloat<Name, VSpec> | null // Returns null on error
static createOrError<Name extends string, VSpec extends string>(
semanticName: SemName<Name>,
validationSpec: FloatValidationSpec<VSpec>,
value: number,
varName?: string
): [SemVFloat<Name, VSpec>, null] | [null, Error]; // Returns a pair [null, Error] on error
}
type SemName<N extends string> = N extends `${infer Name}` ? `${Name}` : never;
class SemVString<Name extends string, VSpec extends string | string[]> {
static tryCreate<Name extends string, VSpec extends string | string[]>(
semanticName: SemName<Name>,
validationSpec: string | string[],
value: string,
varName?: string
): SemVString<Name, VSpec> | never; // Throws on error
static create<Name extends string, VSpec extends string>(
semanticName: SemName<Name>,
validationSpec: string | string[],
value: string,
varName?: string
): SemVString<Name, VSpec> | null; // Returns null on error
static createOrError<Name extends string, VSpec extends string>(
semanticName: SemName<Name>,
validationSpec: string | string[],
value: string,
varName?: string
): [SemVString<Name, VSpec>, null] | [null, Error]; // Returns a pair [null, Error] on error
}
Extracts the validation spec type of the validated type.
For example:
type Month = VInt<'1,12'>;
const monthVSpec: VSpecOf<Month> = '1,12';
const month: Month = VInt.tryCreate(monthVSpec, 1);
type Percent = VFloat<'0,100'>;
const percentVSpec: VSpecOf<Percent> = '0,100';
const percent: Percent = VFloat.tryCreate(percentVSpec, 25.0);
type Url = VString<['1,1024,lowercase', 'url', 'startsWith,https', 'endsWith,.html']>;
const urlVSpec: VSpecOf<Url> = ['1,1024,lowercase', 'url', 'startsWith,https', 'endsWith,.html'];
const url: Url = VString.tryCreate(urlVSpec, 'https://server.domain.com:8080/index.html');
If you want to report a bug, please create a new issue about that.
If you want to request a new feature, for example, a new type of validator for string, please create a new issue about that.