Skip to content
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

Merged
merged 9 commits into from
Oct 15, 2020

Conversation

maysunfaisal
Copy link
Contributor

@maysunfaisal maysunfaisal commented Oct 6, 2020

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/parser devfile/api repo

Which issue(s) this PR fixes:

#4073

PR acceptance criteria:

How to test changes / Special notes to the reviewer:
Run the unit tests in the validate pkg

@openshift-ci-robot openshift-ci-robot added do-not-merge/work-in-progress Indicates that a PR should not merge because it is a work in progress. Required by Prow. kind/cleanup labels Oct 6, 2020
@openshift-ci-robot
Copy link
Collaborator

Skipping CI for Draft Pull Request.
If you want CI signal for your change, please convert it to an actual PR.
You can still manually trigger a test run with /test all

@maysunfaisal maysunfaisal marked this pull request as ready for review October 6, 2020 19:00
@openshift-ci-robot openshift-ci-robot removed the do-not-merge/work-in-progress Indicates that a PR should not merge because it is a work in progress. Required by Prow. label Oct 6, 2020
Copy link
Member

@cdrage cdrage left a 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 {
Copy link
Member

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!

Copy link
Contributor Author

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?

@openshift-ci-robot openshift-ci-robot added lgtm Indicates that a PR is ready to be merged. Required by Prow. approved Indicates a PR has been approved by an approver from all required OWNERS files. Required by Prow. labels Oct 6, 2020
@elsony elsony removed the lgtm Indicates that a PR is ready to be merged. Required by Prow. label Oct 6, 2020
@maysunfaisal
Copy link
Contributor Author

@mik-dass @adisky could you pls review this PR? This PR refactors the validation pkg into odo specific and devfile specific logic; so that the devfile logic can be moved out to devfile/api repo in the future

@yangcao77
Copy link
Contributor

You might want to rebase due to merge of this pr: https://github.com/openshift/odo/pull/3959/files

Copy link
Contributor

@adisky adisky left a 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.

"github.com/openshift/odo/pkg/devfile/parser/data/common"
genericValidation "github.com/openshift/odo/pkg/devfile/validate/generic"
Copy link
Contributor

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

Copy link
Contributor Author

@maysunfaisal maysunfaisal Oct 7, 2020

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

err = validateCommand(command, commandsMap, components)
if err != nil {
return err
}
}

return
err = genericValidation.ValidateCommands(commands, commandsMap, components)
Copy link
Contributor

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)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

we can return here

if len(invalidVolumeMounts) > 0 {
return &MissingVolumeMountError{volumeName: strings.Join(invalidVolumeMounts, ",")}
}
err := genericValidation.ValidateComponents(components)
Copy link
Contributor

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
Copy link
Contributor

@mik-dass mik-dass Oct 7, 2020

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))
}

Comment on lines 71 to 73
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)
Copy link
Contributor

@mik-dass mik-dass Oct 7, 2020

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?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

good point

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

updated!, pls help review!

@girishramnani
Copy link
Contributor

perfect 1000 lines added!!

err = validateCommand(command, commandsMap, components)
if err != nil {
return err
}
}

return
err = genericValidation.ValidateCommands(commands, commandsMap, components)
Copy link
Contributor

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)

if len(invalidVolumeMounts) > 0 {
return &MissingVolumeMountError{volumeName: strings.Join(invalidVolumeMounts, ",")}
}
err := genericValidation.ValidateComponents(components)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

return genericValidation.ValidateComponents(components)

@maysunfaisal
Copy link
Contributor Author

maysunfaisal commented Oct 7, 2020

@girishramnani I had to clean up some test codes which were commented out, so that added to the lines. Now its approx -200.

@maysunfaisal
Copy link
Contributor Author

/retest

@codecov
Copy link

codecov bot commented Oct 8, 2020

Codecov Report

Merging #4085 into master will decrease coverage by 0.18%.
The diff coverage is 63.63%.

Impacted file tree graph

@@            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     
Impacted Files Coverage Δ
pkg/devfile/parser/data/common/command_helper.go 77.27% <ø> (-3.50%) ⬇️
pkg/devfile/validate/errors.go 0.00% <0.00%> (-43.75%) ⬇️
pkg/devfile/validate/generic/validate.go 0.00% <0.00%> (ø)
pkg/devfile/validate/validate.go 23.80% <0.00%> (+2.07%) ⬆️
pkg/devfile/validate/generic/errors.go 14.28% <14.28%> (ø)
pkg/devfile/validate/components.go 80.00% <50.00%> (-13.11%) ⬇️
pkg/devfile/validate/generic/events.go 27.27% <55.55%> (ø)
pkg/devfile/validate/commands.go 50.00% <66.66%> (-45.46%) ⬇️
pkg/devfile/validate/generic/commands.go 91.11% <91.11%> (ø)
pkg/devfile/adapters/common/utils.go 94.23% <100.00%> (ø)
... and 16 more

Continue to review full report at Codecov.

Legend - Click here to learn more
Δ = absolute <relative> (impact), ø = not affected, ? = missing data
Powered by Codecov. Last update b1245a5...92e09f6. Read the comment docs.

Copy link
Contributor

@adisky adisky left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Comment on lines 25 to 40
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)
}
Copy link
Contributor

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?

Copy link
Contributor

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.

Copy link
Contributor Author

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

Comment on lines 47 to 78

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)
}

Copy link
Contributor

@adisky adisky Oct 12, 2020

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 {
Copy link
Contributor

@adisky adisky Oct 12, 2020

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

Copy link
Contributor Author

@maysunfaisal maysunfaisal Oct 13, 2020

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:

  1. validation pkg is generally for validating logic directly related to the devfile schema and enforcing it
  2. 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!

Copy link
Contributor

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?

Copy link
Contributor Author

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.

@openshift-ci-robot
Copy link
Collaborator

[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 /approve in a comment
Approvers can cancel approval by writing /approve cancel in a comment

@maysunfaisal maysunfaisal requested a review from adisky October 14, 2020 15:33
"github.com/openshift/odo/pkg/devfile/parser/data/common"
"github.com/openshift/odo/pkg/util"
"k8s.io/apimachinery/pkg/api/resource"
)
Copy link
Contributor

@adisky adisky Oct 15, 2020

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

Copy link
Contributor Author

@maysunfaisal maysunfaisal Oct 15, 2020

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  1. 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..
  2. 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

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

gotcha!

@adisky
Copy link
Contributor

adisky commented Oct 15, 2020

/lgtm

@openshift-ci-robot openshift-ci-robot added the lgtm Indicates that a PR is ready to be merged. Required by Prow. label Oct 15, 2020
@openshift-merge-robot openshift-merge-robot merged commit 84512b9 into redhat-developer:master Oct 15, 2020
@rm3l rm3l added the area/refactoring Issues or PRs related to code refactoring label Jun 16, 2023
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
approved Indicates a PR has been approved by an approver from all required OWNERS files. Required by Prow. area/refactoring Issues or PRs related to code refactoring lgtm Indicates that a PR is ready to be merged. Required by Prow.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

10 participants