From 2457e536c6c631fd67816f3e92ffe6834a4874c8 Mon Sep 17 00:00:00 2001 From: Tobias McNulty Date: Tue, 24 May 2022 08:29:40 -0400 Subject: [PATCH] Specify timeout for migrate + allow extending batch jobs to be run (#59) --- .gitignore | 2 ++ CHANGES.rst | 9 ++++- defaults/main.yml | 26 ++++++++++++-- tasks/batchjob_tasks.yml | 28 +++++++++++++++ tasks/main.yml | 74 ++++++-------------------------------- templates/batchjob.yaml.j2 | 4 +-- 6 files changed, 74 insertions(+), 69 deletions(-) create mode 100644 tasks/batchjob_tasks.yml diff --git a/.gitignore b/.gitignore index 849c49a..1dc15c9 100644 --- a/.gitignore +++ b/.gitignore @@ -1,3 +1,5 @@ .idea .vscode .python-version +.direnv +.envrc diff --git a/CHANGES.rst b/CHANGES.rst index d7a77c4..5aa79ad 100644 --- a/CHANGES.rst +++ b/CHANGES.rst @@ -4,6 +4,14 @@ caktus.django-k8s Changes ------- +v1.5.1 on May 24th, 2022 +~~~~~~~~~~~~~~~~~~~~~ +* Rename ``k8s_migration_command`` to ``k8s_migrations_command`` (the old name will continue + working for now, but update your projects!) +* Add ``k8s_migrations_timeout`` variable +* Support further customizing batch jobs run before and after deploys via the new + ``k8s_predeploy_batchjobs`` and ``k8s_postdeploy_batchjobs`` variables + v1.5.0 on April 20th, 2022 ~~~~~~~~~~~~~~~~~~~~~ @@ -20,7 +28,6 @@ v1.5.0 on April 20th, 2022 * Add support for mounting data volumes via Secrets within containers - v1.4.0 on Oct 14, 2021 ~~~~~~~~~~~~~~~~~~~~~~ diff --git a/defaults/main.yml b/defaults/main.yml index e1f7f44..c52f2dd 100644 --- a/defaults/main.yml +++ b/defaults/main.yml @@ -172,13 +172,20 @@ k8s_worker_containers: k8s_worker_beat_enabled: false k8s_migrations_enabled: true -k8s_migration_command: +k8s_migrations_timeout: 120 # seconds +k8s_migration_command: # DEPRECATED: keep name without trailing 's' for backwards compatibility - python - manage.py - migrate - --noinput - -v - "2" +k8s_migrations_command: "{{ k8s_migration_command }}" +k8s_migrations_batchjob: + job_name: migrate + batch_command: "{{ k8s_migrations_command }}" + enabled: "{{ k8s_migrations_enabled }}" + timeout: "{{ k8s_migrations_timeout }}" k8s_collectstatic_enabled: true k8s_collectstatic_timeout: 120 # seconds @@ -189,8 +196,21 @@ k8s_collectstatic_command: - --noinput - -v - "2" - -k8s_batchjob_state: "{{ (k8s_migrations_enabled or k8s_collectstatic_enabled) | ternary('present', 'absent') }}" +k8s_collectstatic_batchjob: + job_name: collectstatic + batch_command: "{{ k8s_collectstatic_command }}" + enabled: "{{ k8s_collectstatic_enabled }}" + timeout: "{{ k8s_collectstatic_timeout }}" + +# k8s_predeploy_batchjobs are one-off jobs that are run *before* every deploy. You can keep +# the default or override to customize the jobs needed for your project. +k8s_predeploy_batchjobs: + - "{{ k8s_migrations_batchjob }}" + - "{{ k8s_collectstatic_batchjob }}" +# k8s_postdeploy_batchjobs is a hook for running batch jobs *after* every deploy. +k8s_postdeploy_batchjobs: [] +# k8s_batchjob_state is a hook used to delete job secrets if no batch jobs are enabled +k8s_batchjob_state: "{{ (k8s_predeploy_batchjobs + k8s_postdeploy_batchjobs) | map(attribute='enabled', default=true) | select | ternary('present', 'absent') }}" k8s_s3_cluster_name: "" # name of EKS cluster in AWS k8s_s3_region: "us-east-1" diff --git a/tasks/batchjob_tasks.yml b/tasks/batchjob_tasks.yml new file mode 100644 index 0000000..20e16fa --- /dev/null +++ b/tasks/batchjob_tasks.yml @@ -0,0 +1,28 @@ +- name: remove any old "{{ batchjob.job_name }}" jobs + k8s: + api_key: "{{ k8s_auth_api_key }}" + host: "{{ k8s_auth_host }}" + ca_cert: "{{ k8s_auth_ssl_ca_cert }}" + definition: "{{ lookup('template', 'batchjob.yaml.j2') }}" + state: absent + wait: yes + validate: + fail_on_error: yes + strict: yes + when: batchjob.enabled | default(true) +- name: run job "{{ batchjob.job_name }}" + k8s: + api_key: "{{ k8s_auth_api_key }}" + host: "{{ k8s_auth_host }}" + ca_cert: "{{ k8s_auth_ssl_ca_cert }}" + definition: "{{ lookup('template', 'batchjob.yaml.j2') }}" + state: present + wait: yes + wait_condition: + type: Complete + status: "True" + wait_timeout: "{{ batchjob.timeout | default(120) }}" + validate: + fail_on_error: yes + strict: yes + when: batchjob.enabled | default(true) diff --git a/tasks/main.yml b/tasks/main.yml index 1bf6383..826932b 100644 --- a/tasks/main.yml +++ b/tasks/main.yml @@ -158,69 +158,11 @@ type: Opaque stringData: "{{ k8s_environment_variables | from_yaml }}" -# Run migrations if wanted -- when: k8s_migrations_enabled - vars: - job_name: "migrate" - batch_command: "{{ k8s_migration_command }}" - block: - - name: remove any old migration jobs - k8s: - api_key: "{{ k8s_auth_api_key }}" - host: "{{ k8s_auth_host }}" - ca_cert: "{{ k8s_auth_ssl_ca_cert }}" - definition: "{{ lookup('template', 'batchjob.yaml.j2') }}" - state: absent - wait: yes - validate: - fail_on_error: yes - strict: yes - - name: run migrations - k8s: - api_key: "{{ k8s_auth_api_key }}" - host: "{{ k8s_auth_host }}" - ca_cert: "{{ k8s_auth_ssl_ca_cert }}" - definition: "{{ lookup('template', 'batchjob.yaml.j2') }}" - state: present - wait: yes - wait_condition: - type: Complete - status: "True" - validate: - fail_on_error: yes - strict: yes - -- when: k8s_collectstatic_enabled - vars: - job_name: "collectstatic" - batch_command: "{{ k8s_collectstatic_command }}" - block: - - name: remove any old collectstatic jobs - k8s: - api_key: "{{ k8s_auth_api_key }}" - host: "{{ k8s_auth_host }}" - ca_cert: "{{ k8s_auth_ssl_ca_cert }}" - definition: "{{ lookup('template', 'batchjob.yaml.j2') }}" - state: absent - wait: yes - validate: - fail_on_error: yes - strict: yes - - name: run collectstatic - k8s: - api_key: "{{ k8s_auth_api_key }}" - host: "{{ k8s_auth_host }}" - ca_cert: "{{ k8s_auth_ssl_ca_cert }}" - definition: "{{ lookup('template', 'batchjob.yaml.j2') }}" - state: present - wait: yes - wait_condition: - type: Complete - status: "True" - wait_timeout: "{{ k8s_collectstatic_timeout }}" - validate: - fail_on_error: yes - strict: yes +# Run each pre-deploy batch job in turn, if wanted +- loop: "{{ k8s_predeploy_batchjobs }}" + loop_control: + loop_var: batchjob + include_tasks: batchjob_tasks.yml - name: Create/update templates in Kubernetes k8s: @@ -272,3 +214,9 @@ command: kubectl --namespace "{{ k8s_namespace }}" rollout restart statefulset/celery-beat no_log: True when: k8s_worker_beat_enabled + +# Run each post-deploy batch job in turn, if wanted +- loop: "{{ k8s_postdeploy_batchjobs }}" + loop_control: + loop_var: batchjob + include_tasks: batchjob_tasks.yml diff --git a/templates/batchjob.yaml.j2 b/templates/batchjob.yaml.j2 index 3e694b7..b6fdb95 100644 --- a/templates/batchjob.yaml.j2 +++ b/templates/batchjob.yaml.j2 @@ -2,7 +2,7 @@ apiVersion: batch/v1 kind: Job metadata: namespace: "{{ k8s_namespace }}" - name: "{{ job_name }}" + name: "{{ batchjob.job_name }}" spec: template: spec: @@ -10,7 +10,7 @@ spec: - name: "{{ k8s_container_name }}" image: "{{ k8s_container_image }}:{{ k8s_container_image_tag }}" imagePullPolicy: "{{ k8s_container_image_pull_policy }}" - command: {{ batch_command }} + command: {{ batchjob.batch_command }} env: - name: GET_HOSTS_FROM value: dns