-
Notifications
You must be signed in to change notification settings - Fork 771
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Improve messages in an array format #1498
base: main
Are you sure you want to change the base?
Conversation
399be98
to
9b7b10c
Compare
cad83ce
to
259be0c
Compare
Codecov ReportAll modified and coverable lines are covered by tests ✅
Additional details and impacted files@@ Coverage Diff @@
## main #1498 +/- ##
=========================================
Coverage 96.53% 96.53%
- Complexity 972 973 +1
=========================================
Files 201 201
Lines 2396 2397 +1
=========================================
+ Hits 2313 2314 +1
Misses 83 83 ☔ View full report in Codecov by Sentry. |
b47c0b5
to
d719129
Compare
904ddb5
to
75a2315
Compare
75a2315
to
70c60ae
Compare
], | ||
18 => [ | ||
'messages' => ['allOf' => '`false` must pass all the rules'], | ||
'details' => ['arrayVal' => '`false` must be an array value'], |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I added this key because some rules have children but they're respective to the parent validation, not a child. I thought I would add an extra key there to identify those. I think that one would always use details
instead of messages
when it's available because messages
would usually be only headers of nesting validations.
'messages' => ['each' => 'Each item in `["perm1": true, "perm2": false, "perm3": "boom!"]` must be valid'], | ||
'children' => [ | ||
'perm3' => [ | ||
'messages' => ['boolVal' => '"boom!" must be a boolean value'], |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not replacing the input of each rule with the key, because I think that, since the structure is already in the array, it's best to keep the original message instead, so you can actually see the value that failed.
], | ||
], | ||
], | ||
] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I gotta say, we can added a whole bunch of other things here, I'm just not sure how they can be useful. Each element could contain the keys:
messages
: all the messages that failsdetails
: more details about the messages that failedchildren
: all the children (which in turn could have all the keys I'm mentioning hereinput
: the input that was proceeded by the rulepath
: the path of the rule that failed
But when I start to really think about this, I think this is not even an array in an exception anymore, it could be an object that contains all this information and handles all of them properly.
Maybe I'm just thinking of providing this array to the exception because of how the library worked before, but I could just drop the whole array thing, and return a proper object in Validator::validate()
that would contain all the failed messages and the extra information that is needed -- maybe I'm just conditioned to how the library worked in the past.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I can offer some perspective on what I was thinking when I made the original library.
The array thing and a lot of things (like setting properties in runtime objects) were implementation details, mostly private stuff, that could be done in whatever way fits best (I decided to go for the solution that had lass code back then).
To me, the reportError
(or was it reportException
?) method was part of the validatable interface that could be extended and the thing I wanted to look nice. Together with the base lib, it offered two standard ways of creating exceptions:
- ValidationException, which is a simple single message object.
- AbstractNestedException, which is a more elaborate object, mostly used by aggregate validators, but actually an object of its own.
AbstractNestedException had things like findMessages
with paths back then, although the workings were slightly different. Since each validator had to had a dedicated exception, I went for an abstract instead of a concrete named object like ValidationResults
.
The decision to use exceptions also played a role in the design of Validatable::check
, which I think doesn't exist anymore. It was designed to catch exceptions by type early instead of running the whole thing (the decision whether to fail fast or not). This complicated the internal workings of the project, and it was a compromise I'm not sure we still need to make. It seems our users don't care about fail fast that much.
So, you're still dealign with a lot of interlocked design decisions I made back then :D. You don't need to follow any of them if they don't make sense anymore. I would just go for whatever fullfills your vision best. That was what I did: I just moved things around until it was cohese.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for giving me some perspective, @alganet, also it's always nice to hear from you! That does give me more peace with changing things a bit more, although at this point, only the API of the previous Validator
remains, and not all of it. At times, I even wonder how you perceive it, since you created the library.
I think that failing fast is still important in some cases. We'll have Circuit in version 3.0 to fulfil that purpose, but users will need to be intentional about it.
One thing that I liked a lot about check()
was the ability to pick the top message, instead of the whole thing, and this will be similar in version 3.0.
Since you're here, I'd love to get your perspective on something else. When you created the library were the validation messages meant for engineers or end-users? That's something that I'm often conflicted about.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I thought of use cases for both engineers and end-users.
The findMessages
and getMessages
APIs were designed to work with HTML forms. It was designed so the engineer can pick, for each field, what he wants ("oh, for this I want detailed messages like the domain is wrong, the tld is wrong.... and for this one I just want a simple message").
Some of the features were designed to later work with APIs as well. That's why findMessages
had selectors. You could kind of use the same validator, but change the exception catching part to deliver different reports to different kinds of users. There was also some feng shui to adhere informally (duck typing) to the Respect\Rest interface (you could use a validator directly as a when() routine, which was also part of the design we don't exactly need anymore).
The getFullMessage
API that returned an ASCII tree was designed for CLI applications, print debugging, stuff like that. It gave a full usable text block as a message. I was expecting to see that in logs, tests, stdout from services.
So, in a sense, I did for both, but I always had some specific use cases in mind.
The way I perceive what you're changing is to move to a canonical, fully serializable "report". Mine was different, in my idea you could "query" this non-linear exception tree for a specific serialization. There are compromises in both, and I don't prefer any over the other, both are cool.
One thing that bothered me in my original design was that the validation tree and the result tree were sometimes not the same. I discovered too late that these trees were not equal. I never truly figured this part out, and the API grew from the assumption that these two trees (the composite validator and the result exception) were "equivalent". However, several objects "knew inside" that there were violations to that equivalency. I sometimes see you wrestling with similar problems, but I honestly don't have much to add :) I think it's an unsolvable problem, and you'll do similar in spirit to what I did (just trim the rough edges as best as possible), but with a more robust implementation design.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What you mentioned about the trees is so true... It got a bit better with the Result
since we don't loose some information that we would have lost if we were only dealing with exceptions (since we would only have exceptions when the validation failed, not if it succeeded), but still, there's quite a bit of overhead, Result
objects that serve no purpose but to hold the validation of its children. I'm reducing the overhead with the Reducer
rule, but still.
As for this specific change connected to your last comment, just as you, I also thought of getMessages()
as a useful and simple to handle HTML forms (or API requests). That's why I'm a bit reluctant to create this super structure here, it seems like when people get to the point of processing this structure, an array wouldn't be as useful anyways.
Good to hear your thoughts. It was also good to be reminded of the findMessages()
method. I've never used it myself, but I see how useful that can be, and I would like to keep it.
I'm thinking of repurposing the validate()
method, to return an object with failures. It could be something like this.
$failures = v::key('foo', v::intVal()->positive()->greaterThan())->validate($input);
if ($failures === null) {
echo 'No failures' . PHP_EOL;
return;
}
// Getting all messages
echo $failures->getMessage() . PHP_EOL;
echo $failures->getFullMessage() . PHP_EOL;
echo print_r($failures->getMessages(), true) . PHP_EOL;
// Finding specific messages
echo $failures->findMessage('foo') . PHP_EOL;
echo print_r($failures->findMessages(['foo' => 'Something went wrong']), true) . PHP_EOL;
Thoughts?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think it looks cool!
I would use $result
instead of $failures
, and maybe do a $result->hasPassed()
method instead of the null check. I have grown wary of optional nulls in interfaces recently :)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I thought about that too, and I suggested returning the failures because I think it's no use for the user to deal with anything but the failures, and consequently no reason for this object to contain passing results. Apart from that, the Rule::evaluate()
method already returns a Result
.
I do get what you're saying about interfaces that return nullable objects. Maybe something in-between could work:
$failures = v::key('foo', v::intVal()->positive()->greaterThan())->validate($input);
if (count($failures) === 0) {
echo 'No failures' . PHP_EOL;
return;
}
It looks very similar to Symfony Validator 🙃 But alternatively, it could be something like this:
$failures = v::key('foo', v::intVal()->positive()->greaterThan())->validate($input);
if ($failures->isEmpty()) {
echo 'No failures' . PHP_EOL;
return;
}
What do you think?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I like it (the latest snippet)! 🐼
Related to the discussion #1427