diff --git a/roles/helm-apps/README.md b/roles/helm-apps/README.md
new file mode 100644
index 0000000000000000000000000000000000000000..27b480cb0e753481f2253da06af4d611d94fa35d
--- /dev/null
+++ b/roles/helm-apps/README.md
@@ -0,0 +1,39 @@
+Role Name
+=========
+
+This role is intended to be used to fetch and deploy Helm Charts as part of
+cluster installation or upgrading with kubespray.
+
+Requirements
+------------
+
+The role needs to be executed on a host with access to the Kubernetes API, and
+with the helm binary in place.
+
+Role Variables
+--------------
+
+See meta/argument_specs.yml
+
+Playbook example:
+
+```yaml
+---
+- hosts: kube_control_plane[0]
+  gather_facts: no
+  roles:
+    - name: helm-apps
+      releases:
+        - name: app
+          namespace: app
+          chart_ref: simple-app/simple-app
+        - name: app2
+          namespace: app
+          chart_ref: simple-app/simple-app
+          wait_timeout: "10m" # override the same option in `release_common_opts`
+      repositories: "{{ repos }}"
+        - repo_name: simple-app
+          repo_url: "https://blog.leiwang.info/simple-app"
+      release_common_opts: "{{ helm_params }}"
+        wait_timeout: "5m"
+```
diff --git a/roles/helm-apps/meta/argument_specs.yml b/roles/helm-apps/meta/argument_specs.yml
new file mode 100644
index 0000000000000000000000000000000000000000..d1be9a81520979441ac09faba77019dfb299973d
--- /dev/null
+++ b/roles/helm-apps/meta/argument_specs.yml
@@ -0,0 +1,93 @@
+---
+argument_specs:
+  main:
+    short_description: Install a list of Helm charts.
+    options:
+      releases:
+        type: list
+        elements: dict
+        required: true
+        description: |
+          List of dictionaries passed as arguments to kubernetes.core.helm.
+          Arguments passed here will override  those in `helm_settings`.  For
+          structure of the dictionary, see the documentation for
+          kubernetes.core.helm ansible module.
+        options:
+          chart_ref:
+            type: path
+            required: true
+          chart_version:
+            type: str
+          name:
+            type: str
+            required: true
+          namespace:
+            type: str
+            required: true
+          values:
+            type: dict
+          # Possibly general options
+          create_namespace:
+            type: bool
+          chart_repo_url:
+            type: str
+          disable_hook:
+            type: bool
+          history_max:
+            type: int
+          purge:
+            type: bool
+          replace:
+            type: bool
+          skip_crds:
+            type: bool
+          wait:
+            type: bool
+            default: true
+          wait_timeout:
+            type: str
+
+      repositories:
+        type: list
+        elements: dict
+        description: |
+          List of dictionaries passed as arguments to
+          kubernetes.core.helm_repository.
+        default: []
+        options:
+          name:
+            type: str
+            required: true
+          password:
+            type: str
+          username:
+            type: str
+          url:
+            type: str
+      release_common_opts:
+        type: dict
+        description: |
+          Common arguments for every helm invocation.
+        default: {}
+        options:
+          create_namespace:
+            type: bool
+            default: true
+          chart_repo_url:
+            type: str
+          disable_hook:
+            type: bool
+          history_max:
+            type: int
+          purge:
+            type: bool
+          replace:
+            type: bool
+          skip_crds:
+            type: bool
+          wait:
+            type: bool
+            default: true
+          wait_timeout:
+            type: str
+            default: "5m"
diff --git a/roles/helm-apps/tasks/main.yml b/roles/helm-apps/tasks/main.yml
new file mode 100644
index 0000000000000000000000000000000000000000..ed55c5ae8b09c2a6114974e6bd4bde04d44100de
--- /dev/null
+++ b/roles/helm-apps/tasks/main.yml
@@ -0,0 +1,17 @@
+---
+- name: Add Helm repositories
+  kubernetes.core.helm_repository: "{{ helm_repository_defaults | combine(item) }}"
+  loop: "{{ repositories }}"
+
+- name: Update Helm repositories
+  kubernetes.core.helm:
+    state: absent
+    binary_path: "{{ bin_dir }}/helm"
+    release_name: dummy  # trick needed to refresh in separate step
+    release_namespace: kube-system
+    update_repo_cache: true
+  when: repositories != []
+
+- name: Install Helm Applications
+  kubernetes.core.helm: "{{ helm_defaults | combine(release_common_opts, item) }}"
+  loop: "{{ releases }}"
diff --git a/roles/helm-apps/vars/main.yml b/roles/helm-apps/vars/main.yml
new file mode 100644
index 0000000000000000000000000000000000000000..a7baa66bd4dcaffa3410fb0728f3399648ac64e8
--- /dev/null
+++ b/roles/helm-apps/vars/main.yml
@@ -0,0 +1,7 @@
+---
+helm_defaults:
+  atomic: true
+  binary_path: "{{ bin_dir }}/helm"
+
+helm_repository_defaults:
+  binary_path: "{{ bin_dir }}/helm"