-
Notifications
You must be signed in to change notification settings - Fork 242
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
Refactor validate pkg for generic and odo validation #4085
Refactor validate pkg for generic and odo validation #4085
Conversation
Skipping CI for Draft Pull Request. |
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 refactoring and moving the tests!
/approve
@@ -59,6 +59,16 @@ func (dc DevfileCommand) GetExecWorkingDir() string { | |||
return "" | |||
} | |||
|
|||
// IsExec checks if the command is an exec command | |||
func (dc DevfileCommand) IsExec() bool { |
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.
New function means new tests :)
Despite this test being extremely small haha, we should add a test function for this!
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.
@cdrage yes, I'll add the new test in ;)
could you remove the lgtm label? it looks like it got added?
You might want to rebase due to merge of this pr: https://github.com/openshift/odo/pull/3959/files |
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.
@maysunfaisal If generic validation is extracted out for library, so the library is going to provide separate function to validate each resource? or it is only the first step for refactoring?
i think odo does not need to call each function isEventValid
, ValidateCommands
etc, it should call a single function generic.Validate()
and then if everything is good from generic validation, odo starts own validation, I think both the validation should be decoupled more.
pkg/devfile/validate/commands.go
Outdated
"github.com/openshift/odo/pkg/devfile/parser/data/common" | ||
genericValidation "github.com/openshift/odo/pkg/devfile/validate/generic" |
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 the alias also not needed, generic.ValidateCommands is pretty intuitive
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.
updated!, pls help review!
I've also added a single call for generic validation like you suggested and it makes sense.
And yes, I'm refactoring the generic validation to a separate package because down the road devfile/api will provide validation for all devfile data acc to the schema. So separating out the generic validation will help in moving it out when it's required. And Odo will remain with its specific validation for its usage. The issue to track this is devfile/api#151
pkg/devfile/validate/commands.go
Outdated
err = validateCommand(command, commandsMap, components) | ||
if err != nil { | ||
return err | ||
} | ||
} | ||
|
||
return | ||
err = genericValidation.ValidateCommands(commands, commandsMap, components) |
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.
we can return here
return ValidateCompositeCommand(&command, parentCommands, devfileCommands, components) | ||
} | ||
|
||
err = ValidateExecCommand(command, components) |
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.
we can return here
pkg/devfile/validate/components.go
Outdated
if len(invalidVolumeMounts) > 0 { | ||
return &MissingVolumeMountError{volumeName: strings.Join(invalidVolumeMounts, ",")} | ||
} | ||
err := genericValidation.ValidateComponents(components) |
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.
we can return here
isEventPresent := false | ||
|
||
if _, ok := commands[strings.ToLower(eventName)]; ok { | ||
isEventPresent = true |
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 we don't need this variable. We can simplify it like
if _, ok := commands[strings.ToLower(eventName)]; !ok {
klog.V(2).Infof("%s type event %s does not map to a valid devfile command", eventType, eventName)
eventErrorMsg[strings.ToLower(eventName)] = append(eventErrorMsg[strings.ToLower(eventName)], fmt.Sprintf("event %s does not map to a valid devfile command", eventName))
}
pkg/devfile/validate/events.go
Outdated
if command.GetID() == strings.ToLower(eventName) { | ||
isEventPresent = true | ||
|
||
// Check if the devfile command is valid | ||
// Check if the devfile command is a valid odo devfile command | ||
err := validateCommand(command, commands, components) |
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.
Since the commands
map contain all the commands in the devfile and they are all validated before this call, can't we just check if a eventName
is in the commands
map instead of validating each event again? What am I missing?
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.
good point
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.
updated!, pls help review!
perfect 1000 lines added!! |
pkg/devfile/validate/commands.go
Outdated
err = validateCommand(command, commandsMap, components) | ||
if err != nil { | ||
return err | ||
} | ||
} | ||
|
||
return | ||
err = genericValidation.ValidateCommands(commands, commandsMap, components) |
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.
return genericValidation.ValidateCommands(commands, commandsMap, components)
pkg/devfile/validate/components.go
Outdated
if len(invalidVolumeMounts) > 0 { | ||
return &MissingVolumeMountError{volumeName: strings.Join(invalidVolumeMounts, ",")} | ||
} | ||
err := genericValidation.ValidateComponents(components) |
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.
return genericValidation.ValidateComponents(components)
@girishramnani I had to clean up some test codes which were commented out, so that added to the lines. Now its approx -200. |
/retest |
Codecov Report
@@ Coverage Diff @@
## master #4085 +/- ##
==========================================
- Coverage 43.17% 42.98% -0.19%
==========================================
Files 147 150 +3
Lines 12561 12464 -97
==========================================
- Hits 5423 5358 -65
+ Misses 6562 6534 -28
+ Partials 576 572 -4
Continue to review full report at Codecov.
|
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.
Mostly looks good to me, a nit here update the comment https://github.com/openshift/odo/blob/master/pkg/devfile/parse.go#L41
type ExecCommandMissingComponentError struct { | ||
commandId string | ||
} | ||
|
||
func (e *ExecCommandMissingComponentError) Error() string { | ||
return fmt.Sprintf("exec command %q must reference a component", e.commandId) | ||
} | ||
|
||
// ExecCommandMissingCommandLineError returns an error if the exec command does not have a command line | ||
type ExecCommandMissingCommandLineError struct { | ||
commandId string | ||
} | ||
|
||
func (e *ExecCommandMissingCommandLineError) Error() string { | ||
return fmt.Sprintf("exec command %q must have a command", e.commandId) | ||
} |
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.
should this all be covered in a single error like MissingInformationError
?
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.
This error refactoring could be done in separate PR, for now its ok.
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've already refactored, just need to push my changes
|
||
func (e *InvalidCommandError) Error() string { | ||
return fmt.Sprintf("the command %q should be of type %s", e.commandId, e.commandType) | ||
} | ||
|
||
// ExecCommandInvalidContainerError returns an error if the exec command references an invalid container component | ||
type ExecCommandInvalidContainerError struct { | ||
commandId string | ||
} | ||
|
||
func (e *ExecCommandInvalidContainerError) Error() string { | ||
return fmt.Sprintf("the command %q does not map to a container component", e.commandId) | ||
} | ||
|
||
// CompositeDirectReferenceError returns an error if the composite command directly references itself | ||
type CompositeDirectReferenceError struct { | ||
commandId string | ||
} | ||
|
||
func (e *CompositeDirectReferenceError) Error() string { | ||
return fmt.Sprintf("the composite command %q cannot reference itself", e.commandId) | ||
} | ||
|
||
// CompositeIndirectReferenceError returns an error if the composite command indirectly references itself | ||
type CompositeIndirectReferenceError struct { | ||
commandId string | ||
} | ||
|
||
func (e *CompositeIndirectReferenceError) Error() string { | ||
return fmt.Sprintf("the composite command %q cannot indirectly reference itself", e.commandId) | ||
} | ||
|
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 the errors could me refactored more here, like instead of having separate error for each validation fail it could be more common InvalidCommand
MissingInformation
|
||
// Validate all the events | ||
if err := validateEvents(events, commandsMap, components); err != nil { | ||
if err := validateCommands(commandsMap); err != nil { |
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 ValidateContainerName
defined below should also be part of generic validation, it could be added inside component validation
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.
There are a few reasons why I did not do it:
- validation pkg is generally for validating logic directly related to the devfile schema and enforcing it
- having said point 1, this func validates how the kube resource is to be created which is not really a devfile data validation, and it should not have been in validate pkg in the first place and is relying on adapter util funcs which further adds reason behind my point no 1.
Odo is the only container based tool. Workspace may need it but I dont see Console using it? 🤔
Lmk, what 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.
@maysunfaisal It is ok to not to add in library, just wanted an opinion essentially it is the validation of component name specified in the devfile, the container component would be translated to containers to all other tools, no? what console is translating devfile containers to if specified?
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.
@adisky I've updated the pkg to validate the component names and made it more cleaner, pls have a look! Thx.
Signed-off-by: Maysun J Faisal <[email protected]>
Signed-off-by: Maysun J Faisal <[email protected]>
Signed-off-by: Maysun J Faisal <[email protected]>
Signed-off-by: Maysun J Faisal <[email protected]>
Signed-off-by: Maysun J Faisal <[email protected]>
Signed-off-by: Maysun J Faisal <[email protected]>
[APPROVALNOTIFIER] This PR is APPROVED This pull-request has been approved by: cdrage The full list of commands accepted by this bot can be found here. The pull request process is described here
Needs approval from an approver in each of these files:
Approvers can indicate their approval by writing |
Signed-off-by: Maysun J Faisal <[email protected]>
Signed-off-by: Maysun J Faisal <[email protected]>
Signed-off-by: Maysun J Faisal <[email protected]>
"github.com/openshift/odo/pkg/devfile/parser/data/common" | ||
"github.com/openshift/odo/pkg/util" | ||
"k8s.io/apimachinery/pkg/api/resource" | ||
) |
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.
we should not import odo packages in library, can we move this function in generic package ?util.ValidateK8sResourceName
"github.com/openshift/odo/pkg/util"
not sure about this one, anyways the constants are not related with adapters, it is a devfile spec so can be moved in types package.
"github.com/openshift/odo/pkg/devfile/adapters/common
for below one i understand, it will get updated when we use types from devfile/api
github.com/openshift/odo/pkg/devfile/parser/data/common
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.
- So devfile/parser already has a copy of the util func ValidateK8sResourceName https://github.com/devfile/parser/blob/master/pkg/util/util.go#L1044. I plan on using that when its exported out. It will just be changing the import. Also this func is used by other pkgs in odo..
- For
adaptersCommon "github.com/openshift/odo/pkg/devfile/adapters/common"
, we are just using constants. There will be new const when its moved to devfile/parser. Since this is still in odo, I dont want to maintain two separate const for the same thing.
Lmk, what you think @adisky
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.
gotcha!
/lgtm |
Signed-off-by: Maysun J Faisal [email protected]
What type of PR is this?
/kind cleanup
/kind code-refactoring
What does does this PR do / why we need it:
We need this PR to refactor the validation pkg to odo specific validation and devfile generic validation, so that the generic validation can be exported out to the
devfile/parserdevfile/api repoWhich issue(s) this PR fixes:
#4073
PR acceptance criteria:
Unit test
Integration test
Documentation
I have read the test guidelines
How to test changes / Special notes to the reviewer:
Run the unit tests in the validate pkg