Skip to content

Conversation

@nightkr
Copy link
Contributor

@nightkr nightkr commented Mar 19, 2025

Description

See stackabletech/listener-operator#285, part of stackabletech/listener-operator#288

This PR adds a two new fields to ListenerClass:

spec:
  loadBalancerClass: foo
  loadBalancerAllocateNodePorts: true

Which are mapped to Service.spec.{loadBalancerClass,allocateLoadBalancerNodePorts}, respectively. Currently these are ignored unless serviceType: LoadBalancer; in v1alpha2, serviceType should be turned into a rich enum instead, and these fields be moved into it, like so:

spec:
  serviceType:
    loadBalancer:
      loadBalancerClass: foo
      allocateNodePorts: true

Definition of Done Checklist

  • Not all of these items are applicable to all PRs, the author should update this template to only leave the boxes in that are relevant
  • Please make sure all these things are done and tick the boxes
# Author
- [ ] Changes are OpenShift compatible
- [ ] CRD changes approved
- [ ] Integration tests passed (for non trivial changes)
# Reviewer
- [x] Code contains useful comments
- [ ] (Integration-)Test cases added → should be tested in stackabletech/listener-operator#288
- [ ] Documentation added or updated → should be documented in stackabletech/listener-operator#288
- [x] Changelog updated
- [x] Cargo.toml only contains references to git tags (not specific commits or branches)
# Acceptance
- [ ] Feature Tracker has been updated
- [ ] Proper release label has been added

@nightkr
Copy link
Contributor Author

nightkr commented Mar 19, 2025

Moving the CRD changes into RFC

@siegfriedweber
Copy link
Member

siegfriedweber commented Mar 25, 2025

The suggestion looks okay for me. Nevertheless, I would like to present my thoughts that lead me to another suggestion:

  • How would the spec look like in v1alpha2 for ClusterIP and NodePort where usually nothing is defined?
    spec:
      serviceType:
        clusterIp: {}
    The default use case will look slightly uglier than now.
  • The additional Service properties are sometimes specific to the Service type (e.g. loadBalancerClass) and sometimes they are not (e.g. publishNotReadyAddresses). Kubernetes decided against a deep structure in the Service spec for whatever reason. Why should we introduce one if users already got used to the flat structure?
  • The additional properties "are" not a service type, they are part of the service with a specific type, so serviceType: <type identifier> feels more accurate than serviceType: <structure where the next key is the type identifier>.
  • We prefixed serviceType with service because it is the type of the Service, not the type of the ListenerClass. It might be consistent, to also prefix the other Service properties, e.g. serviceLoadBalancerClass.
  • The Service property allocateLoadBalancerNodePorts is named loadBalancerAllocateNodePorts in the Listener, probably to emphasize that it belongs to the Service type loadBalancer. It is easier for the user, if Kubernetes knowledge can just be applied to our platform without translation.
  • In v1alpha2, loadBalancerClass is called loadBalancerClass in the loadBalancer structure, but loadBalancerAllocateNodePorts is renamed to allocateNodePorts. In which cases will the prefix be removed and in which ones not?

The ListenerClass requires Service properties. Why not just add an according structure where the properties can be set in exactly the same way, as they are set in Services:

spec:
  serviceProperties:
    type: LoadBalancer:
    loadBalancerClass: manual
    allocateLoadBalancerNodePorts: false

If we do not want to introduce a breaking change, we could just keep the field serviceType:

spec:
  serviceType: LoadBalancer
  serviceProperties:
    loadBalancerClass: manual
    allocateLoadBalancerNodePorts: false

@nightkr
Copy link
Contributor Author

nightkr commented Mar 26, 2025

How would the spec look like in v1alpha2 for ClusterIP and NodePort where usually nothing is defined?

Yeah. Not super pretty, but also consistent with emptyDir: {} and http01: {}.

The additional Service properties are sometimes specific to the Service type (e.g. loadBalancerClass) and sometimes they are not (e.g. publishNotReadyAddresses). Kubernetes decided against a deep structure in the Service spec for whatever reason. Why should we introduce one if users already got used to the flat structure?

Looking back at v1.0.0 (https://github.com/kubernetes/kubernetes/blob/v1.0.0/api/swagger-spec/v1.json#L13468-L13499), they seem to have made the same mistake as us: made it a string enum before they had any per-type options, and then they were unable to change it since it was already a stable API.

They also do some weird pseudo-enumisms (For example: "This field will be wiped when a service is updated to a non 'LoadBalancer' type." for loadBalancerClass).

We prefixed serviceType with service because it is the type of the Service, not the type of the ListenerClass. It might be consistent, to also prefix the other Service properties, e.g. serviceLoadBalancerClass.

In retrospect, I think this might've been a mistake. Creating a backing Service is one way to implement a Listener, but it's not really core to the API contract at all (for example, I could see future ListenerClass types that use Ingress or Gateway instead, or which bypass K8s networking entirely).

It's a tricky balance between letting people reuse existing K8s knowledge and owning our own API that we can adapt to our own needs.

The additional properties "are" not a service type, they are part of the service with a specific type

They are parameters for instantiating that service type, and is similar to how rich enums are handled elsewhere (Volume). Unless you mean that you object to the word "type", in which case I agree but don't have a better option to offer.

In v1alpha2, loadBalancerClass is called loadBalancerClass in the loadBalancer structure, but loadBalancerAllocateNodePorts is renamed to allocateNodePorts. In which cases will the prefix be removed and in which ones not?

Mentally, I read loadBalancerClass as "load-balancer class" (one concept that implies that it is only active in load-balancer mode), but loadBalancerAllocateNodePorts as "when in load-balancer mode, allocate node ports?" (one concept guarded by a conditional). I would drop the conditional when implied by context, but would not drop half the concept name.

The ListenerClass requires Service properties. Why not just add an according structure where the properties can be set in exactly the same way, as they are set in Services:

My response to this is similar to the "prefix everything with service" comment: I'm skeptical to this because it doesn't really define a clear identity for listeners.

If you want the prefix for "anything that currently relates to Services at all" then the prefix applies to, well, the whole CRD and we gain pretty much nothing.

If you want the prefix for "things that are passed directly through to the Service" then I'm hesitant to apply it to anything because I think we should own the API, and what it means. The exception would be in defining some sort of generic serviceOverride mechanism, perhaps (similar to our existing podOverride). I would also class our existing serviceAnnotations under this bucket of "overrides for pragmatic reasons, even if it doesn't quite line up with our vision for our API".

@nightkr nightkr marked this pull request as ready for review April 28, 2025 11:28
@nightkr nightkr requested a review from siegfriedweber May 6, 2025 10:55
siegfriedweber
siegfriedweber previously approved these changes May 6, 2025
Copy link
Member

@siegfriedweber siegfriedweber left a comment

Choose a reason for hiding this comment

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

LGTM

siegfriedweber
siegfriedweber previously approved these changes May 13, 2025
@nightkr nightkr added this pull request to the merge queue May 13, 2025
@github-merge-queue github-merge-queue bot removed this pull request from the merge queue due to no response for status checks May 13, 2025
@nightkr
Copy link
Contributor Author

nightkr commented May 26, 2025

Clippy runs fine locally, the remaining CI failure is "just" the pinning issue that we've already been discussing.

@nightkr nightkr enabled auto-merge May 26, 2025 10:16
@nightkr nightkr requested a review from siegfriedweber May 26, 2025 10:16
@nightkr nightkr added this pull request to the merge queue May 26, 2025
Merged via the queue into main with commit 077302c May 26, 2025
12 of 14 checks passed
@nightkr nightkr deleted the feature/listenerclass-loadbalancerclass branch May 26, 2025 12:31
github-merge-queue bot pushed a commit to stackabletech/listener-operator that referenced this pull request May 28, 2025
* Map ListenerClass.spec.loadBalancerClass into Service.spec

Fixes #285, requires stackabletech/operator-rs#986

* Switch to stackable-operator 0.93.2

* Add a test for custom lbclasses

* Documentation

* Remove unused direct rustls dependency

Seems like this was fixed elsewhere in our dependency tree

* Changelog

* Also document nodeport allocation control

* Add openshift dimension to custom-lbclass test

* Try to clarify why custom-lbclass is separate

* Also cover nodeport allocation in the custom-lbclass test

* Also assert the ports in custom-lbclass

(By @siegfriedweber)
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.

3 participants