-
Notifications
You must be signed in to change notification settings - Fork 600
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
upgrader #3110
upgrader #3110
Changes from all commits
f7db861
459aa1d
7911d41
c18b303
7258551
4c02c56
1fc6cb3
ce330f8
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,46 @@ | ||
/* | ||
Copyright 2020 The Knative Authors | ||
|
||
Licensed under the Apache License, Version 2.0 (the "License"); | ||
you may not use this file except in compliance with the License. | ||
You may obtain a copy of the License at | ||
|
||
http://www.apache.org/licenses/LICENSE-2.0 | ||
|
||
Unless required by applicable law or agreed to in writing, software | ||
distributed under the License is distributed on an "AS IS" BASIS, | ||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
See the License for the specific language governing permissions and | ||
limitations under the License. | ||
*/ | ||
|
||
package main | ||
|
||
import ( | ||
// Uncomment the following line to load the gcp plugin (only required to authenticate against GKE clusters). | ||
// _ "k8s.io/client-go/plugin/pkg/client/auth/gcp" | ||
|
||
"context" | ||
"fmt" | ||
"os" | ||
|
||
"k8s.io/client-go/kubernetes" | ||
kubeclient "knative.dev/pkg/client/injection/kube/client" | ||
"knative.dev/pkg/injection/sharedmain" | ||
"knative.dev/pkg/signals" | ||
|
||
versioned "knative.dev/eventing/pkg/client/clientset/versioned" | ||
eventingclient "knative.dev/eventing/pkg/client/injection/client" | ||
broker "knative.dev/eventing/pkg/upgrader/broker/v0.15.0" | ||
) | ||
|
||
func main() { | ||
ctx := signals.NewContext() | ||
cfg := sharedmain.ParseAndGetConfigOrDie() | ||
ctx = context.WithValue(ctx, kubeclient.Key{}, kubernetes.NewForConfigOrDie(cfg)) | ||
ctx = context.WithValue(ctx, eventingclient.Key{}, versioned.NewForConfigOrDie(cfg)) | ||
if err := broker.Upgrade(ctx); err != nil { | ||
fmt.Printf("Broker Upgrade failed with: %v\n", err) | ||
os.Exit(1) | ||
} | ||
} |
Original file line number | Diff line number | Diff line change | ||||||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
@@ -0,0 +1,116 @@ | ||||||||||||||||||||||
# Upgrade script (optional) to upgrade to v0.15.0 of Eventing | ||||||||||||||||||||||
|
||||||||||||||||||||||
This directory contains a job that updates all of v1alpha1 based Brokers that | ||||||||||||||||||||||
are using Spec.ChannelTemplate to specify which channel they use. You are not | ||||||||||||||||||||||
able to create new ones specifying the ChannelTemplate, and you can either | ||||||||||||||||||||||
manually update them yourself, or run the tool provided here, which will do the | ||||||||||||||||||||||
following for any Broker that is using Spec.ChannelTemplate: | ||||||||||||||||||||||
|
||||||||||||||||||||||
1. Creates a ConfigMap in the same namespace as the Broker named: | ||||||||||||||||||||||
`broker-upgrade-auto-gen-config-<brokername>` with the content from | ||||||||||||||||||||||
Spec.ChannelTemplate. | ||||||||||||||||||||||
1. Set Broker Spec.Config to point to this ConfigMap | ||||||||||||||||||||||
1. Set Broker Spec.ChannelTemplate to nil | ||||||||||||||||||||||
|
||||||||||||||||||||||
To run the upgrade script: | ||||||||||||||||||||||
|
||||||||||||||||||||||
```shell | ||||||||||||||||||||||
kubectl apply -f https://github.com/knative/eventing/releases/download/v0.15.0/upgrade-to-v0.15.0.yaml | ||||||||||||||||||||||
``` | ||||||||||||||||||||||
|
||||||||||||||||||||||
It will create a job called v0.15.0-upgrade in the knative-eventing namespace. | ||||||||||||||||||||||
If you installed to different namespace, you need to modify the upgrade.yaml | ||||||||||||||||||||||
appropriately. Also the job by default runs as `eventing-controller` service | ||||||||||||||||||||||
account, you can also modify that but the service account will need to have | ||||||||||||||||||||||
permissions to list `Namespace`s, list and patch `Broker`s, and create | ||||||||||||||||||||||
`ConfigMap`s. | ||||||||||||||||||||||
|
||||||||||||||||||||||
# Examples | ||||||||||||||||||||||
|
||||||||||||||||||||||
If for example you have an existing v1lpha1 Broker with the following Spec: | ||||||||||||||||||||||
|
||||||||||||||||||||||
```yaml | ||||||||||||||||||||||
spec: | ||||||||||||||||||||||
channelTemplateSpec: | ||||||||||||||||||||||
apiVersion: messaging.knative.dev/v1alpha1 | ||||||||||||||||||||||
kind: InMemoryChannel | ||||||||||||||||||||||
``` | ||||||||||||||||||||||
|
||||||||||||||||||||||
The tool will create a ConfigMap that looks like so: | ||||||||||||||||||||||
|
||||||||||||||||||||||
```yaml | ||||||||||||||||||||||
apiVersion: v1 | ||||||||||||||||||||||
data: | ||||||||||||||||||||||
channelTemplateSpec: |2 | ||||||||||||||||||||||
|
||||||||||||||||||||||
apiVersion: "messaging.knative.dev/v1alpha1" | ||||||||||||||||||||||
kind: "InMemoryChannel" | ||||||||||||||||||||||
kind: ConfigMap | ||||||||||||||||||||||
metadata: | ||||||||||||||||||||||
name: broker-upgrade-auto-gen-config-newbroker | ||||||||||||||||||||||
namespace: test-broker-6 | ||||||||||||||||||||||
ownerReferences: | ||||||||||||||||||||||
- apiVersion: eventing.knative.dev/v1alpha1 | ||||||||||||||||||||||
blockOwnerDeletion: true | ||||||||||||||||||||||
controller: true | ||||||||||||||||||||||
kind: Broker | ||||||||||||||||||||||
name: newbroker | ||||||||||||||||||||||
Comment on lines
+53
to
+57
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Format markdown:
Suggested change
|
||||||||||||||||||||||
``` | ||||||||||||||||||||||
|
||||||||||||||||||||||
And your Broker will then look like this after the upgrade: | ||||||||||||||||||||||
|
||||||||||||||||||||||
```yaml | ||||||||||||||||||||||
spec: | ||||||||||||||||||||||
config: | ||||||||||||||||||||||
apiVersion: v1 | ||||||||||||||||||||||
kind: ConfigMap | ||||||||||||||||||||||
name: broker-upgrade-auto-gen-config-newbroker | ||||||||||||||||||||||
namespace: test-broker-6 | ||||||||||||||||||||||
``` | ||||||||||||||||||||||
|
||||||||||||||||||||||
For KafkaChannels it might look like something like this: | ||||||||||||||||||||||
|
||||||||||||||||||||||
```yaml | ||||||||||||||||||||||
spec: | ||||||||||||||||||||||
channelTemplateSpec: | ||||||||||||||||||||||
apiVersion: messaging.knative.dev/v1alpha1 | ||||||||||||||||||||||
kind: KafkaChannel | ||||||||||||||||||||||
spec: | ||||||||||||||||||||||
numPartitions: 1 | ||||||||||||||||||||||
replicationFactor: 1 | ||||||||||||||||||||||
``` | ||||||||||||||||||||||
|
||||||||||||||||||||||
The resulting ConfigMap will be: | ||||||||||||||||||||||
|
||||||||||||||||||||||
```yaml | ||||||||||||||||||||||
apiVersion: v1 | ||||||||||||||||||||||
data: | ||||||||||||||||||||||
channelTemplateSpec: |2 | ||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Is this 2 intentional? |
||||||||||||||||||||||
|
||||||||||||||||||||||
apiVersion: "messaging.knative.dev/v1alpha1" | ||||||||||||||||||||||
kind: "KafkaChannel" | ||||||||||||||||||||||
spec: | ||||||||||||||||||||||
numPartitions: 1 | ||||||||||||||||||||||
replicationFactor: 1 | ||||||||||||||||||||||
kind: ConfigMap | ||||||||||||||||||||||
metadata: | ||||||||||||||||||||||
name: broker-upgrade-auto-gen-config-newbroker-kafka | ||||||||||||||||||||||
namespace: test-broker-6 | ||||||||||||||||||||||
ownerReferences: | ||||||||||||||||||||||
- apiVersion: eventing.knative.dev/v1alpha1 | ||||||||||||||||||||||
blockOwnerDeletion: true | ||||||||||||||||||||||
controller: true | ||||||||||||||||||||||
kind: Broker | ||||||||||||||||||||||
name: newbroker-kafka | ||||||||||||||||||||||
Comment on lines
+100
to
+104
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Format markdown:
Suggested change
|
||||||||||||||||||||||
``` | ||||||||||||||||||||||
|
||||||||||||||||||||||
And the Broker will look like this: | ||||||||||||||||||||||
|
||||||||||||||||||||||
```yaml | ||||||||||||||||||||||
spec: | ||||||||||||||||||||||
config: | ||||||||||||||||||||||
apiVersion: v1 | ||||||||||||||||||||||
kind: ConfigMap | ||||||||||||||||||||||
name: broker-upgrade-auto-gen-config-newbroker-kafka | ||||||||||||||||||||||
namespace: test-broker-6 | ||||||||||||||||||||||
``` |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
apiVersion: batch/v1 | ||
kind: Job | ||
metadata: | ||
name: v0.15.0-upgrade | ||
namespace: knative-eventing | ||
labels: | ||
eventing.knative.dev/release: devel | ||
spec: | ||
template: | ||
metadata: | ||
annotations: | ||
sidecar.istio.io/inject: "false" | ||
spec: | ||
serviceAccountName: eventing-controller | ||
restartPolicy: Never | ||
containers: | ||
- name: upgrade-brokers | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. perhaps the name is a bit too generic ? It's same as here: 🤔 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It's just the container, I'd rather leave it and not bake the version in there as well? It's already in the job. |
||
image: ko://knative.dev/eventing/cmd/upgrade/v0.15.0 |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,182 @@ | ||
/* | ||
Copyright 2020 The Knative Authors | ||
|
||
Licensed under the Apache License, Version 2.0 (the "License"); | ||
you may not use this file except in compliance with the License. | ||
You may obtain a copy of the License at | ||
|
||
http://www.apache.org/licenses/LICENSE-2.0 | ||
|
||
Unless required by applicable law or agreed to in writing, software | ||
distributed under the License is distributed on an "AS IS" BASIS, | ||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
See the License for the specific language governing permissions and | ||
limitations under the License. | ||
*/ | ||
|
||
package broker | ||
|
||
import ( | ||
"context" | ||
"fmt" | ||
"strings" | ||
|
||
"github.com/ghodss/yaml" | ||
// "gopkg.in/yaml.v2" | ||
corev1 "k8s.io/api/core/v1" | ||
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" | ||
types "k8s.io/apimachinery/pkg/types" | ||
|
||
"knative.dev/eventing/pkg/apis/eventing/v1alpha1" | ||
eventingclient "knative.dev/eventing/pkg/client/injection/client" | ||
"knative.dev/pkg/apis/duck" | ||
duckv1 "knative.dev/pkg/apis/duck/v1" | ||
kubeclient "knative.dev/pkg/client/injection/kube/client" | ||
"knative.dev/pkg/kmeta" | ||
"knative.dev/pkg/logging" | ||
) | ||
|
||
// Upgrade upgrades all the brokers by seeing if they have Spec.ChannelTemplate | ||
// and if they do, create a ConfigMap in that namespace with the contents of the | ||
// Spec.ChannelTemplate, that's owned by the Broker, then update the Broker | ||
// to have Spec.Config to point to the newly created ConfigMap and remove | ||
// the ChannelTemplate. | ||
func Upgrade(ctx context.Context) error { | ||
logger := logging.FromContext(ctx) | ||
|
||
nsClient := kubeclient.Get(ctx).CoreV1().Namespaces() | ||
namespaces, err := nsClient.List(metav1.ListOptions{}) | ||
if err != nil { | ||
logger.Warnf("Failed to list namespaces: %v", err) | ||
return err | ||
} | ||
for _, ns := range namespaces.Items { | ||
err = processNamespace(ctx, ns.Name) | ||
if err != nil { | ||
return err | ||
} | ||
} | ||
return nil | ||
} | ||
|
||
func processNamespace(ctx context.Context, ns string) error { | ||
logger := logging.FromContext(ctx) | ||
logger.Infof("Processing Brokers in namespace: %q", ns) | ||
|
||
eventingClient := eventingclient.Get(ctx) | ||
brokerClient := eventingClient.EventingV1alpha1().Brokers(ns) | ||
brokers, err := brokerClient.List(metav1.ListOptions{}) | ||
if err != nil { | ||
logger.Warnf("Failed to list brokers for namespace %q: %v", ns, err) | ||
return err | ||
} | ||
for _, broker := range brokers.Items { | ||
patch, err := processBroker(ctx, broker) | ||
if err != nil { | ||
logger.Warnf("Failed to process a Broker \"%s/%s\" : %v", broker.Namespace, broker.Name, err) | ||
return err | ||
} | ||
if len(patch) == 0 { | ||
logger.Infof("Broker \"%s/%s\" does not require updating", broker.Namespace, broker.Name) | ||
continue | ||
} | ||
|
||
// Ok, there are differences, apply the patch | ||
logger.Infof("Patching Broker \"%s/%s\" with %q", broker.Namespace, broker.Name, string(patch)) | ||
patched, err := brokerClient.Patch(broker.Name, types.MergePatchType, patch) | ||
if err != nil { | ||
logger.Warnf("Failed to patch \"%s/%s\" : %v", broker.Namespace, broker.Name, err) | ||
return err | ||
} | ||
logger.Infof("Patched \"%s/%s\" successfully new Spec: %+v", broker.Namespace, broker.Name, patched.Spec) | ||
} | ||
return nil | ||
} | ||
|
||
// Process a broker, create a ConfigMap representing the ChannelTemplate | ||
// and point the Config to it. | ||
// Returns non-empty patch bytes if a patch is necessary. | ||
func processBroker(ctx context.Context, broker v1alpha1.Broker) ([]byte, error) { | ||
logger := logging.FromContext(ctx) | ||
if broker.Spec.ChannelTemplate == nil || broker.Spec.ChannelTemplate.Kind == "" { | ||
logger.Infof("Broker \"%s/%s\" is not using channeltemplate, skipping...", broker.Namespace, broker.Name) | ||
return []byte{}, nil | ||
} | ||
|
||
modified := broker.DeepCopy() | ||
|
||
// If the Broker already has a Config, don't modify it, just nil out the channel template. | ||
if broker.Spec.Config == nil || broker.Spec.Config.Name == "" { | ||
cm, err := createConfigMap(ctx, broker) | ||
if err != nil { | ||
return []byte{}, err | ||
} | ||
modified.Spec.Config = cm | ||
} | ||
modified.Spec.ChannelTemplate = nil | ||
|
||
patch, err := duck.CreateMergePatch(broker, modified) | ||
if err != nil { | ||
logger.Warnf("Failed to create patch for \"%s/%s\" : %v", broker.Namespace, broker.Name, err) | ||
return []byte{}, err | ||
} | ||
logger.Infof("Patch for \"%s/%s\": %q", broker.Namespace, broker.Name, string(patch)) | ||
// If there is nothing to patch, we are good, just return. | ||
// Empty patch is {}, hence we check for that. | ||
if len(patch) <= 2 { | ||
return []byte{}, nil | ||
} | ||
return patch, nil | ||
} | ||
|
||
func createConfigMap(ctx context.Context, broker v1alpha1.Broker) (*duckv1.KReference, error) { | ||
// Generating the spec portion is a bit goofy cause we have turn the runtime raw into | ||
// a yaml blob that we stick into the configmap (with proper indentation). | ||
data := "" | ||
if broker.Spec.ChannelTemplate.Spec != nil { | ||
bytes, err := yaml.JSONToYAML(broker.Spec.ChannelTemplate.Spec.Raw) | ||
if err != nil { | ||
return nil, err | ||
} | ||
logging.FromContext(ctx).Infof("BYTES: %q", string(bytes)) | ||
data = fmt.Sprintf(` | ||
apiVersion: %q | ||
kind: %q | ||
spec: | ||
%s | ||
`, broker.Spec.ChannelTemplate.APIVersion, broker.Spec.ChannelTemplate.Kind, strings.ReplaceAll(strings.TrimSpace(string(bytes)), "\n", "\n ")) | ||
} else { | ||
data = fmt.Sprintf(` | ||
apiVersion: %q | ||
kind: %q | ||
`, broker.Spec.ChannelTemplate.APIVersion, broker.Spec.ChannelTemplate.Kind) | ||
} | ||
|
||
cm := &corev1.ConfigMap{ | ||
TypeMeta: metav1.TypeMeta{ | ||
Kind: "ConfigMap", | ||
APIVersion: "v1", | ||
}, | ||
ObjectMeta: metav1.ObjectMeta{ | ||
Name: "broker-upgrade-auto-gen-config-" + broker.Name, | ||
Namespace: broker.Namespace, | ||
OwnerReferences: []metav1.OwnerReference{ | ||
*kmeta.NewControllerRef(&broker), | ||
}, | ||
}, | ||
Data: map[string]string{"channelTemplateSpec": data}, | ||
} | ||
|
||
logging.FromContext(ctx).Infof("Creating configmap: %+v", cm) | ||
_, err := kubeclient.Get(ctx).CoreV1().ConfigMaps(broker.Namespace).Create(cm) | ||
if err != nil { | ||
logging.FromContext(ctx).Errorf("Failed to create broker config map \"%s/%s\": %v", broker.Namespace, broker.Name, err) | ||
return nil, err | ||
} | ||
return &duckv1.KReference{ | ||
APIVersion: "v1", | ||
Kind: "ConfigMap", | ||
Namespace: broker.Namespace, | ||
Name: "broker-upgrade-auto-gen-config-" + broker.Name, | ||
}, 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.
Is this 2 intentional?
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.
Yeah, it's because of showing the block there:
https://yaml.org/spec/1.2/spec.html#id2793979