Overlays and patches
Introduction
Patches are partial configuration blocks that can be overlaid onto a given package.
You can store them and ship them with a package to provide a quick way to modify settings and generate different packages from a given base package, overlaying patches on top until you get your desired state.
Patches work by utilizing the Package
object model and manipulating it using kustomize
.
The name
parameter of the object being overlaid (for example: a database or table) is used as a key and array types will be merged according to the strategic merge mechanism.
There is no mechanism to patch code files currently, only configuration files i.e. databases and pipelines.
Using patches
Adding a pipeline
-
To create a patch, use the
add
command and specifypatch
as your new object:kxi package -q init mypkg --force kxi package -q add --to mypkg patch --name my-first-patch cat mypkg/patches/my-first-patch.yaml
# yaml-language-server: $schema=https://code.kx.com/insights/enterprise/packaging/schemas/package.json#/$defs/OverlayWrapper uuid: 16f6b31d-90d6-4e59-8120-18ba487a80ca kind: Package apiVersion: pakx/v1 metadata: name: target spec: uuid: b4f5bed5-d96a-4186-8b70-e718a269011c manifest: {} tables: uuid: 34f63948-6f4a-4d49-babe-94f3e50f943b schemas: [] databases: [] pipelines: [] router: null views: [] deployment_config: null data_files: {} version_info: ''
This adds a
patch
file to thepatches
directory. The new file includes the structure required for a patch file, but no pipeline.Be careful when you modify the
patch
file- Do not modify the headers above the
spec:
field - It is good practice to remove everything inside
spec:
except what you want to patch
- Do not modify the headers above the
-
You can see the patch object is now present in the manifest file:
kxi package info mypkg
==PATH== /builds/kxdev/documentation/insights/mypkg ==OBJECT== Package ==Manifest== uuid: ee55e8ea-8f38-4c34-ae85-d64393d7f838 name: mypkg version: 0.0.1 metadata: description: '' authors: - {} entrypoints: default: init.q patches: - uuid: abf62d14-c973-43c5-b976-fef9a2eaeef0 name: my-first-patch
-
Modify the
patch
to add a newpipeline
:patchspec=$(cat mypkg/patches/my-first-patch.yaml | grep spec: -B10 | cat - <(echo " pipelines: [{name: mynewpipeline, spec: init.q}]")) echo "$patchspec" > mypkg/patches/my-first-patch.yaml cat mypkg/patches/my-first-patch.yaml
# yaml-language-server: $schema=https://code.kx.com/insights/enterprise/packaging/schemas/package.json#/$defs/OverlayWrapper uuid: 16f6b31d-90d6-4e59-8120-18ba487a80ca kind: Package apiVersion: pakx/v1 metadata: name: target spec: pipelines: [{name: mynewpipeline, spec: init.q}]
-
Apply or "overlay" this patch onto your current package using the
overlay
command:Without specifying any other information, the overlay command resolves All the patches found in the manifest:
kxi package overlay mypkg kxi package info mypkg | grep pipelines: -A5
Found patch to overlay: mypkg/patches/my-first-patch.yaml Writing /builds/kxdev/documentation/insights/mypkg/manifest.yaml pipelines: mynewpipeline: file: pipelines/mynewpipeline.yaml views: {} patches: - uuid: abf62d14-c973-43c5-b976-fef9a2eaeef0
When you specify patches after the package name, you can select any valid patch file, they dont even need to be located inside a package.
kxi package overlay mypkg mypkg/patches/my-first-patch.yaml kxi package info mypkg | grep pipelines: -A5
Writing /builds/kxdev/documentation/insights/mypkg/manifest.yaml pipelines: mynewpipeline: file: pipelines/mynewpipeline.yaml views: {} patches: - uuid: abf62d14-c973-43c5-b976-fef9a2eaeef0
Modifying tables
To modify a table, you must patch the Database because tables are contained inside the Database
construct :
kxi package -q init mypkg --force
kxi package -q add --to mypkg database --name mydb
# This automatically populates a "default" table
kxi package -q add --to mypkg patch --name my-db-patch
read -r -d '' dbspec <<'EOF'
databases:
- name: mydb
tables:
schemas:
- name: default
description: We've changed the default table's description too!
type: basic
columns:
- name: myfloat
type: float
description: we've added a new float column called myfloat
- name: x2
type: symbol
description: The default x2 column has changed type
EOF
# The patch might look a little complex, but it's just a few nested structs
# N.B We have indented it here to nest it under the `spec` field
patchspec=$(cat mypkg/patches/my-db-patch.yaml | grep spec: -B10 | cat - <(echo " $dbspec"))
echo "$patchspec" > mypkg/patches/my-db-patch.yaml
# This is just to stitch the header with the dbspec we wrote
cat mypkg/patches/my-db-patch.yaml
kxi package overlay mypkg
echo""
echo "----Resultant schema for 'default' table----"
echo "--------------------------------------------"
cat mypkg/databases/mydb/tables/default.yaml
# yaml-language-server: $schema=https://code.kx.com/insights/enterprise/packaging/schemas/package.json#/$defs/OverlayWrapper
uuid: b4e8305f-0ef6-4a98-b38d-95eef4a75e2d
kind: Package
apiVersion: pakx/v1
metadata:
name: target
spec:
databases:
- name: mydb
tables:
schemas:
- name: default
description: We've changed the default table's description too!
type: basic
columns:
- name: myfloat
type: float
description: we've added a new float column called myfloat
- name: x2
type: symbol
description: The default x2 column has changed type
Found patch to overlay: mypkg/patches/my-db-patch.yaml
Writing /builds/kxdev/documentation/insights/mypkg/manifest.yaml
----Resultant schema for 'default' table----
--------------------------------------------
# yaml-language-server: $schema=https://code.kx.com/insights/enterprise/packaging/schemas/package.json#/$defs/Table
name: default
uuid: 5d4acf00-b30c-4dd3-b76b-00df3ba39e60
columns:
- description: we've added a new float column called myfloat
name: myfloat
type: float
- description: The default x2 column has changed type
name: x2
type: symbol
description: We've changed the default table's description too!
type: basic
Column ordering
After patching you can observe the columns have changed order. The new column has been added in at the top. If your application requires specific column ordering, then you need to do this manually.
Referencing dependency patches
You can also reference patches held in other packages to minimize the need to copy patch specs that you use regularly.
Referenced dependencies must already be installed.
Referencing a patch from a dependency can only be done after that dependency is installed.
In the below example you explicitly run: kxi package install myotherdep
. This allows the kxi package
to resolve it properly when you then call overlay.
You do this in the following way:
kxi package -q init myotherpkg --force
# setup
kxi package -q add --to myotherpkg patch --name patch-from-afar
patchspec=$(cat myotherpkg/patches/patch-from-afar.yaml | grep spec: -B10 | cat - <(echo " pipelines: [{name: mynewpipeline, spec: init.q}]"))
echo "$patchspec" > myotherpkg/patches/patch-from-afar.yaml
# add a patch to myother pkg
_=$(kxi package -q install myotherpkg)
# install myotherpkg so that you can reference it
kxi package -q add --to mypkg dep --name myotherpkg --version 0.0.1
# add a dependency on myotherpkg to mypkg
kxi package -q add --to mypkg patch --name patch-from-afar --dep myotherpkg
# add a patch in mypkg referencing a patch in myotherpkg
kxi package -q overlay mypkg
# see that you have successfully used the patch adding a new pipeline from myotherpkg
kxi package info mypkg | grep pipelines: -A5
pipelines:
mynewpipeline:
file: pipelines/mynewpipeline.yaml
views: {}
patches:
- uuid: b418c242-823d-47fb-a366-a2b0c0a98cc2