Search code examples
openshiftkubernetes-cronjobimagestream

Can an OpenShift CronJob benefit from ImageStream?


I have my CronJob working fine without the use of an image stream.

The job runs every 15 minutes and always pulls a tagged image, e.g. my-cron:stable.

Since the image is always pulled and the schedule tells the cluster when to run my job, what do I gain from knowing that there's an updated version of my image?

If the image changes and there's a running instance of my job, I want the job to complete using the current version of the image.

In the next scheduled run the updated image is pulled (AlwaysPull). So it seems I don't gain much tracking changes to an image stream for cron jobs.


Solution

  • By using an ImageStream reference, you can avoid having to include the container image repository hostname hostname and port, and the project name in your Cron Job definition.

    The docker repository reference looks likes this:

    image-registry.openshift-image-registry.svc:5000/my-project/my-is:latest
    

    The value of the equivalent annotation placed on a Cron Job looks like this:

    [
      {
        "from": {
          "kind": "ImageStreamTag",
          "name": "my-is:latest"
        },
        "fieldPath": "spec.jobTemplate.spec.template.spec.containers[?(@.name==\"my-container\")].image"
      }
    ]
    

    On the one hand, this is longer. On the other hand, it includes less redundant information.

    So, compared to other types of kubernetes resources, Image Streams don't add a great deal of functionality to Cron Jobs. But you might benefit from not having to hardcode the project name if for instance you kept the Cron Job YAML in Git and wanted to apply it to several different projects.


    Kubernetes-native resources which contain a pod can be updated automatically in response to an image stream tag update by adding the image.openshift.io/triggers annotation.

    This annotation can be placed on CronJobs, Deployments, StatefulSets, DaemonSets, Jobs, ReplicationControllers, etc.

    The easiest way to do this is with the oc command.

    $ oc set triggers cronjob/my-cronjob
    NAME                 TYPE    VALUE  AUTO
    cronjobs/my-cronjob  config         true
    
    $ oc set triggers cronjob/my-cronjob --from-image=my-is:latest -c my-container
    cronjob.batch/my-cronjob updated
    
    $ oc set triggers cronjob/my-cronjob
    NAME                 TYPE    VALUE                        AUTO
    cronjobs/my-cronjob  config                               true
    cronjobs/my-cronjob  image   my-is:latest (my-container)  true
    

    The effect of the oc set triggers command was to add the annotation to the CronJob, which we can examine with:

    $ oc get cronjob/my-cronjob -o json | jq '.metadata.annotations["image.openshift.io/triggers"]' -r | jq
    [
      {
        "from": {
          "kind": "ImageStreamTag",
          "name": "my-is:latest"
        },
        "fieldPath": "spec.jobTemplate.spec.template.spec.containers[?(@.name==\"my-container\")].image"
      }
    ]
    

    This is documented in Images - Triggering updates on image stream changes - but the syntax in the documentation appears to be wrong, so use oc set triggers if you find that the annotation you write by hand doesn't work.