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

Change Flyte CR naming scheme to better support namespace_mapping #5480

Open
wants to merge 1 commit into
base: master
Choose a base branch
from

Conversation

ddl-ebrown
Copy link
Contributor

@ddl-ebrown ddl-ebrown commented Jun 14, 2024

@ddl-rliu did most of the work on this one - making this an upstream PR as it resolved a real issue for us.

Tracking issue

Why are the changes needed?

  • Typically Flyte is configured so that each project / domain has its
    own Kubernetes namespace.

    Certain environments may change this behavior by using the Flyteadmin
    namespace_mapping setting to put all executions in fewer (or a singular)
    Kubernetes namespace. This is problematic because it can lead to
    collisions in the naming of the CR that flyteadmin generates.

What changes were proposed in this pull request?

  • This patch fixes 2 important things to make this work properly inside
    of Flyte:

    • it adds a random element to the CR name in Flyte so that the CR is
      named by the execution + some unique value when created by
      flyteadmin

      Without this change, an execution Foo in project A will prevent an
      execution Foo in project B from launching, because the name of the
      CR thats generated in Kubernetes assumes that the namespace the
      CRs are put into is different for project A and project B

      When namespace_mapping is set to a singular value, that assumption
      is wrong

    • it makes sure that when flytepropeller cleans up the CR resource
      that it uses Kubernetes labels to find the correct CR -- so instead
      of assuming that it can use the execution name, it instead uses the
      project, domain and execution labels

How was this patch tested?

This is deployed in a live Flyte setup where we have automated tests. We observed that the CR names were correctly unique after this and the initial collision no longer occurred.

Setup process

Screenshots

Check all the applicable boxes

  • I updated the documentation accordingly.
  • All new and existing tests passed.
  • All commits are signed-off.

Related PRs

Docs link

@ddl-ebrown ddl-ebrown force-pushed the change-flyte-CR-naming-scheme branch from 806c40b to 438dd97 Compare June 14, 2024 22:39
Copy link

codecov bot commented Jun 14, 2024

Codecov Report

Attention: Patch coverage is 88.23529% with 4 lines in your changes missing coverage. Please review.

Project coverage is 36.18%. Comparing base (7136919) to head (a268b61).

Files with missing lines Patch % Lines
...ropeller/pkg/compiler/transformers/k8s/workflow.go 80.00% 3 Missing and 1 partial ⚠️
Additional details and impacted files
@@           Coverage Diff           @@
##           master    #5480   +/-   ##
=======================================
  Coverage   36.18%   36.18%           
=======================================
  Files        1302     1302           
  Lines      109614   109644   +30     
=======================================
+ Hits        39660    39680   +20     
- Misses      65809    65818    +9     
- Partials     4145     4146    +1     
Flag Coverage Δ
unittests-datacatalog 51.37% <ø> (ø)
unittests-flyteadmin 55.32% <100.00%> (-0.01%) ⬇️
unittests-flytecopilot 12.17% <ø> (ø)
unittests-flytectl 62.18% <ø> (ø)
unittests-flyteidl 7.12% <ø> (ø)
unittests-flyteplugins 53.34% <ø> (ø)
unittests-flytepropeller 41.74% <80.00%> (+0.02%) ⬆️
unittests-flytestdlib 55.35% <ø> (ø)

Flags with carried forward coverage won't be shown. Click here to find out more.

☔ View full report in Codecov by Sentry.
📢 Have feedback on the report? Share it here.

@ddl-ebrown ddl-ebrown force-pushed the change-flyte-CR-naming-scheme branch from 438dd97 to 532888b Compare June 14, 2024 22:44
ctx,
v1.DeleteOptions{PropagationPolicy: &deletePropagationBackground},
v1.ListOptions{
LabelSelector: v1.FormatLabelSelector(executionLabelSelector(data.ExecutionID)),
Copy link
Contributor Author

Choose a reason for hiding this comment

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

Even though new executions will have different CR names, this deletion mechanism is fully backwards compatible - thanks for a good solution @ddl-rliu !

@ddl-ebrown ddl-ebrown force-pushed the change-flyte-CR-naming-scheme branch 2 times, most recently from 33e6e7d to f97c77a Compare June 14, 2024 23:27
@ddl-ebrown ddl-ebrown force-pushed the change-flyte-CR-naming-scheme branch from f97c77a to d994304 Compare June 14, 2024 23:52
@@ -165,6 +165,10 @@ func TestDynamic(t *testing.T) {
Name: "name",
},
"namespace")
// make sure real CR has randomized suffix
Copy link
Contributor Author

Choose a reason for hiding this comment

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

This is probably the simplest way to ensure existing tests continue to pass

rand.Seed(seed)
// K8s has a limitation of 63 chars
name = name[:minInt(63-ExecutionIDSuffixLength, len(name))]
execName := name + "-" + rand.String(ExecutionIDSuffixLength-1)
Copy link
Contributor Author

Choose a reason for hiding this comment

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

Without this randomization, use of namespace_mapping in the config

const (
namespaceMappingKey = "namespace_mapping"
defaultTemplate = "{{ project }}-{{ domain }}"
)
var namespaceMappingConfig = config.MustRegisterSection(namespaceMappingKey, &interfaces.NamespaceMappingConfig{
Template: defaultTemplate,
})
with a value like foo causes problems when executions have the same names across projects

@davidmirror-ops davidmirror-ops requested a review from hamersaw June 17, 2024 12:21
Copy link
Contributor

@hamersaw hamersaw left a comment

Choose a reason for hiding this comment

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

.

@@ -159,6 +161,17 @@ func generateName(wfID *core.Identifier, execID *core.WorkflowExecutionIdentifie
}
}

const ExecutionIDSuffixLength = 21
Copy link
Contributor

Choose a reason for hiding this comment

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

Can we make this a configurable value and if set to 0 (default?) then have it disabled (so no random characters are appended). My concern is that if anything relies on the CR name, this will break it.

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 think we found / updated the spots where the name is a "contract" -- but if we want to be extra super careful we could make this configurable.

That said, I think the tradeoff we have to consider is:

  • backward compatibility vs
  • adding extra config / managing different behaviors

I would probably vote for not introducing an extra config and keeping this behavior not configurable (I'd argue prior behavior was a bug), but admit to not knowing the potential blast radius beyond core Flyte (i.e. plugins and such).

I'm happy to go either way since it's not my project :)

Copy link
Contributor

Choose a reason for hiding this comment

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

Yeah, I understand the config bloat all to well :). As you suggest, my main concern is breaking backwards compatibility here. I know there are Flyte users that rely on the FlyteWorkflow CR to be named identical to the execution ID, which this would break. For me to be comfortable merging this, I feel it should be defaulted to the current behavior. cc @eapolinario thoughts?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Ah thanks!

If you know there are users depending on the existing CR naming scheme somehow, then making this behavior configurable seems like the only thing to do right now. I can update my PR.

Not sure if you track potential breaking change tickets anywhere, but maybe file one away to remove the option to configure this on the next major release boundary?

Copy link
Contributor

Choose a reason for hiding this comment

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

Thanks for the feedback! I've updated the PR to address this and explain a bit more here: #5480 (comment)

@kumare3
Copy link
Contributor

kumare3 commented Jun 25, 2024

I am not in favor of this, as randomness will lead to leaky workflows and duplicates. We should use the project id itself or generate a consistent hash to increase inter project execution entropy

@ddl-ebrown
Copy link
Contributor Author

I am not in favor of this, as randomness will lead to leaky workflows and duplicates. We should use the project id itself or generate a consistent hash to increase inter project execution entropy

Ah thanks @kumare3 for the heads up! We clearly didn't realize there was something internal to Flyte that depends on deterministic naming for CRs -- will make some updates taking that into account as well

@ddl-ebrown
Copy link
Contributor Author

I am not in favor of this, as randomness will lead to leaky workflows and duplicates. We should use the project id itself or generate a consistent hash to increase inter project execution entropy

Ah thanks @kumare3 for the heads up! We clearly didn't realize there was something internal to Flyte that depends on deterministic naming for CRs -- will make some updates taking that into account as well

Also, should mention @kumare3 that if by "leaky" you meant "CR might not be deleted from the cluster", the deletion process is robust because this uses the actual key of the workflow in conjunction with CR labels to perform deletes, rather than the CR name.

If there are dupe CRs for the same workflow though, that's clearly an issue regardless :)

@EngHabu
Copy link
Contributor

EngHabu commented Jun 25, 2024

@ddl-ebrown I agree with not introducing randomization... specially that the name already starts with a random string :-)

Instead, I would update this call to use something like project-domain-rand(10) and hash that and that becomes the execution name...

I would also make the length of the execution name configurable in flyteadmin. so in your deployment you can make it longer and give you better entropy...

@ddl-rliu ddl-rliu force-pushed the change-flyte-CR-naming-scheme branch 6 times, most recently from 33a99f0 to 93bb9a5 Compare July 18, 2024 22:46
}
// make sure real CR has randomized suffix
assert.Regexp(t, regexp.MustCompile("name-[bcdfghjklmnpqrstvwxz2456789]{20}"), flyteWf.Name)
// then reset for the sake of serialization comparison
flyteWf.Name = "name"

Copy link
Contributor

@ddl-rliu ddl-rliu Jul 18, 2024

Choose a reason for hiding this comment

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

The cr-name-scheme-change is now non-default, and tests are added here, so removing these checks.

Comment on lines 252 to 257
hashedIdentifier := hashIdentifier(core.Identifier{
Project: project,
Domain: domain,
Name: name,
})
rand.Seed(int64(hashedIdentifier))
Copy link
Contributor

@ddl-rliu ddl-rliu Jul 18, 2024

Choose a reason for hiding this comment

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

By hashing exactly these fields project, domain, name we prevent duplicate CRs from being created for the same execution (they will hash to the same CR name and k8s will prevent those duplicate CRs).

Edit: Regarding the hashIdentifier function, see this for reference.

Comment on lines 259 to 268
if workflowCRNameHashLength := config.GetConfig().WorkflowCRNameHashLength; workflowCRNameHashLength > 0 {
obj.ObjectMeta.Name = rand.String(workflowCRNameHashLength)
} else {
obj.ObjectMeta.Name = name
}
Copy link
Contributor

@ddl-rliu ddl-rliu Jul 18, 2024

Choose a reason for hiding this comment

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

By default WorkflowCRNameHashLength is 0 so the hash CR naming scheme will only be used when the deployment is specially configured to do so. See 93bb9a5#diff-67941e1320bd66fda570ebcf1f1c5c5221d608b136d662705e168a07a5d166ef

rand.Seed(int64(hashedIdentifier))

if workflowCRNameHashLength := config.GetConfig().WorkflowCRNameHashLength; workflowCRNameHashLength > 0 {
obj.ObjectMeta.Name = rand.String(workflowCRNameHashLength)
Copy link
Contributor Author

Choose a reason for hiding this comment

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

Couldn't the name of the CR just incorporate the k8s namespace though and still be stable? I think using just a hash for the CR name loses some valuable information when you're glancing at the resource names now.

Copy link
Contributor

@ddl-rliu ddl-rliu Jul 23, 2024

Choose a reason for hiding this comment

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

Updated #5480 (comment)

Now, for a setting like WorkflowCRNameHashLength: 32, the CR name will look like mywfname-f5ptmmd66pktp59nphfdjkqr9szp8fv5 or mywfname-wnw9zlg22z5wmdjggdcrnzdrpw59m4bc. Note that the suffix is deterministic, and uses a hash of the name+project+domain.

I opted for this approach rather than mywfname-project, this is because of the character limit of the CR name and how this affects executions that have a long execution name and a long project name, where improperly handled truncation will lead to undesired behavior such as executions not starting as a result of name conflicts. For reference.

@ddl-rliu ddl-rliu force-pushed the change-flyte-CR-naming-scheme branch 2 times, most recently from f46550d to d4a0e72 Compare July 23, 2024 18:02
Comment on lines 264 to 269
base := name + "-"
maxNameLength := allowedExecutionNameLength - workflowCRNameHashLength
if len(base) > maxNameLength {
base = base[:maxNameLength]
}
obj.ObjectMeta.Name = fmt.Sprintf("%s%s", base, rand.String(workflowCRNameHashLength))
Copy link
Contributor

Choose a reason for hiding this comment

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

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Not to bikeshed too much on this one, but a couple things...

You can skip some of that slice logic around length if you just use format specifiers to set the "max"

maxNameLength := allowedExecutionNameLength - workflowCRNameHashLength - 1
if s, err := strconv.Atoi(maxNameLength); err == nil {
    format := "[%." + s+ "s]-%s"
    obj.ObjectMeta.Name = fmt.Sprintf(format, base, rand.String(workflowCRNameHashLength))
}

Even better -- does WorkflowCRNameHashLength really need to be configurable or can it just be on / off? If it's sufficiently long enough to prevent collisions (and I think we can pick something), then you could simplify this length calculation logic and just limit the string statically using a format specifier like this (let's assume its set to 24, for example):

maxBase := allowedExecutionNameLength - workflowCRNameHashLength - 1
obj.ObjectMeta.Name = fmt.Sprintf("[%.38s]-%s", base, rand.String(workflowCRNameHashLength))

Copy link
Contributor Author

@ddl-ebrown ddl-ebrown Jul 25, 2024

Choose a reason for hiding this comment

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

The k8s rand util you're using has 32 possible characters... I think once you hit a string of length 24 you're in UUID territory (which is 5.31x10^36 and generally considered more than anyone would ever need)

I don't think something that long is remotely necessary for this use case though - the primary concern is that the seeded random number generator doesn't produce a duplicate random string given two different seeds (i.e. for an execution with the same name and domain, but in a different project). For that, even something like 10 characters should be enough I think, which is why I would advocate for picking something sensible. I know maintainers suggested making it configurable, but that feels unnecessary IMHO.

32^10 is 1.13x10^15

Using the collision calculator at https://devina.io/collision-calculator to understand this in more human-friendly terms

~1 year (or 3.43e+7 seconds) needed, in order to have a 1% probability of at least one collision if 500 ID's are generated every hour.

Copy link
Contributor

@ddl-rliu ddl-rliu Jul 25, 2024

Choose a reason for hiding this comment

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

A fixed suffix length of 10 sounds good to me. Agreed about the sufficient amount of entropy on relatively high values for the suffix length. Just updated the PR: 15b57cb

@ddl-rliu ddl-rliu force-pushed the change-flyte-CR-naming-scheme branch 2 times, most recently from 70accda to 15b57cb Compare July 25, 2024 23:22
 - Typically Flyte is configured so that each project / domain has its
   own Kubernetes namespace.

   Certain environments may change this behavior by using the Flyteadmin
   namespace_mapping setting to put all executions in fewer (or a singular)
   Kubernetes namespace. This is problematic because it can lead to
   collisions in the naming of the CR that flyteadmin generates.

 - This patch fixes 2 important things to make this work properly inside
   of Flyte:

   * it adds a random element to the CR name in Flyte so that the CR is
     named by the execution + some unique value when created by
     flyteadmin

     Without this change, an execution Foo in project A will prevent an
     execution Foo in project B from launching, because the name of the
     CR thats generated in Kubernetes *assumes* that the namespace the
     CRs are put into is different for project A and project B

     When namespace_mapping is set to a singular value, that assumption
     is wrong

   * it makes sure that when flytepropeller cleans up the CR resource
     that it uses Kubernetes labels to find the correct CR -- so instead
     of assuming that it can use the execution name, it instead uses the
     project, domain and execution labels

 - Use deterministic hash of execution id, name, project, as the FlyteWorkflow
   CR name. Add workflow-cr-name-hash-length to flytepropeller config.

Signed-off-by: ddl-ebrown <[email protected]>
Signed-off-by: ddl-rliu <[email protected]>
@ddl-ebrown ddl-ebrown force-pushed the change-flyte-CR-naming-scheme branch from 15b57cb to a268b61 Compare August 28, 2024 16:38
@davidmirror-ops davidmirror-ops requested a review from pvditt October 3, 2024 16:50
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants