Skip to content

Default coercion and validation is... inconsistent #3459

@eapache

Description

@eapache

@rmosolgo another spin out of #3448. Filing as an issue not a PR because this might have implications across many classes, and probably needs some discussion.

The built-in scalars are confusingly inconsistent in what they will coerce for output, and how they produce errors.

  • The default Int scalar uses to_i when coercing outputs and then calls schema.type_error if the resulting int is out of bounds (GraphQL requires ints to be in the 32-bit signed range). In ruby "non_numeric_string".to_i produces 0, not an error.
  • BigInt and Float behave similarly, though Float is not bounded (which I believe it should be?)
  • Boolean is even more liberal, it just runs !! which as far as I know will convert any ruby object of any type whatsoever into a boolean.
  • ID, similarly, just calls to_s which is a method that exists on every ruby object (even nil.to_s returns "").
  • ISO8601Date uses Date.parse which I believe raises errors on invalid arguments?
  • ISO8601DateTime uses Time.parse which raises on invalid arguments, but then rescues all exceptions and converts them into GraphQL::Errors.
  • JSON is the best yet, it does a complete literal pass-through:
    def self.coerce_result(value, _context)
    value
    end
  • String calls to_s but then validates the string is in UTF-8 (mutating the passed-in string if it isn't frozen!?) and reports encoding errors using the schema.type_error callback.

All of these are inconsistent with each other, but also with scalar coercion on input where AFAICT the best practice is for your coercion method to return nil on invalid input?

Questions

  1. Should outputs be validated at all or are these output coercion methods just conveniences and we should trust the developer?
  2. Should input (and maybe output) validation be tied up with coercion like this or should validation and coercion be separate methods?
  3. What is the right way to signal an error in these cases? Raising? Calling schema.type_error? Returning nil?

I think whatever our answers to the above, some of the default scalars will need changing.

cc @benjie @swalkinshaw

Metadata

Metadata

Assignees

No one assigned

    Labels

    No labels
    No labels

    Projects

    No projects

    Milestone

    No milestone

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions