Skip to content

Commit

Permalink
chore/Optionally add additional ports to the prefect server service (#…
Browse files Browse the repository at this point in the history
…114)

* Optionally add additional ports to the prefect server service

* Update internal/controller/prefectserver_controller_test.go

Co-authored-by: Mitchell Nielsen <[email protected]>

* Update internal/controller/prefectserver_controller_test.go

Co-authored-by: Mitchell Nielsen <[email protected]>

---------

Co-authored-by: Mitchell Nielsen <[email protected]>
  • Loading branch information
jimid27 and mitchnielsen authored Oct 11, 2024
1 parent f81064f commit 791a989
Show file tree
Hide file tree
Showing 6 changed files with 128 additions and 1 deletion.
3 changes: 3 additions & 0 deletions api/v1/prefectserver_types.go
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,9 @@ type PrefectServerSpec struct {
// ExtraContainers defines additional containers to add to the Prefect Server Deployment
ExtraContainers []corev1.Container `json:"extraContainers,omitempty"`

// ExtraServicePorts defines additional ports to expose on the Prefect Server Service
ExtraServicePorts []corev1.ServicePort `json:"extraServicePorts,omitempty"`

// Ephemeral defines whether the server will be deployed with an ephemeral storage backend
Ephemeral *EphemeralConfiguration `json:"ephemeral,omitempty"`

Expand Down
7 changes: 7 additions & 0 deletions api/v1/zz_generated.deepcopy.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

76 changes: 76 additions & 0 deletions deploy/charts/prefect-operator/crds/prefect.io_prefectservers.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -1394,6 +1394,82 @@ spec:
- name
type: object
type: array
extraServicePorts:
description: ExtraServicePorts defines additional ports to expose
on the Prefect Server Service
items:
description: ServicePort contains information on service's port.
properties:
appProtocol:
description: |-
The application protocol for this port.
This is used as a hint for implementations to offer richer behavior for protocols that they understand.
This field follows standard Kubernetes label syntax.
Valid values are either:
* Un-prefixed protocol names - reserved for IANA standard service names (as per
RFC-6335 and https://www.iana.org/assignments/service-names).
* Kubernetes-defined prefixed names:
* 'kubernetes.io/h2c' - HTTP/2 prior knowledge over cleartext as described in https://www.rfc-editor.org/rfc/rfc9113.html#name-starting-http-2-with-prior-
* 'kubernetes.io/ws' - WebSocket over cleartext as described in https://www.rfc-editor.org/rfc/rfc6455
* 'kubernetes.io/wss' - WebSocket over TLS as described in https://www.rfc-editor.org/rfc/rfc6455
* Other protocols should use implementation-defined prefixed names such as
mycompany.com/my-custom-protocol.
type: string
name:
description: |-
The name of this port within the service. This must be a DNS_LABEL.
All ports within a ServiceSpec must have unique names. When considering
the endpoints for a Service, this must match the 'name' field in the
EndpointPort.
Optional if only one ServicePort is defined on this service.
type: string
nodePort:
description: |-
The port on each node on which this service is exposed when type is
NodePort or LoadBalancer. Usually assigned by the system. If a value is
specified, in-range, and not in use it will be used, otherwise the
operation will fail. If not specified, a port will be allocated if this
Service requires one. If this field is specified when creating a
Service which does not need it, creation will fail. This field will be
wiped when updating a Service to no longer need it (e.g. changing type
from NodePort to ClusterIP).
More info: https://kubernetes.io/docs/concepts/services-networking/service/#type-nodeport
format: int32
type: integer
port:
description: The port that will be exposed by this service.
format: int32
type: integer
protocol:
default: TCP
description: |-
The IP protocol for this port. Supports "TCP", "UDP", and "SCTP".
Default is TCP.
type: string
targetPort:
anyOf:
- type: integer
- type: string
description: |-
Number or name of the port to access on the pods targeted by the service.
Number must be in the range 1 to 65535. Name must be an IANA_SVC_NAME.
If this is a string, it will be looked up as a named port in the
target Pod's container ports. If this is not specified, the value
of the 'port' field is used (an identity map).
This field is ignored for services with clusterIP=None, and should be
omitted or set equal to the 'port' field.
More info: https://kubernetes.io/docs/concepts/services-networking/service/#defining-a-service
x-kubernetes-int-or-string: true
required:
- port
type: object
type: array
image:
description: Image defines the exact image to deploy for the Prefect
Server, overriding Version
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -68,7 +68,7 @@ spec:
type: object
extraContainers:
description: ExtraContainers defines additional containers to add
to the Prefect Server Deployment
to each worker in the Work Pool
items:
description: A single application container that you want to run
within a pod.
Expand Down
4 changes: 4 additions & 0 deletions internal/controller/prefectserver_controller.go
Original file line number Diff line number Diff line change
Expand Up @@ -585,6 +585,10 @@ func (r *PrefectServerReconciler) prefectServerService(server *prefectiov1.Prefe
},
},
}
// Append any extra ports into the Service if configured.
if server.Spec.ExtraServicePorts != nil {
service.Spec.Ports = append(service.Spec.Ports, server.Spec.ExtraServicePorts...)
}

// TODO: handle errors from SetControllerReference.
_ = ctrl.SetControllerReference(server, &service, r.Scheme)
Expand Down
37 changes: 37 additions & 0 deletions internal/controller/prefectserver_controller_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -550,6 +550,43 @@ var _ = Describe("PrefectServer controller", func() {
container := deployment.Spec.Template.Spec.Containers[1]
Expect(container.Name).To(Equal("extra-container"))
})

It("should update the Service with the extra port", func() {
// Update the PrefectServer with an extra port
Expect(k8sClient.Get(ctx, name, prefectserver)).To(Succeed())
prefectserver.Spec.ExtraServicePorts = []corev1.ServicePort{
{
Name: "extra-port",
Port: 4300,
TargetPort: intstr.FromString("extra-port"),
Protocol: corev1.ProtocolTCP,
},
}

Expect(k8sClient.Update(ctx, prefectserver)).To(Succeed())
// Reconcile to apply the changes
controllerReconciler := &PrefectServerReconciler{
Client: k8sClient,
Scheme: k8sClient.Scheme(),
}
_, err := controllerReconciler.Reconcile(ctx, reconcile.Request{
NamespacedName: name,
})
Expect(err).NotTo(HaveOccurred())

// Check if the Service was updated with the extra port
service := &corev1.Service{}
Eventually(func() error {
return k8sClient.Get(ctx, types.NamespacedName{
Namespace: namespaceName,
Name: prefectserver.Name,
}, service)
}).Should(Succeed())

Expect(service.Spec.Ports).To(HaveLen(2))
port := service.Spec.Ports[1]
Expect(port.Name).To(Equal("extra-port"))
})
})

Context("When evaluating changes with any server", func() {
Expand Down

0 comments on commit 791a989

Please sign in to comment.