mirror of
				https://github.com/1Password/onepassword-operator.git
				synced 2025-10-31 19:59:40 +00:00 
			
		
		
		
	Compare commits
	
		
			25 Commits
		
	
	
		
			v1.0.1
			...
			update-pat
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 1590dd9b89 | ||
|   | 49d984c6f2 | ||
|   | 72cad7284c | ||
|   | 6043e0da0b | ||
|   | 753cc5e9a3 | ||
|   | 8cfe98073e | ||
|   | 96b42e7c52 | ||
|   | 579b5848da | ||
|   | b50d864b50 | ||
|   | 1643385d9b | ||
|   | 9441214733 | ||
|   | 7e4e988813 | ||
|   | 68f084080e | ||
|   | 859c9e3462 | ||
|   | 9dabac4a55 | ||
|   | d927a08790 | ||
|   | 933f7c4e2c | ||
|   | 81eb9a521f | ||
|   | eb32bd7f94 | ||
|   | a5781af949 | ||
|   | 0aa5781acd | ||
|   | 700be4426f | ||
|   | 76ef9aa372 | ||
|   | d7e6704314 | ||
|   | 2443979602 | 
							
								
								
									
										7
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -28,6 +28,9 @@ jobs: | |||||||
|           tags: | |           tags: | | ||||||
|             type=semver,pattern={{version}} |             type=semver,pattern={{version}} | ||||||
|             type=semver,pattern={{major}}.{{minor}} |             type=semver,pattern={{major}}.{{minor}} | ||||||
|  |       - name: Get the version from tag | ||||||
|  |         id: get_version | ||||||
|  |         run: echo ::set-output name=VERSION::${GITHUB_REF#refs/tags/v} | ||||||
|       - |       - | ||||||
|         name: Set up QEMU |         name: Set up QEMU | ||||||
|         uses: docker/setup-qemu-action@v1 |         uses: docker/setup-qemu-action@v1 | ||||||
| @@ -46,9 +49,9 @@ jobs: | |||||||
|         with: |         with: | ||||||
|           context: . |           context: . | ||||||
|           file: Dockerfile |           file: Dockerfile | ||||||
|           platforms: linux/amd64,linux/arm64 |           platforms: linux/amd64,linux/arm64,linux/arm/v7 | ||||||
|           push: true |           push: true | ||||||
|           tags: ${{ steps.meta.outputs.tags }} |           tags: ${{ steps.meta.outputs.tags }} | ||||||
|           labels: ${{ steps.meta.outputs.labels }} |           labels: ${{ steps.meta.outputs.labels }} | ||||||
|           build-args: | |           build-args: | | ||||||
|             operator_version=${{ github.event.ref }} |             operator_version=${{ steps.get_version.outputs.VERSION }} | ||||||
|   | |||||||
| @@ -12,6 +12,14 @@ | |||||||
|  |  | ||||||
| --- | --- | ||||||
|  |  | ||||||
|  | [//]: # (START/v1.0.2) | ||||||
|  | # v1.0.2 | ||||||
|  |  | ||||||
|  | ## Fixes | ||||||
|  |  * Name normalizer added to handle non-conforming item names. | ||||||
|  |  | ||||||
|  | --- | ||||||
|  |  | ||||||
| [//]: # (START/v1.0.1) | [//]: # (START/v1.0.1) | ||||||
| # v1.0.1 | # v1.0.1 | ||||||
|  |  | ||||||
|   | |||||||
| @@ -16,7 +16,7 @@ ARG operator_version=dev | |||||||
| RUN CGO_ENABLED=0 \ | RUN CGO_ENABLED=0 \ | ||||||
|     GO111MODULE=on \ |     GO111MODULE=on \ | ||||||
|     go build \ |     go build \ | ||||||
|     -ldflags "-X version.Version=$operator_version" \ |     -ldflags "-X \"github.com/1Password/onepassword-operator/version.Version=$operator_version\"" \ | ||||||
|     -mod vendor \ |     -mod vendor \ | ||||||
|     -a -o manager main.go |     -a -o manager main.go | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								Makefile
									
									
									
									
									
								
							| @@ -20,12 +20,12 @@ test/coverage:	## Run test suite with coverage report | |||||||
| 	go test -v ./... -cover | 	go test -v ./... -cover | ||||||
|  |  | ||||||
| build:	## Build operator Docker image | build:	## Build operator Docker image | ||||||
| 	@docker build -f Dockerfile --build-arg operator_version=$(curVersion) -t $(DOCKER_IMG_TAG) | 	@docker build -f Dockerfile --build-arg operator_version=$(curVersion) -t $(DOCKER_IMG_TAG) . | ||||||
| 	@echo "Successfully built and tagged image." | 	@echo "Successfully built and tagged image." | ||||||
| 	@echo "Tag: $(DOCKER_IMG_TAG)" | 	@echo "Tag: $(DOCKER_IMG_TAG)" | ||||||
|  |  | ||||||
| build/local:	## Build local version of the operator Docker image | build/local:	## Build local version of the operator Docker image | ||||||
| 	@docker build -f Dockerfile -t local/$(DOCKER_IMG_TAG) | 	@docker build -f Dockerfile -t local/$(DOCKER_IMG_TAG) . | ||||||
|  |  | ||||||
| build/binary: clean	## Build operator binary | build/binary: clean	## Build operator binary | ||||||
| 	@mkdir -p dist | 	@mkdir -p dist | ||||||
|   | |||||||
							
								
								
									
										40
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										40
									
								
								README.md
									
									
									
									
									
								
							| @@ -2,7 +2,7 @@ | |||||||
|  |  | ||||||
| The 1Password Connect Kubernetes Operator provides the ability to integrate Kubernetes with 1Password. This Operator manages `OnePasswordItem` Custom Resource Definitions (CRDs) that define the location of an Item stored in 1Password. The `OnePasswordItem` CRD, when created, will be used to compose a Kubernetes Secret containing the contents of the specified item. | The 1Password Connect Kubernetes Operator provides the ability to integrate Kubernetes with 1Password. This Operator manages `OnePasswordItem` Custom Resource Definitions (CRDs) that define the location of an Item stored in 1Password. The `OnePasswordItem` CRD, when created, will be used to compose a Kubernetes Secret containing the contents of the specified item. | ||||||
|  |  | ||||||
| The 1Password Connect Kubernetes Operator also allows for Kubernetes Secrets to be composed from a 1Password Item through annotation of an Item Path on a deployment. | The 1Password Connect Kubernetes Operator also allows for Kubernetes Secrets to be composed from a 1Password Item through annotation of an Item Reference on a deployment. | ||||||
|  |  | ||||||
| The 1Password Connect Kubernetes Operator will continually check for updates from 1Password for any Kubernetes Secret that it has generated. If a Kubernetes Secret is updated, any Deployment using that secret can be automatically restarted. | The 1Password Connect Kubernetes Operator will continually check for updates from 1Password for any Kubernetes Secret that it has generated. If a Kubernetes Secret is updated, any Deployment using that secret can be automatically restarted. | ||||||
|  |  | ||||||
| @@ -13,8 +13,8 @@ Prerequisites: | |||||||
| - [1Password Command Line Tool Installed](https://1password.com/downloads/command-line/) | - [1Password Command Line Tool Installed](https://1password.com/downloads/command-line/) | ||||||
| - [kubectl installed](https://kubernetes.io/docs/tasks/tools/install-kubectl/) | - [kubectl installed](https://kubernetes.io/docs/tasks/tools/install-kubectl/) | ||||||
| - [docker installed](https://docs.docker.com/get-docker/) | - [docker installed](https://docs.docker.com/get-docker/) | ||||||
| - [Generated a 1password-credentials.json file and issued a 1Password Connect API Token for the K8s Operator integration](https://support.b5dev.com/cs/connect) | - [Generated a 1password-credentials.json file and issued a 1Password Connect API Token for the K8s Operator integration](https://support.1password.com/secrets-automation/) | ||||||
| - [1Password Connect deployed to Kubernetes](https://support.b5dev.com/cs/connect-deploy-kubernetes/#step-2-deploy-a-connect-server). **NOTE**: If customization of the 1Password Connect deployment is not required you can skip this prerequisite. | - [1Password Connect deployed to Kubernetes](https://support.1password.com/connect-deploy-kubernetes/#step-2-deploy-a-1password-connect-server). **NOTE**: If customization of the 1Password Connect deployment is not required you can skip this prerequisite. | ||||||
|  |  | ||||||
| ### Quickstart for Deploying 1Password Connect to Kubernetes | ### Quickstart for Deploying 1Password Connect to Kubernetes | ||||||
|  |  | ||||||
| @@ -53,15 +53,15 @@ Adding this environment variable will have the operator automatically deploy a d | |||||||
| "Create a Connect token for the operator and save it as a Kubernetes Secret:  | "Create a Connect token for the operator and save it as a Kubernetes Secret:  | ||||||
|  |  | ||||||
| ```bash | ```bash | ||||||
| $ kubectl create secret generic op-operator-connect-token --from-literal=token=<OP_CONNECT_TOKEN>" | $ kubectl create secret generic onepassword-token --from-literal=token=<OP_CONNECT_TOKEN>" | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| If you do not have a token for the operator, you can generate a token and save it to kubernetes with the following command: | If you do not have a token for the operator, you can generate a token and save it to kubernetes with the following command: | ||||||
| ```bash | ```bash | ||||||
| $ kubectl create secret generic op-operator-connect-token --from-literal=token=$(op create connect token <server> op-k8s-operator --vault <vault>) | $ kubectl create secret generic onepassword-token --from-literal=token=$(op create connect token <server> op-k8s-operator --vault <vault>) | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| [More information on generating a token can be found here](https://support.1password.com/cs/secrets-automation/#appendix-issue-additional-access-tokens) | [More information on generating a token can be found here](https://support.1password.com/secrets-automation/#appendix-issue-additional-access-tokens) | ||||||
|  |  | ||||||
| **Set Permissions For Operator** | **Set Permissions For Operator** | ||||||
|  |  | ||||||
| @@ -84,9 +84,9 @@ An sample Deployment yaml can be found at `/deploy/operator.yaml`. | |||||||
|  |  | ||||||
| To further configure the 1Password Kubernetes Operator the Following Environment variables can be set in the operator yaml: | To further configure the 1Password Kubernetes Operator the Following Environment variables can be set in the operator yaml: | ||||||
|  |  | ||||||
| - **WATCH_NAMESPACE:** comma separated list of what Namespaces to watch for changes. |  | ||||||
| - **OP_CONNECT_HOST** (required): Specifies the host name within Kubernetes in which to access the 1Password Connect. | - **OP_CONNECT_HOST** (required): Specifies the host name within Kubernetes in which to access the 1Password Connect. | ||||||
| - **POLLING_INTERVAL** (default: 600)**:** The number of seconds the 1Password Kubernetes Operator will wait before checking for updates from 1Password Connect. | - **WATCH_NAMESPACE:** (default: watch all namespaces): Comma separated list of what Namespaces to watch for changes. | ||||||
|  | - **POLLING_INTERVAL** (default: 600): The number of seconds the 1Password Kubernetes Operator will wait before checking for updates from 1Password Connect. | ||||||
| - **MANAGE_CONNECT** (default: false): If set to true, on deployment of the operator, a default configuration of the OnePassword Connect Service will be deployed to the `default` namespace. | - **MANAGE_CONNECT** (default: false): If set to true, on deployment of the operator, a default configuration of the OnePassword Connect Service will be deployed to the `default` namespace. | ||||||
| - **AUTO_RESTART** (default: false): If set to true, the operator will restart any deployment using a secret from 1Password Connect. This can be overwritten by namespace, deployment, or individual secret. More details on AUTO_RESTART can be found in the ["Configuring Automatic Rolling Restarts of Deployments"](#configuring-automatic-rolling-restarts-of-deployments) section. | - **AUTO_RESTART** (default: false): If set to true, the operator will restart any deployment using a secret from 1Password Connect. This can be overwritten by namespace, deployment, or individual secret. More details on AUTO_RESTART can be found in the ["Configuring Automatic Rolling Restarts of Deployments"](#configuring-automatic-rolling-restarts-of-deployments) section. | ||||||
|  |  | ||||||
| @@ -106,7 +106,7 @@ kind: OnePasswordItem | |||||||
| metadata: | metadata: | ||||||
|   name: <item_name> #this name will also be used for naming the generated kubernetes secret |   name: <item_name> #this name will also be used for naming the generated kubernetes secret | ||||||
| spec: | spec: | ||||||
|   itemPath: "vaults/<vault_id_or_title>/items/<item_id_or_title>"  |   itemReference: "op://<vault_id_or_title>/<item_id_or_title>"  | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| Deploy the OnePasswordItem to Kubernetes: | Deploy the OnePasswordItem to Kubernetes: | ||||||
| @@ -131,20 +131,25 @@ kind: Deployment | |||||||
| metadata: | metadata: | ||||||
|   name: deployment-example |   name: deployment-example | ||||||
|   annotations: |   annotations: | ||||||
|     operator.1password.io/item-path: "vaults/<vault_id_or_title>/items/<item_id_or_title>" |     operator.1password.io/item-reference: "op://<vault>/<item>" | ||||||
|     operator.1password.io/item-name: "<secret_name>" |     operator.1password.io/item-name: "<secret_name>" | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| Applying this yaml file will create a Kubernetes Secret with the name `<secret_name>` and contents from the location specified at the specified Item Path. | Applying this yaml file will create a Kubernetes Secret with the name `<secret_name>` and contents from the location specified at the specified Item Reference. | ||||||
|  |  | ||||||
| Note: Deleting the Deployment that you've created will automatically delete the created Kubernetes Secret only if the deployment is still annotated with `operator.1password.io/item-path` and `operator.1password.io/item-name` and no other deployment is using the secret. | Note: Deleting the Deployment that you've created will automatically delete the created Kubernetes Secret only if the deployment is still annotated with `operator.1password.io/item-reference` and `operator.1password.io/item-name` and no other deployment is using the secret. | ||||||
|  |  | ||||||
| If a 1Password Item that is linked to a Kubernetes Secret is updated within the POLLING_INTERVAL the associated Kubernetes Secret will be updated. However, if you do not want a specific secret to be updated you can add the tag `operator.1password.io:ignore-secret` to the item stored in 1Password. While this tag is in place, any updates made to an item will not trigger an update to the associated secret in Kubernetes. | If a 1Password Item that is linked to a Kubernetes Secret is updated within the POLLING_INTERVAL the associated Kubernetes Secret will be updated. However, if you do not want a specific secret to be updated you can add the tag `operator.1password.io:ignore-secret` to the item stored in 1Password. While this tag is in place, any updates made to an item will not trigger an update to the associated secret in Kubernetes. | ||||||
|  |  | ||||||
| --- | --- | ||||||
| **NOTE** | **NOTE** | ||||||
|  |  | ||||||
| If multiple 1Password vaults/items have the same `title` when using a title in the access path, the desired action will be performed on the oldest vault/item. Furthermore, titles that include white space characters cannot be used. | If multiple 1Password vaults/items have the same `title` when using a title in the access reference, the desired action will be performed on the oldest vault/item.  | ||||||
|  |  | ||||||
|  | Titles and field names that include white space and other characters that are not a valid [DNS subdomain name](https://kubernetes.io/docs/concepts/configuration/secret/) will create Kubernetes secrets that have titles and fields in the following format: | ||||||
|  |  - Invalid characters before the first alphanumeric character and after the last alphanumeric character will be removed | ||||||
|  |  - All whitespaces between words will be replaced by `-` | ||||||
|  |  - All the letters will be lower-cased. | ||||||
|  |  | ||||||
| --- | --- | ||||||
|  |  | ||||||
| @@ -163,7 +168,8 @@ apiVersion: v1 | |||||||
| kind: Namespace | kind: Namespace | ||||||
| metadata: | metadata: | ||||||
|   name: "example-namespace" |   name: "example-namespace" | ||||||
|   operator.1password.io/auto-restart: "true" |   annotations: | ||||||
|  |     operator.1password.io/auto-restart: "true" | ||||||
| ``` | ``` | ||||||
| If the value is not set, the auto reset settings on the operator will be used. This value can be overwritten by deployment. | If the value is not set, the auto reset settings on the operator will be used. This value can be overwritten by deployment. | ||||||
|  |  | ||||||
| @@ -175,7 +181,8 @@ apiVersion: v1 | |||||||
| kind: Deployment | kind: Deployment | ||||||
| metadata: | metadata: | ||||||
|   name: "example-deployment" |   name: "example-deployment" | ||||||
|   operator.1password.io/auto-restart: "true" |   annotations: | ||||||
|  |     operator.1password.io/auto-restart: "true" | ||||||
| ``` | ``` | ||||||
| If the value is not set, the auto reset settings on the namespace will be used. | If the value is not set, the auto reset settings on the namespace will be used. | ||||||
|  |  | ||||||
| @@ -187,7 +194,8 @@ apiVersion: onepassword.com/v1 | |||||||
| kind: OnePasswordItem | kind: OnePasswordItem | ||||||
| metadata: | metadata: | ||||||
|   name: example |   name: example | ||||||
|   operator.1password.io/auto-restart: "true" |   annotations: | ||||||
|  |     operator.1password.io/auto-restart: "true" | ||||||
| ``` | ``` | ||||||
| If the value is not set, the auto reset settings on the deployment will be used. | If the value is not set, the auto reset settings on the deployment will be used. | ||||||
|  |  | ||||||
|   | |||||||
| @@ -83,9 +83,11 @@ func main() { | |||||||
|  |  | ||||||
| 	printVersion() | 	printVersion() | ||||||
|  |  | ||||||
| 	namespace, err := k8sutil.GetWatchNamespace() | 	namespace := os.Getenv(k8sutil.WatchNamespaceEnvVar) | ||||||
|  |  | ||||||
|  | 	deploymentNamespace, err := k8sutil.GetOperatorNamespace() | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Error(err, "Failed to get watch namespace") | 		log.Error(err, "Failed to get namespace") | ||||||
| 		os.Exit(1) | 		os.Exit(1) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -139,7 +141,7 @@ func main() { | |||||||
| 		go func() { | 		go func() { | ||||||
| 			connectStarted := false | 			connectStarted := false | ||||||
| 			for connectStarted == false { | 			for connectStarted == false { | ||||||
| 				err := op.SetupConnect(mgr.GetClient()) | 				err := op.SetupConnect(mgr.GetClient(), deploymentNamespace) | ||||||
| 				// Cache Not Started is an acceptable error. Retry until cache is started. | 				// Cache Not Started is an acceptable error. Retry until cache is started. | ||||||
| 				if err != nil && !errors.Is(err, &cache.ErrCacheNotStarted{}) { | 				if err != nil && !errors.Is(err, &cache.ErrCacheNotStarted{}) { | ||||||
| 					log.Error(err, "") | 					log.Error(err, "") | ||||||
| @@ -176,7 +178,8 @@ func main() { | |||||||
| 				ticker.Stop() | 				ticker.Stop() | ||||||
| 				return | 				return | ||||||
| 			case <-ticker.C: | 			case <-ticker.C: | ||||||
| 				updatedSecretsPoller.UpdateKubernetesSecretsTask() | 				err := updatedSecretsPoller.UpdateKubernetesSecretsTask() | ||||||
|  | 				log.Error(err, "Error occured during update secret task") | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	}() | 	}() | ||||||
|   | |||||||
| @@ -2,7 +2,6 @@ apiVersion: apps/v1 | |||||||
| kind: Deployment | kind: Deployment | ||||||
| metadata: | metadata: | ||||||
|   name: onepassword-connect |   name: onepassword-connect | ||||||
|   namespace: default |  | ||||||
| spec: | spec: | ||||||
|   selector: |   selector: | ||||||
|     matchLabels: |     matchLabels: | ||||||
|   | |||||||
| @@ -2,7 +2,6 @@ apiVersion: v1 | |||||||
| kind: Service | kind: Service | ||||||
| metadata: | metadata: | ||||||
|   name: onepassword-connect |   name: onepassword-connect | ||||||
|   namespace: default |  | ||||||
| spec: | spec: | ||||||
|   type: NodePort |   type: NodePort | ||||||
|   selector: |   selector: | ||||||
|   | |||||||
| @@ -33,7 +33,7 @@ spec: | |||||||
|           spec: |           spec: | ||||||
|             description: OnePasswordItemSpec defines the desired state of OnePasswordItem |             description: OnePasswordItemSpec defines the desired state of OnePasswordItem | ||||||
|             properties: |             properties: | ||||||
|               itemPath: |               itemReference: | ||||||
|                 type: string |                 type: string | ||||||
|             type: object |             type: object | ||||||
|           status: |           status: | ||||||
|   | |||||||
| @@ -3,4 +3,4 @@ kind: OnePasswordItem | |||||||
| metadata: | metadata: | ||||||
|   name: example |   name: example | ||||||
| spec: | spec: | ||||||
|   itemPath: "vaults/<vault_id>/items/<item_id>" |   itemReference: "op://<vault_id>/<item_id>" | ||||||
|   | |||||||
| @@ -16,6 +16,7 @@ spec: | |||||||
|       containers: |       containers: | ||||||
|         - name: onepassword-connect-operator |         - name: onepassword-connect-operator | ||||||
|           image: 1password/onepassword-operator |           image: 1password/onepassword-operator | ||||||
|  |           imagePullPolicy: Never | ||||||
|           command: ["/manager"] |           command: ["/manager"] | ||||||
|           env: |           env: | ||||||
|             - name: WATCH_NAMESPACE |             - name: WATCH_NAMESPACE | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ import ( | |||||||
|  |  | ||||||
| // OnePasswordItemSpec defines the desired state of OnePasswordItem | // OnePasswordItemSpec defines the desired state of OnePasswordItem | ||||||
| type OnePasswordItemSpec struct { | type OnePasswordItemSpec struct { | ||||||
| 	ItemPath string `json:"itemPath,omitempty"` | 	ItemReference string `json:"itemReference,omitempty"` | ||||||
| } | } | ||||||
|  |  | ||||||
| // OnePasswordItemStatus defines the observed state of OnePasswordItem | // OnePasswordItemStatus defines the observed state of OnePasswordItem | ||||||
|   | |||||||
| @@ -192,11 +192,11 @@ func (r *ReconcileDeployment) HandleApplyingDeployment(namespace string, annotat | |||||||
|  |  | ||||||
| 	secretName := annotations[op.NameAnnotation] | 	secretName := annotations[op.NameAnnotation] | ||||||
| 	if len(secretName) == 0 { | 	if len(secretName) == 0 { | ||||||
| 		reqLog.Info("No 'item-name' annotation set. 'item-path' and 'item-name' must be set as annotations to add new secret.") | 		reqLog.Info("No 'item-name' annotation set. 'item-reference' and 'item-name' must be set as annotations to add new secret.") | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	item, err := op.GetOnePasswordItemByPath(r.opConnectClient, annotations[op.ItemPathAnnotation]) | 	item, err := op.GetOnePasswordItemByReference(r.opConnectClient, annotations[op.ItemReferenceAnnotation]) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return fmt.Errorf("Failed to retrieve item: %v", err) | 		return fmt.Errorf("Failed to retrieve item: %v", err) | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -52,7 +52,7 @@ var ( | |||||||
| 		"password": []byte(password), | 		"password": []byte(password), | ||||||
| 		"username": []byte(username), | 		"username": []byte(username), | ||||||
| 	} | 	} | ||||||
| 	itemPath = fmt.Sprintf("vaults/%v/items/%v", vaultId, itemId) | 	ItemReference = fmt.Sprintf("op://%v/%v", vaultId, itemId) | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var ( | var ( | ||||||
| @@ -76,8 +76,8 @@ var tests = []testReconcileItem{ | |||||||
| 					finalizer, | 					finalizer, | ||||||
| 				}, | 				}, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					op.ItemPathAnnotation: itemPath, | 					op.ItemReferenceAnnotation: ItemReference, | ||||||
| 					op.NameAnnotation:     name, | 					op.NameAnnotation:          name, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
| @@ -90,8 +90,8 @@ var tests = []testReconcileItem{ | |||||||
| 				Name:      "another-deployment", | 				Name:      "another-deployment", | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					op.ItemPathAnnotation: itemPath, | 					op.ItemReferenceAnnotation: ItemReference, | ||||||
| 					op.NameAnnotation:     name, | 					op.NameAnnotation:          name, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Spec: appsv1.DeploymentSpec{ | 			Spec: appsv1.DeploymentSpec{ | ||||||
| @@ -152,8 +152,8 @@ var tests = []testReconcileItem{ | |||||||
| 					finalizer, | 					finalizer, | ||||||
| 				}, | 				}, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					op.ItemPathAnnotation: itemPath, | 					op.ItemReferenceAnnotation: ItemReference, | ||||||
| 					op.NameAnnotation:     name, | 					op.NameAnnotation:          name, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
| @@ -166,8 +166,8 @@ var tests = []testReconcileItem{ | |||||||
| 				Name:      "another-deployment", | 				Name:      "another-deployment", | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					op.ItemPathAnnotation: itemPath, | 					op.ItemReferenceAnnotation: ItemReference, | ||||||
| 					op.NameAnnotation:     name, | 					op.NameAnnotation:          name, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Spec: appsv1.DeploymentSpec{ | 			Spec: appsv1.DeploymentSpec{ | ||||||
| @@ -235,8 +235,8 @@ var tests = []testReconcileItem{ | |||||||
| 					finalizer, | 					finalizer, | ||||||
| 				}, | 				}, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					op.ItemPathAnnotation: itemPath, | 					op.ItemReferenceAnnotation: ItemReference, | ||||||
| 					op.NameAnnotation:     name, | 					op.NameAnnotation:          name, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
| @@ -268,8 +268,8 @@ var tests = []testReconcileItem{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					op.ItemPathAnnotation: itemPath, | 					op.ItemReferenceAnnotation: ItemReference, | ||||||
| 					op.NameAnnotation:     name, | 					op.NameAnnotation:          name, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
| @@ -310,8 +310,8 @@ var tests = []testReconcileItem{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					op.ItemPathAnnotation: itemPath, | 					op.ItemReferenceAnnotation: ItemReference, | ||||||
| 					op.NameAnnotation:     name, | 					op.NameAnnotation:          name, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
| @@ -352,8 +352,8 @@ var tests = []testReconcileItem{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					op.ItemPathAnnotation: itemPath, | 					op.ItemReferenceAnnotation: ItemReference, | ||||||
| 					op.NameAnnotation:     name, | 					op.NameAnnotation:          name, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
|   | |||||||
| @@ -146,7 +146,7 @@ func (r *ReconcileOnePasswordItem) HandleOnePasswordItem(resource *onepasswordv1 | |||||||
| 	secretName := resource.GetName() | 	secretName := resource.GetName() | ||||||
| 	autoRestart := resource.Annotations[op.RestartDeploymentsAnnotation] | 	autoRestart := resource.Annotations[op.RestartDeploymentsAnnotation] | ||||||
|  |  | ||||||
| 	item, err := onepassword.GetOnePasswordItemByPath(r.opConnectClient, resource.Spec.ItemPath) | 	item, err := onepassword.GetOnePasswordItemByReference(r.opConnectClient, resource.Spec.ItemReference) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return fmt.Errorf("Failed to retrieve item: %v", err) | 		return fmt.Errorf("Failed to retrieve item: %v", err) | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -31,6 +31,9 @@ const ( | |||||||
| 	itemId                    = "nwrhuano7bcwddcviubpp4mhfq" | 	itemId                    = "nwrhuano7bcwddcviubpp4mhfq" | ||||||
| 	username                  = "test-user" | 	username                  = "test-user" | ||||||
| 	password                  = "QmHumKc$mUeEem7caHtbaBaJ" | 	password                  = "QmHumKc$mUeEem7caHtbaBaJ" | ||||||
|  | 	firstHost                 = "http://localhost:8080" | ||||||
|  | 	awsKey                    = "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY" | ||||||
|  | 	iceCream                  = "freezing blue 20%" | ||||||
| 	userKey                   = "username" | 	userKey                   = "username" | ||||||
| 	passKey                   = "password" | 	passKey                   = "password" | ||||||
| 	version                   = 123 | 	version                   = 123 | ||||||
| @@ -52,7 +55,7 @@ var ( | |||||||
| 		"password": []byte(password), | 		"password": []byte(password), | ||||||
| 		"username": []byte(username), | 		"username": []byte(username), | ||||||
| 	} | 	} | ||||||
| 	itemPath = fmt.Sprintf("vaults/%v/items/%v", vaultId, itemId) | 	itemReference = fmt.Sprintf("op://%v/%v", vaultId, itemId) | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var ( | var ( | ||||||
| @@ -76,7 +79,7 @@ var tests = []testReconcileItem{ | |||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Spec: onepasswordv1.OnePasswordItemSpec{ | 			Spec: onepasswordv1.OnePasswordItemSpec{ | ||||||
| 				ItemPath: itemPath, | 				ItemReference: itemReference, | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
| 		existingSecret: &corev1.Secret{ | 		existingSecret: &corev1.Secret{ | ||||||
| @@ -108,7 +111,7 @@ var tests = []testReconcileItem{ | |||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 			}, | 			}, | ||||||
| 			Spec: onepasswordv1.OnePasswordItemSpec{ | 			Spec: onepasswordv1.OnePasswordItemSpec{ | ||||||
| 				ItemPath: itemPath, | 				ItemReference: itemReference, | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
| 		existingSecret: &corev1.Secret{ | 		existingSecret: &corev1.Secret{ | ||||||
| @@ -149,7 +152,7 @@ var tests = []testReconcileItem{ | |||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 			}, | 			}, | ||||||
| 			Spec: onepasswordv1.OnePasswordItemSpec{ | 			Spec: onepasswordv1.OnePasswordItemSpec{ | ||||||
| 				ItemPath: itemPath, | 				ItemReference: itemReference, | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
| 		existingSecret: &corev1.Secret{ | 		existingSecret: &corev1.Secret{ | ||||||
| @@ -190,7 +193,7 @@ var tests = []testReconcileItem{ | |||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 			}, | 			}, | ||||||
| 			Spec: onepasswordv1.OnePasswordItemSpec{ | 			Spec: onepasswordv1.OnePasswordItemSpec{ | ||||||
| 				ItemPath: itemPath, | 				ItemReference: itemReference, | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
| 		existingSecret: nil, | 		existingSecret: nil, | ||||||
| @@ -210,6 +213,79 @@ var tests = []testReconcileItem{ | |||||||
| 			passKey: password, | 			passKey: password, | ||||||
| 		}, | 		}, | ||||||
| 	}, | 	}, | ||||||
|  | 	{ | ||||||
|  | 		testName: "Secret from 1Password item with invalid K8s labels", | ||||||
|  | 		customResource: &onepasswordv1.OnePasswordItem{ | ||||||
|  | 			TypeMeta: metav1.TypeMeta{ | ||||||
|  | 				Kind:       onePasswordItemKind, | ||||||
|  | 				APIVersion: onePasswordItemAPIVersion, | ||||||
|  | 			}, | ||||||
|  | 			ObjectMeta: metav1.ObjectMeta{ | ||||||
|  | 				Name:      "!my sECReT it3m%", | ||||||
|  | 				Namespace: namespace, | ||||||
|  | 			}, | ||||||
|  | 			Spec: onepasswordv1.OnePasswordItemSpec{ | ||||||
|  | 				ItemReference: itemReference, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		existingSecret: nil, | ||||||
|  | 		expectedError:  nil, | ||||||
|  | 		expectedResultSecret: &corev1.Secret{ | ||||||
|  | 			ObjectMeta: metav1.ObjectMeta{ | ||||||
|  | 				Name:      "my-secret-it3m", | ||||||
|  | 				Namespace: namespace, | ||||||
|  | 				Annotations: map[string]string{ | ||||||
|  | 					op.VersionAnnotation: fmt.Sprint(version), | ||||||
|  | 				}, | ||||||
|  | 			}, | ||||||
|  | 			Data: expectedSecretData, | ||||||
|  | 		}, | ||||||
|  | 		opItem: map[string]string{ | ||||||
|  | 			userKey: username, | ||||||
|  | 			passKey: password, | ||||||
|  | 		}, | ||||||
|  | 	}, | ||||||
|  | 	{ | ||||||
|  | 		testName: "Secret from 1Password item with fields and sections that have invalid K8s labels", | ||||||
|  | 		customResource: &onepasswordv1.OnePasswordItem{ | ||||||
|  | 			TypeMeta: metav1.TypeMeta{ | ||||||
|  | 				Kind:       onePasswordItemKind, | ||||||
|  | 				APIVersion: onePasswordItemAPIVersion, | ||||||
|  | 			}, | ||||||
|  | 			ObjectMeta: metav1.ObjectMeta{ | ||||||
|  | 				Name:      "!my sECReT it3m%", | ||||||
|  | 				Namespace: namespace, | ||||||
|  | 			}, | ||||||
|  | 			Spec: onepasswordv1.OnePasswordItemSpec{ | ||||||
|  | 				ItemReference: itemReference, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		existingSecret: nil, | ||||||
|  | 		expectedError:  nil, | ||||||
|  | 		expectedResultSecret: &corev1.Secret{ | ||||||
|  | 			ObjectMeta: metav1.ObjectMeta{ | ||||||
|  | 				Name:      "my-secret-it3m", | ||||||
|  | 				Namespace: namespace, | ||||||
|  | 				Annotations: map[string]string{ | ||||||
|  | 					op.VersionAnnotation: fmt.Sprint(version), | ||||||
|  | 				}, | ||||||
|  | 			}, | ||||||
|  | 			Data: map[string][]byte{ | ||||||
|  | 				"password":       []byte(password), | ||||||
|  | 				"username":       []byte(username), | ||||||
|  | 				"first-host":     []byte(firstHost), | ||||||
|  | 				"aws-access-key": []byte(awsKey), | ||||||
|  | 				"ice-cream-type": []byte(iceCream), | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		opItem: map[string]string{ | ||||||
|  | 			userKey:            username, | ||||||
|  | 			passKey:            password, | ||||||
|  | 			"first host":       firstHost, | ||||||
|  | 			"AWS Access Key":   awsKey, | ||||||
|  | 			"😄 ice-cream type": iceCream, | ||||||
|  | 		}, | ||||||
|  | 	}, | ||||||
| } | } | ||||||
|  |  | ||||||
| func TestReconcileOnePasswordItem(t *testing.T) { | func TestReconcileOnePasswordItem(t *testing.T) { | ||||||
| @@ -241,7 +317,10 @@ func TestReconcileOnePasswordItem(t *testing.T) { | |||||||
| 			mocks.GetGetItemFunc = func(uuid string, vaultUUID string) (*onepassword.Item, error) { | 			mocks.GetGetItemFunc = func(uuid string, vaultUUID string) (*onepassword.Item, error) { | ||||||
|  |  | ||||||
| 				item := onepassword.Item{} | 				item := onepassword.Item{} | ||||||
| 				item.Fields = generateFields(testData.opItem["username"], testData.opItem["password"]) | 				item.Fields = []*onepassword.ItemField{} | ||||||
|  | 				for k, v := range testData.opItem { | ||||||
|  | 					item.Fields = append(item.Fields, &onepassword.ItemField{Label: k, Value: v}) | ||||||
|  | 				} | ||||||
| 				item.Version = version | 				item.Version = version | ||||||
| 				item.Vault.ID = vaultUUID | 				item.Vault.ID = vaultUUID | ||||||
| 				item.ID = uuid | 				item.ID = uuid | ||||||
| @@ -257,8 +336,8 @@ func TestReconcileOnePasswordItem(t *testing.T) { | |||||||
| 			// watched resource . | 			// watched resource . | ||||||
| 			req := reconcile.Request{ | 			req := reconcile.Request{ | ||||||
| 				NamespacedName: types.NamespacedName{ | 				NamespacedName: types.NamespacedName{ | ||||||
| 					Name:      name, | 					Name:      testData.customResource.ObjectMeta.Name, | ||||||
| 					Namespace: namespace, | 					Namespace: testData.customResource.ObjectMeta.Namespace, | ||||||
| 				}, | 				}, | ||||||
| 			} | 			} | ||||||
| 			_, err := r.Reconcile(req) | 			_, err := r.Reconcile(req) | ||||||
|   | |||||||
| @@ -4,12 +4,17 @@ import ( | |||||||
| 	"context" | 	"context" | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  |  | ||||||
|  | 	"regexp" | ||||||
|  | 	"strings" | ||||||
|  |  | ||||||
| 	"github.com/1Password/connect-sdk-go/onepassword" | 	"github.com/1Password/connect-sdk-go/onepassword" | ||||||
| 	"github.com/1Password/onepassword-operator/pkg/utils" | 	"github.com/1Password/onepassword-operator/pkg/utils" | ||||||
| 	corev1 "k8s.io/api/core/v1" | 	corev1 "k8s.io/api/core/v1" | ||||||
| 	"k8s.io/apimachinery/pkg/api/errors" | 	"k8s.io/apimachinery/pkg/api/errors" | ||||||
| 	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" | 	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" | ||||||
| 	"k8s.io/apimachinery/pkg/types" | 	"k8s.io/apimachinery/pkg/types" | ||||||
|  | 	kubeValidate "k8s.io/apimachinery/pkg/util/validation" | ||||||
|  |  | ||||||
| 	kubernetesClient "sigs.k8s.io/controller-runtime/pkg/client" | 	kubernetesClient "sigs.k8s.io/controller-runtime/pkg/client" | ||||||
| 	logf "sigs.k8s.io/controller-runtime/pkg/log" | 	logf "sigs.k8s.io/controller-runtime/pkg/log" | ||||||
| ) | ) | ||||||
| @@ -18,7 +23,7 @@ const OnepasswordPrefix = "operator.1password.io" | |||||||
| const NameAnnotation = OnepasswordPrefix + "/item-name" | const NameAnnotation = OnepasswordPrefix + "/item-name" | ||||||
| const VersionAnnotation = OnepasswordPrefix + "/item-version" | const VersionAnnotation = OnepasswordPrefix + "/item-version" | ||||||
| const restartAnnotation = OnepasswordPrefix + "/last-restarted" | const restartAnnotation = OnepasswordPrefix + "/last-restarted" | ||||||
| const ItemPathAnnotation = OnepasswordPrefix + "/item-path" | const ItemReferenceAnnotation = OnepasswordPrefix + "/item-reference" | ||||||
| const RestartDeploymentsAnnotation = OnepasswordPrefix + "/auto-restart" | const RestartDeploymentsAnnotation = OnepasswordPrefix + "/auto-restart" | ||||||
|  |  | ||||||
| var log = logf.Log | var log = logf.Log | ||||||
| @@ -27,8 +32,8 @@ func CreateKubernetesSecretFromItem(kubeClient kubernetesClient.Client, secretNa | |||||||
|  |  | ||||||
| 	itemVersion := fmt.Sprint(item.Version) | 	itemVersion := fmt.Sprint(item.Version) | ||||||
| 	annotations := map[string]string{ | 	annotations := map[string]string{ | ||||||
| 		VersionAnnotation:  itemVersion, | 		VersionAnnotation:       itemVersion, | ||||||
| 		ItemPathAnnotation: fmt.Sprintf("vaults/%v/items/%v", item.Vault.ID, item.ID), | 		ItemReferenceAnnotation: fmt.Sprintf("op://%v/%v", item.Vault.ID, item.ID), | ||||||
| 	} | 	} | ||||||
| 	if autoRestart != "" { | 	if autoRestart != "" { | ||||||
| 		_, err := utils.StringToBool(autoRestart) | 		_, err := utils.StringToBool(autoRestart) | ||||||
| @@ -63,7 +68,7 @@ func CreateKubernetesSecretFromItem(kubeClient kubernetesClient.Client, secretNa | |||||||
| func BuildKubernetesSecretFromOnePasswordItem(name, namespace string, annotations map[string]string, item onepassword.Item) *corev1.Secret { | func BuildKubernetesSecretFromOnePasswordItem(name, namespace string, annotations map[string]string, item onepassword.Item) *corev1.Secret { | ||||||
| 	return &corev1.Secret{ | 	return &corev1.Secret{ | ||||||
| 		ObjectMeta: metav1.ObjectMeta{ | 		ObjectMeta: metav1.ObjectMeta{ | ||||||
| 			Name:        name, | 			Name:        formatSecretName(name), | ||||||
| 			Namespace:   namespace, | 			Namespace:   namespace, | ||||||
| 			Annotations: annotations, | 			Annotations: annotations, | ||||||
| 		}, | 		}, | ||||||
| @@ -75,8 +80,33 @@ func BuildKubernetesSecretData(fields []*onepassword.ItemField) map[string][]byt | |||||||
| 	secretData := map[string][]byte{} | 	secretData := map[string][]byte{} | ||||||
| 	for i := 0; i < len(fields); i++ { | 	for i := 0; i < len(fields); i++ { | ||||||
| 		if fields[i].Value != "" { | 		if fields[i].Value != "" { | ||||||
| 			secretData[fields[i].Label] = []byte(fields[i].Value) | 			key := formatSecretName(fields[i].Label) | ||||||
|  | 			secretData[key] = []byte(fields[i].Value) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	return secretData | 	return secretData | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // formatSecretName rewrites a value to be a valid Secret name or Secret data key. | ||||||
|  | // | ||||||
|  | // The Secret meta.name and data keys must be valid DNS subdomain names (https://kubernetes.io/docs/concepts/configuration/secret/#overview-of-secrets) | ||||||
|  | func formatSecretName(value string) string { | ||||||
|  | 	if errs := kubeValidate.IsDNS1123Subdomain(value); len(errs) == 0 { | ||||||
|  | 		return value | ||||||
|  | 	} | ||||||
|  | 	return createValidSecretName(value) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var invalidDNS1123Chars = regexp.MustCompile("[^a-z0-9-]+") | ||||||
|  |  | ||||||
|  | func createValidSecretName(value string) string { | ||||||
|  | 	result := strings.ToLower(value) | ||||||
|  | 	result = invalidDNS1123Chars.ReplaceAllString(result, "-") | ||||||
|  |  | ||||||
|  | 	if len(result) > kubeValidate.DNS1123SubdomainMaxLength { | ||||||
|  | 		result = result[0:kubeValidate.DNS1123SubdomainMaxLength] | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// first and last character MUST be alphanumeric | ||||||
|  | 	return strings.Trim(result, "-") | ||||||
|  | } | ||||||
|   | |||||||
| @@ -6,6 +6,8 @@ import ( | |||||||
| 	"strings" | 	"strings" | ||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
|  | 	kubeValidate "k8s.io/apimachinery/pkg/util/validation" | ||||||
|  |  | ||||||
| 	"github.com/1Password/connect-sdk-go/onepassword" | 	"github.com/1Password/connect-sdk-go/onepassword" | ||||||
| 	corev1 "k8s.io/api/core/v1" | 	corev1 "k8s.io/api/core/v1" | ||||||
| 	"k8s.io/apimachinery/pkg/types" | 	"k8s.io/apimachinery/pkg/types" | ||||||
| @@ -41,7 +43,7 @@ func TestCreateKubernetesSecretFromOnePasswordItem(t *testing.T) { | |||||||
| 		t.Errorf("Secret was not created: %v", err) | 		t.Errorf("Secret was not created: %v", err) | ||||||
| 	} | 	} | ||||||
| 	compareFields(item.Fields, createdSecret.Data, t) | 	compareFields(item.Fields, createdSecret.Data, t) | ||||||
| 	compareAnnotationsToItem(createdSecret.Annotations, item, t) | 	compareAnnotationsToItem(item.Vault.ID, item.ID, createdSecret.Annotations, item, t) | ||||||
| } | } | ||||||
|  |  | ||||||
| func TestUpdateKubernetesSecretFromOnePasswordItem(t *testing.T) { | func TestUpdateKubernetesSecretFromOnePasswordItem(t *testing.T) { | ||||||
| @@ -77,7 +79,7 @@ func TestUpdateKubernetesSecretFromOnePasswordItem(t *testing.T) { | |||||||
| 		t.Errorf("Secret was not found: %v", err) | 		t.Errorf("Secret was not found: %v", err) | ||||||
| 	} | 	} | ||||||
| 	compareFields(newItem.Fields, updatedSecret.Data, t) | 	compareFields(newItem.Fields, updatedSecret.Data, t) | ||||||
| 	compareAnnotationsToItem(updatedSecret.Annotations, newItem, t) | 	compareAnnotationsToItem(newItem.Vault.ID, newItem.ID, updatedSecret.Annotations, newItem, t) | ||||||
| } | } | ||||||
| func TestBuildKubernetesSecretData(t *testing.T) { | func TestBuildKubernetesSecretData(t *testing.T) { | ||||||
| 	fields := generateFields(5) | 	fields := generateFields(5) | ||||||
| @@ -101,7 +103,7 @@ func TestBuildKubernetesSecretFromOnePasswordItem(t *testing.T) { | |||||||
| 	item.Fields = generateFields(5) | 	item.Fields = generateFields(5) | ||||||
|  |  | ||||||
| 	kubeSecret := BuildKubernetesSecretFromOnePasswordItem(name, namespace, annotations, item) | 	kubeSecret := BuildKubernetesSecretFromOnePasswordItem(name, namespace, annotations, item) | ||||||
| 	if kubeSecret.Name != name { | 	if kubeSecret.Name != strings.ToLower(name) { | ||||||
| 		t.Errorf("Expected name value: %v but got: %v", name, kubeSecret.Name) | 		t.Errorf("Expected name value: %v but got: %v", name, kubeSecret.Name) | ||||||
| 	} | 	} | ||||||
| 	if kubeSecret.Namespace != namespace { | 	if kubeSecret.Namespace != namespace { | ||||||
| @@ -113,11 +115,45 @@ func TestBuildKubernetesSecretFromOnePasswordItem(t *testing.T) { | |||||||
| 	compareFields(item.Fields, kubeSecret.Data, t) | 	compareFields(item.Fields, kubeSecret.Data, t) | ||||||
| } | } | ||||||
|  |  | ||||||
| func compareAnnotationsToItem(annotations map[string]string, item onepassword.Item, t *testing.T) { | func TestBuildKubernetesSecretFixesInvalidLabels(t *testing.T) { | ||||||
| 	actualVaultId, actualItemId, err := ParseVaultIdAndItemIdFromPath(annotations[ItemPathAnnotation]) | 	name := "inV@l1d k8s secret%name" | ||||||
| 	if err != nil { | 	expectedName := "inv-l1d-k8s-secret-name" | ||||||
| 		t.Errorf("Was unable to parse Item Path") | 	namespace := "someNamespace" | ||||||
|  | 	annotations := map[string]string{ | ||||||
|  | 		"annotationKey": "annotationValue", | ||||||
| 	} | 	} | ||||||
|  | 	item := onepassword.Item{} | ||||||
|  |  | ||||||
|  | 	item.Fields = []*onepassword.ItemField{ | ||||||
|  | 		{ | ||||||
|  | 			Label: "label w%th invalid ch!rs-", | ||||||
|  | 			Value: "value1", | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			Label: strings.Repeat("x", kubeValidate.DNS1123SubdomainMaxLength+1), | ||||||
|  | 			Value: "name exceeds max length", | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	kubeSecret := BuildKubernetesSecretFromOnePasswordItem(name, namespace, annotations, item) | ||||||
|  |  | ||||||
|  | 	// Assert Secret's meta.name was fixed | ||||||
|  | 	if kubeSecret.Name != expectedName { | ||||||
|  | 		t.Errorf("Expected name value: %v but got: %v", name, kubeSecret.Name) | ||||||
|  | 	} | ||||||
|  | 	if kubeSecret.Namespace != namespace { | ||||||
|  | 		t.Errorf("Expected namespace value: %v but got: %v", namespace, kubeSecret.Namespace) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// assert labels were fixed for each data key | ||||||
|  | 	for key := range kubeSecret.Data { | ||||||
|  | 		if !validLabel(key) { | ||||||
|  | 			t.Errorf("Expected valid kubernetes label, got %s", key) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func compareAnnotationsToItem(actualVaultId, actualItemId string, annotations map[string]string, item onepassword.Item, t *testing.T) { | ||||||
| 	if actualVaultId != item.Vault.ID { | 	if actualVaultId != item.Vault.ID { | ||||||
| 		t.Errorf("Expected annotation vault id to be %v but was %v", item.Vault.ID, actualVaultId) | 		t.Errorf("Expected annotation vault id to be %v but was %v", item.Vault.ID, actualVaultId) | ||||||
| 	} | 	} | ||||||
| @@ -157,10 +193,9 @@ func generateFields(numToGenerate int) []*onepassword.ItemField { | |||||||
| 	return fields | 	return fields | ||||||
| } | } | ||||||
|  |  | ||||||
| func ParseVaultIdAndItemIdFromPath(path string) (string, string, error) { | func validLabel(v string) bool { | ||||||
| 	splitPath := strings.Split(path, "/") | 	if err := kubeValidate.IsDNS1123Subdomain(v); len(err) > 0 { | ||||||
| 	if len(splitPath) == 4 && splitPath[0] == "vaults" && splitPath[2] == "items" { | 		return false | ||||||
| 		return splitPath[1], splitPath[3], nil |  | ||||||
| 	} | 	} | ||||||
| 	return "", "", fmt.Errorf("%q is not an acceptable path for One Password item. Must be of the format: `vaults/{vault_id}/items/{item_id}`", path) | 	return true | ||||||
| } | } | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ import ( | |||||||
|  |  | ||||||
| const ( | const ( | ||||||
| 	OnepasswordPrefix            = "operator.1password.io" | 	OnepasswordPrefix            = "operator.1password.io" | ||||||
| 	ItemPathAnnotation           = OnepasswordPrefix + "/item-path" | 	ItemReferenceAnnotation      = OnepasswordPrefix + "/item-reference" | ||||||
| 	NameAnnotation               = OnepasswordPrefix + "/item-name" | 	NameAnnotation               = OnepasswordPrefix + "/item-name" | ||||||
| 	VersionAnnotation            = OnepasswordPrefix + "/item-version" | 	VersionAnnotation            = OnepasswordPrefix + "/item-version" | ||||||
| 	RestartAnnotation            = OnepasswordPrefix + "/last-restarted" | 	RestartAnnotation            = OnepasswordPrefix + "/last-restarted" | ||||||
|   | |||||||
| @@ -22,7 +22,7 @@ func TestFilterAnnotations(t *testing.T) { | |||||||
| 	if len(filteredAnnotations) != 2 { | 	if len(filteredAnnotations) != 2 { | ||||||
| 		t.Errorf("Unexpected number of filtered annotations returned. Expected 2, got %v", len(filteredAnnotations)) | 		t.Errorf("Unexpected number of filtered annotations returned. Expected 2, got %v", len(filteredAnnotations)) | ||||||
| 	} | 	} | ||||||
| 	_, found := filteredAnnotations[ItemPathAnnotation] | 	_, found := filteredAnnotations[ItemReferenceAnnotation] | ||||||
| 	if !found { | 	if !found { | ||||||
| 		t.Errorf("One Password Annotation was filtered when it should not have been") | 		t.Errorf("One Password Annotation was filtered when it should not have been") | ||||||
| 	} | 	} | ||||||
| @@ -87,7 +87,7 @@ func TestGetNoAnnotationsForDeployment(t *testing.T) { | |||||||
|  |  | ||||||
| func getValidAnnotations() map[string]string { | func getValidAnnotations() map[string]string { | ||||||
| 	return map[string]string{ | 	return map[string]string{ | ||||||
| 		ItemPathAnnotation: "vaults/b3e4c7fc-8bf7-4c22-b8bb-147539f10e4f/items/b3e4c7fc-8bf7-4c22-b8bb-147539f10e4f", | 		ItemReferenceAnnotation: "op://b3e4c7fc-8bf7-4c22-b8bb-147539f10e4f/b3e4c7fc-8bf7-4c22-b8bb-147539f10e4f", | ||||||
| 		NameAnnotation:     "secretName", | 		NameAnnotation:          "secretName", | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ package onepassword | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 	v1 "k8s.io/apimachinery/pkg/apis/meta/v1" | ||||||
| 	"os" | 	"os" | ||||||
|  |  | ||||||
| 	appsv1 "k8s.io/api/apps/v1" | 	appsv1 "k8s.io/api/apps/v1" | ||||||
| @@ -17,13 +18,13 @@ var logConnectSetup = logf.Log.WithName("ConnectSetup") | |||||||
| var deploymentPath = "deploy/connect/deployment.yaml" | var deploymentPath = "deploy/connect/deployment.yaml" | ||||||
| var servicePath = "deploy/connect/service.yaml" | var servicePath = "deploy/connect/service.yaml" | ||||||
|  |  | ||||||
| func SetupConnect(kubeClient client.Client) error { | func SetupConnect(kubeClient client.Client, deploymentNamespace string) error { | ||||||
| 	err := setupService(kubeClient, servicePath) | 	err := setupService(kubeClient, servicePath, deploymentNamespace) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	err = setupDeployment(kubeClient, deploymentPath) | 	err = setupDeployment(kubeClient, deploymentPath, deploymentNamespace) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| @@ -31,22 +32,22 @@ func SetupConnect(kubeClient client.Client) error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func setupDeployment(kubeClient client.Client, deploymentPath string) error { | func setupDeployment(kubeClient client.Client, deploymentPath string, deploymentNamespace string) error { | ||||||
| 	existingDeployment := &appsv1.Deployment{} | 	existingDeployment := &appsv1.Deployment{} | ||||||
|  |  | ||||||
| 	// check if deployment has already been created | 	// check if deployment has already been created | ||||||
| 	err := kubeClient.Get(context.Background(), types.NamespacedName{Name: "onepassword-connect", Namespace: "default"}, existingDeployment) | 	err := kubeClient.Get(context.Background(), types.NamespacedName{Name: "onepassword-connect", Namespace: deploymentNamespace}, existingDeployment) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		if errors.IsNotFound(err) { | 		if errors.IsNotFound(err) { | ||||||
| 			logConnectSetup.Info("No existing Connect deployment found. Creating Deployment") | 			logConnectSetup.Info("No existing Connect deployment found. Creating Deployment") | ||||||
| 			return createDeployment(kubeClient, deploymentPath) | 			return createDeployment(kubeClient, deploymentPath, deploymentNamespace) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	return err | 	return err | ||||||
| } | } | ||||||
|  |  | ||||||
| func createDeployment(kubeClient client.Client, deploymentPath string) error { | func createDeployment(kubeClient client.Client, deploymentPath string, deploymentNamespace string) error { | ||||||
| 	deployment, err := getDeploymentToCreate(deploymentPath) | 	deployment, err := getDeploymentToCreate(deploymentPath, deploymentNamespace) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| @@ -59,12 +60,16 @@ func createDeployment(kubeClient client.Client, deploymentPath string) error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func getDeploymentToCreate(deploymentPath string) (*appsv1.Deployment, error) { | func getDeploymentToCreate(deploymentPath string, deploymentNamespace string) (*appsv1.Deployment, error) { | ||||||
| 	f, err := os.Open(deploymentPath) | 	f, err := os.Open(deploymentPath) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	deployment := &appsv1.Deployment{} | 	deployment := &appsv1.Deployment{ | ||||||
|  | 		ObjectMeta: v1.ObjectMeta{ | ||||||
|  | 			Namespace: deploymentNamespace, | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	err = yaml.NewYAMLOrJSONDecoder(f, 4096).Decode(deployment) | 	err = yaml.NewYAMLOrJSONDecoder(f, 4096).Decode(deployment) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @@ -73,26 +78,30 @@ func getDeploymentToCreate(deploymentPath string) (*appsv1.Deployment, error) { | |||||||
| 	return deployment, nil | 	return deployment, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func setupService(kubeClient client.Client, servicePath string) error { | func setupService(kubeClient client.Client, servicePath string, deploymentNamespace string) error { | ||||||
| 	existingService := &corev1.Service{} | 	existingService := &corev1.Service{} | ||||||
|  |  | ||||||
| 	//check if service has already been created | 	//check if service has already been created | ||||||
| 	err := kubeClient.Get(context.Background(), types.NamespacedName{Name: "onepassword-connect", Namespace: "default"}, existingService) | 	err := kubeClient.Get(context.Background(), types.NamespacedName{Name: "onepassword-connect", Namespace: deploymentNamespace}, existingService) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		if errors.IsNotFound(err) { | 		if errors.IsNotFound(err) { | ||||||
| 			logConnectSetup.Info("No existing Connect service found. Creating Service") | 			logConnectSetup.Info("No existing Connect service found. Creating Service") | ||||||
| 			return createService(kubeClient, servicePath) | 			return createService(kubeClient, servicePath, deploymentNamespace) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	return err | 	return err | ||||||
| } | } | ||||||
|  |  | ||||||
| func createService(kubeClient client.Client, servicePath string) error { | func createService(kubeClient client.Client, servicePath string, deploymentNamespace string) error { | ||||||
| 	f, err := os.Open(servicePath) | 	f, err := os.Open(servicePath) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	service := &corev1.Service{} | 	service := &corev1.Service{ | ||||||
|  | 		ObjectMeta: v1.ObjectMeta{ | ||||||
|  | 			Namespace: deploymentNamespace, | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	err = yaml.NewYAMLOrJSONDecoder(f, 4096).Decode(service) | 	err = yaml.NewYAMLOrJSONDecoder(f, 4096).Decode(service) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
|   | |||||||
| @@ -25,7 +25,7 @@ func TestServiceSetup(t *testing.T) { | |||||||
| 	// Create a fake client to mock API calls. | 	// Create a fake client to mock API calls. | ||||||
| 	client := fake.NewFakeClientWithScheme(s, objs...) | 	client := fake.NewFakeClientWithScheme(s, objs...) | ||||||
|  |  | ||||||
| 	err := setupService(client, "../../deploy/connect/service.yaml") | 	err := setupService(client, "../../deploy/connect/service.yaml", defaultNamespacedName.Namespace) | ||||||
|  |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		t.Errorf("Error Setting Up Connect: %v", err) | 		t.Errorf("Error Setting Up Connect: %v", err) | ||||||
| @@ -50,7 +50,7 @@ func TestDeploymentSetup(t *testing.T) { | |||||||
| 	// Create a fake client to mock API calls. | 	// Create a fake client to mock API calls. | ||||||
| 	client := fake.NewFakeClientWithScheme(s, objs...) | 	client := fake.NewFakeClientWithScheme(s, objs...) | ||||||
|  |  | ||||||
| 	err := setupDeployment(client, "../../deploy/connect/deployment.yaml") | 	err := setupDeployment(client, "../../deploy/connect/deployment.yaml", defaultNamespacedName.Namespace) | ||||||
|  |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		t.Errorf("Error Setting Up Connect: %v", err) | 		t.Errorf("Error Setting Up Connect: %v", err) | ||||||
|   | |||||||
| @@ -11,11 +11,16 @@ import ( | |||||||
|  |  | ||||||
| var logger = logf.Log.WithName("retrieve_item") | var logger = logf.Log.WithName("retrieve_item") | ||||||
|  |  | ||||||
| func GetOnePasswordItemByPath(opConnectClient connect.Client, path string) (*onepassword.Item, error) { | const ( | ||||||
| 	vaultValue, itemValue, err := ParseVaultAndItemFromPath(path) | 	secretReferencePrefix = "op://" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func GetOnePasswordItemByReference(opConnectClient connect.Client, reference string) (*onepassword.Item, error) { | ||||||
|  | 	vaultValue, itemValue, err := ParseReference(reference) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	vaultId, err := getVaultId(opConnectClient, vaultValue) | 	vaultId, err := getVaultId(opConnectClient, vaultValue) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @@ -33,12 +38,28 @@ func GetOnePasswordItemByPath(opConnectClient connect.Client, path string) (*one | |||||||
| 	return item, nil | 	return item, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func ParseVaultAndItemFromPath(path string) (string, string, error) { | func ParseReference(reference string) (string, string, error) { | ||||||
| 	splitPath := strings.Split(path, "/") | 	if !strings.HasPrefix(reference, secretReferencePrefix) { | ||||||
| 	if len(splitPath) == 4 && splitPath[0] == "vaults" && splitPath[2] == "items" { | 		return "", "", fmt.Errorf("secret reference should start with `op://`") | ||||||
| 		return splitPath[1], splitPath[3], nil |  | ||||||
| 	} | 	} | ||||||
| 	return "", "", fmt.Errorf("%q is not an acceptable path for One Password item. Must be of the format: `vaults/{vault_id}/items/{item_id}`", path) | 	path := strings.TrimPrefix(reference, secretReferencePrefix) | ||||||
|  |  | ||||||
|  | 	splitPath := strings.Split(path, "/") | ||||||
|  | 	if len(splitPath) != 2 { | ||||||
|  | 		return "", "", fmt.Errorf("Invalid secret reference : %s. Secret references should match op://<vault>/<item>", reference) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	vault := splitPath[0] | ||||||
|  | 	if vault == "" { | ||||||
|  | 		return "", "", fmt.Errorf("Invalid secret reference : %s. Vault can't be empty.", reference) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	item := splitPath[1] | ||||||
|  | 	if item == "" { | ||||||
|  | 		return "", "", fmt.Errorf("Invalid secret reference : %s. Item can't be empty.", reference) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return vault, item, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func getVaultId(client connect.Client, vaultIdentifier string) (string, error) { | func getVaultId(client connect.Client, vaultIdentifier string) (string, error) { | ||||||
|   | |||||||
| @@ -110,13 +110,13 @@ func (h *SecretUpdateHandler) updateKubernetesSecrets() (map[string]map[string]* | |||||||
| 	for i := 0; i < len(secrets.Items); i++ { | 	for i := 0; i < len(secrets.Items); i++ { | ||||||
| 		secret := secrets.Items[i] | 		secret := secrets.Items[i] | ||||||
|  |  | ||||||
| 		itemPath := secret.Annotations[ItemPathAnnotation] | 		itemReference := secret.Annotations[ItemReferenceAnnotation] | ||||||
| 		currentVersion := secret.Annotations[VersionAnnotation] | 		currentVersion := secret.Annotations[VersionAnnotation] | ||||||
| 		if len(itemPath) == 0 || len(currentVersion) == 0 { | 		if len(itemReference) == 0 || len(currentVersion) == 0 { | ||||||
| 			continue | 			continue | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		item, err := GetOnePasswordItemByPath(h.opConnectClient, secret.Annotations[ItemPathAnnotation]) | 		item, err := GetOnePasswordItemByReference(h.opConnectClient, secret.Annotations[ItemReferenceAnnotation]) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return nil, fmt.Errorf("Failed to retrieve item: %v", err) | 			return nil, fmt.Errorf("Failed to retrieve item: %v", err) | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -51,7 +51,7 @@ var ( | |||||||
| 		"password": []byte(password), | 		"password": []byte(password), | ||||||
| 		"username": []byte(username), | 		"username": []byte(username), | ||||||
| 	} | 	} | ||||||
| 	itemPath = fmt.Sprintf("vaults/%v/items/%v", vaultId, itemId) | 	itemReference = fmt.Sprintf("op://%v/%v", vaultId, itemId) | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var defaultNamespace = &corev1.Namespace{ | var defaultNamespace = &corev1.Namespace{ | ||||||
| @@ -73,8 +73,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					NameAnnotation:     "unlrelated secret", | 					NameAnnotation:          "unlrelated secret", | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
| @@ -83,8 +83,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:  "old version", | 					VersionAnnotation:       "old version", | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Data: expectedSecretData, | 			Data: expectedSecretData, | ||||||
| @@ -95,8 +95,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:  fmt.Sprint(itemVersion), | 					VersionAnnotation:       fmt.Sprint(itemVersion), | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Data: expectedSecretData, | 			Data: expectedSecretData, | ||||||
| @@ -149,8 +149,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:  "old version", | 					VersionAnnotation:       "old version", | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Data: expectedSecretData, | 			Data: expectedSecretData, | ||||||
| @@ -161,8 +161,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:  fmt.Sprint(itemVersion), | 					VersionAnnotation:       fmt.Sprint(itemVersion), | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Data: expectedSecretData, | 			Data: expectedSecretData, | ||||||
| @@ -186,8 +186,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 					NameAnnotation:     name, | 					NameAnnotation:          name, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
| @@ -196,8 +196,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:  "old version", | 					VersionAnnotation:       "old version", | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Data: expectedSecretData, | 			Data: expectedSecretData, | ||||||
| @@ -208,8 +208,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:  fmt.Sprint(itemVersion), | 					VersionAnnotation:       fmt.Sprint(itemVersion), | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Data: expectedSecretData, | 			Data: expectedSecretData, | ||||||
| @@ -255,8 +255,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:  "old version", | 					VersionAnnotation:       "old version", | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Data: expectedSecretData, | 			Data: expectedSecretData, | ||||||
| @@ -267,8 +267,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:  fmt.Sprint(itemVersion), | 					VersionAnnotation:       fmt.Sprint(itemVersion), | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Data: expectedSecretData, | 			Data: expectedSecretData, | ||||||
| @@ -292,8 +292,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 					NameAnnotation:     name, | 					NameAnnotation:          name, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
| @@ -302,8 +302,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:  fmt.Sprint(itemVersion), | 					VersionAnnotation:       fmt.Sprint(itemVersion), | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Data: expectedSecretData, | 			Data: expectedSecretData, | ||||||
| @@ -314,8 +314,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:  fmt.Sprint(itemVersion), | 					VersionAnnotation:       fmt.Sprint(itemVersion), | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Data: expectedSecretData, | 			Data: expectedSecretData, | ||||||
| @@ -369,8 +369,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:  "old version", | 					VersionAnnotation:       "old version", | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Data: expectedSecretData, | 			Data: expectedSecretData, | ||||||
| @@ -381,8 +381,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:  fmt.Sprint(itemVersion), | 					VersionAnnotation:       fmt.Sprint(itemVersion), | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Data: expectedSecretData, | 			Data: expectedSecretData, | ||||||
| @@ -439,7 +439,7 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:            "old version", | 					VersionAnnotation:            "old version", | ||||||
| 					ItemPathAnnotation:           itemPath, | 					ItemReferenceAnnotation:      itemReference, | ||||||
| 					RestartDeploymentsAnnotation: "true", | 					RestartDeploymentsAnnotation: "true", | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| @@ -452,7 +452,7 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:            fmt.Sprint(itemVersion), | 					VersionAnnotation:            fmt.Sprint(itemVersion), | ||||||
| 					ItemPathAnnotation:           itemPath, | 					ItemReferenceAnnotation:      itemReference, | ||||||
| 					RestartDeploymentsAnnotation: "true", | 					RestartDeploymentsAnnotation: "true", | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| @@ -510,7 +510,7 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:            "old version", | 					VersionAnnotation:            "old version", | ||||||
| 					ItemPathAnnotation:           itemPath, | 					ItemReferenceAnnotation:      itemReference, | ||||||
| 					RestartDeploymentsAnnotation: "false", | 					RestartDeploymentsAnnotation: "false", | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| @@ -523,7 +523,7 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:            fmt.Sprint(itemVersion), | 					VersionAnnotation:            fmt.Sprint(itemVersion), | ||||||
| 					ItemPathAnnotation:           itemPath, | 					ItemReferenceAnnotation:      itemReference, | ||||||
| 					RestartDeploymentsAnnotation: "false", | 					RestartDeploymentsAnnotation: "false", | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| @@ -580,8 +580,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:  "old version", | 					VersionAnnotation:       "old version", | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Data: expectedSecretData, | 			Data: expectedSecretData, | ||||||
| @@ -592,8 +592,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:  fmt.Sprint(itemVersion), | 					VersionAnnotation:       fmt.Sprint(itemVersion), | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Data: expectedSecretData, | 			Data: expectedSecretData, | ||||||
| @@ -657,8 +657,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:  "old version", | 					VersionAnnotation:       "old version", | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Data: expectedSecretData, | 			Data: expectedSecretData, | ||||||
| @@ -669,8 +669,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:  fmt.Sprint(itemVersion), | 					VersionAnnotation:       fmt.Sprint(itemVersion), | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Data: expectedSecretData, | 			Data: expectedSecretData, | ||||||
| @@ -730,8 +730,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:  "old version", | 					VersionAnnotation:       "old version", | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Data: expectedSecretData, | 			Data: expectedSecretData, | ||||||
| @@ -742,8 +742,8 @@ var tests = []testUpdateSecretTask{ | |||||||
| 				Name:      name, | 				Name:      name, | ||||||
| 				Namespace: namespace, | 				Namespace: namespace, | ||||||
| 				Annotations: map[string]string{ | 				Annotations: map[string]string{ | ||||||
| 					VersionAnnotation:  fmt.Sprint(itemVersion), | 					VersionAnnotation:       fmt.Sprint(itemVersion), | ||||||
| 					ItemPathAnnotation: itemPath, | 					ItemReferenceAnnotation: itemReference, | ||||||
| 				}, | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			Data: expectedSecretData, | 			Data: expectedSecretData, | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user