Skip to content

rez-go/constraints

Repository files navigation

[Data] Constraints

An experimental package for defining data constraints in Go. Not the same as type constrains.

Note that this project is in experimental stage. The interfaces are not guaranteed to stay as they are now. Everything is subject to change.

Background

When attempting to restructure errors in my project, I realized that some of the errors could be classified as data errors.

When we are talking about data-related errors, most instances are coming out from data validation. An error which coming from a validation process should provide enough information about why the data is considered as invalid, i.e., the rules or constraints it violated.

For example, we define TooShortError. We expect it to contains details about how is the minimum before a value is considered invalid. So, the error is actually containing information about the constraint. Rather than defining the errors, we could just make a generic error, something like DataValidationError, and make it holding the information about the constraint(s) a data is violating. Let the presentation layer translate it based on the consumer.

We want the constraints to be structured so that it could be nicely passed as understandable data to other systems, including to the end-user presentation layer, which could be translated to any human language.

Goals

  • Readable constraints

    We must design the module and packages so that the struct, function and variable names are easy to understand.

  • Focus on simple rule for each constraint

    Avoid defining constraints which could be constructed from simple constraints. For example we might see that we should be able to define value range constraints, but these could be constructed from Maximum and Minimum.

  • Easy to programatically dig deep into a constraint

    Use types.

  • Easy to generate rules for other systems

    E.g., JSON schema.

Design

Let's start from how we want to use this package.

First, we declare the individual constraints. We should be able to declare the constraints inline, but by assigning them to variables, we can refer them when we are trying to figure out which constraint(s) coming out of a validation.

var (
  usernameMinLength = StringMinLength(6)
  usernameMaxLength = StringMaxLength(32)
  // All these constraints in this example could be declared as a
  // single regular expression pattern, but we are trying to design
  // a mechanism which is more readable, constructed of smaller, clear
  // rules rather than putting all the rules into a complex pattern.
  //
  // We might want to find a way to declare this kind of constraint,
  // but we will limit how far we will go before we reinvent regular
  // expression.
  usernameAllowedCharacters = Func(
    `allowed characters are A to Z, 0 to 9 and underscore`,
    regexp.MustCompile(`^[A-Za-z0-9_]+$`).MatchString)
  // Name the variables based on the semantic rather than the
  // description of the constraint (what a constraint is for, rather than
  // what the constraint does). For the description of the constraint,
  // we put it into the constraint instance itself.
  //
  // The variable name and the description are good if they sound good if
  // we merge them:
  // "username [starts with] a letter".
  usernameStartsWith = Func(
    `starts with a letter`,
    func(v string) bool {
      if v != "" {
        r, _ := utf8.DecodeRuneInString(v)
        return (r >= 'a' && r <= 'z') || (r >= 'A' && r <= 'Z')
      }
      return false
    })
  usernameEndsWith = Negate(
    StringSuffix("_"),
    `ends with anything but underscore`)
  usernameNoConsecutiveUnderscore = StringNoConsecutiveRune('_')
)

Next we define the constraint set:

var usernameConstraints = Set(
    usernameMinLength,
    usernameMaxLength,
    usernameAllowedCharacters,
    usernameStartsWith,
    usernameEndsWith,
    usernameNoConsecutiveUnderscore,
)

We could generate a decent instruction from the constraint:

fmt.Printf("username: %s\n", usernameConstraints.ConstraintDescription())

Which would print something like username: min length 6, max length 32, allowed characters are .... Ideally, a constraint is mapped to some well thought messages if it's going to be displayed to human.

Then we can use the set:

violatedConstraints := usernameConstraints.ValidateAll(`h-llo`)
for _, vc := range violatedConstraints {
    switch vc {
    case usernameMinLength:
        // this case must be hit.
    case usernameAllowedCharacters:
        // this case must also be hit.
    }
}

The violatedConstraints contains all the constraints the data violates. This could be a good case for end user as they could fix their input in one go rather than back-and-forth.

Or as error in a request handler:

err := ValidOrError(username, usernameConstraints)
if err != nil {
    return ArgumentError("username", err)
}

The error contains a structured information which describes the violations so that we can encode it and put it into response.

Ideally, we'd like to make it easy to generate validation directive for other systems, e.g., JSON Schema.

//...

jsonSchemaField := JSONSchemaField("username", usernameConstraints)

//...

Another examples

func doSomethingToSelf(ctx context.Context, userID iam.UserID) error {
	if err := ContextConstraintSet(
		ContextNotNil,
		ContextUserOnly,
		ContextUserMatches(userID),
	).ValidOrError(ctx); err != nil {
		return err
	}

	// ...
}

Hacking

  • We are still in designing stage. Anything could be suggested to be changed. We want some discussions.
  • We want to make this to be practical and relevant to real-world usages. Any examples of real-world usage will be much appreciated.
  • We won't include rules which don't have strict, static constraints. We might won't even include rules for standard formats. We will never include validations for formats like phone numbers, email addresses, internet domains and postal addresses. We should provide enough facilities to others to provide module for certain type of validation.
  • We might want to limit this module to contain only the contracts and limited utilites, and put common constraints into their own module (discuss!).
  • Generating constraints for other systems, e.g., JSON schema, must be done in external module. We haven't actually designed this module the for this kind of usage. Built in rules must not know about other systems but they must provide enough information for other modules to generate the the rules in their own format from built-in rules.

Technical Constraints

We limit the dependencies only to Go's stdlib.

References

We want to align this package with existing standards.

About

No description, website, or topics provided.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published