Skip to content

Commit

Permalink
feat(sessions): expose user secrets in interactive sessions (#591)
Browse files Browse the repository at this point in the history
  • Loading branch information
mdonadoni committed Jun 26, 2024
1 parent a7c9c85 commit 9a08565
Show file tree
Hide file tree
Showing 3 changed files with 62 additions and 71 deletions.
81 changes: 41 additions & 40 deletions reana_workflow_controller/k8s.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@
current_k8s_corev1_api_client,
current_k8s_networking_api_client,
)
from reana_commons.k8s.secrets import REANAUserSecretsStore
from reana_commons.k8s.volumes import (
get_k8s_cvmfs_volumes,
get_workspace_volume,
Expand Down Expand Up @@ -77,6 +78,19 @@ def __init__(
name=deployment_name,
labels={"reana_workflow_mode": "session"},
)
self._session_container = client.V1Container(
name=self.deployment_name, image=self.image, env=[], volume_mounts=[]
)
self._pod_spec = client.V1PodSpec(
containers=[self._session_container],
volumes=[],
node_selector=REANA_RUNTIME_SESSIONS_KUBERNETES_NODE_LABEL,
# Disable service discovery with env variables, so that the environment is
# not polluted with variables like `REANA_SERVER_SERVICE_HOST`
enable_service_links=False,
automount_service_account_token=False,
)

self.kubernetes_objects = {
"ingress": self._build_ingress(),
"service": self._build_service(metadata),
Expand Down Expand Up @@ -149,15 +163,6 @@ def _build_deployment(self, metadata):
:param metadata: Common Kubernetes metadata for the interactive
deployment.
"""
container = client.V1Container(name=self.deployment_name, image=self.image)
pod_spec = client.V1PodSpec(
containers=[container],
node_selector=REANA_RUNTIME_SESSIONS_KUBERNETES_NODE_LABEL,
# Disable service discovery with env variables, so that the environment is
# not polluted with variables like `REANA_SERVER_SERVICE_HOST`
enable_service_links=False,
automount_service_account_token=False,
)
labels = {
"app": self.deployment_name,
"reana_workflow_mode": "session",
Expand All @@ -166,7 +171,7 @@ def _build_deployment(self, metadata):
}
template = client.V1PodTemplateSpec(
metadata=client.V1ObjectMeta(labels=labels),
spec=pod_spec,
spec=self._pod_spec,
)
spec = client.V1DeploymentSpec(
selector=client.V1LabelSelector(match_labels=labels),
Expand All @@ -184,36 +189,26 @@ def _build_deployment(self, metadata):

def add_command(self, command):
"""Add a command to the deployment."""
self.kubernetes_objects["deployment"].spec.template.spec.containers[
0
].command = command
self._session_container.command = command

def add_command_arguments(self, args):
"""Add command line arguments in addition to the command."""
self.kubernetes_objects["deployment"].spec.template.spec.containers[
0
].args = args
self._session_container.args = args

def add_reana_shared_storage(self):
"""Add the REANA shared file system volume mount to the deployment."""
volume_mount, volume = get_workspace_volume(self.workspace)
self.kubernetes_objects["deployment"].spec.template.spec.containers[
0
].volume_mounts = [volume_mount]
self.kubernetes_objects["deployment"].spec.template.spec.volumes = [volume]
self._session_container.volume_mounts.append(volume_mount)
self._pod_spec.volumes.append(volume)

def add_cvmfs_repo_mounts(self, cvmfs_repos):
"""Add mounts for the provided CVMFS repositories to the deployment.
:param cvmfs_mounts: List of CVMFS repos to make available.
"""
cvmfs_volume_mounts, cvmfs_volumes = get_k8s_cvmfs_volumes(cvmfs_repos)
self.kubernetes_objects["deployment"].spec.template.spec.volumes.extend(
cvmfs_volumes
)
self.kubernetes_objects["deployment"].spec.template.spec.containers[
0
].volume_mounts.extend(cvmfs_volume_mounts)
self._pod_spec.volumes.extend(cvmfs_volumes)
self._session_container.volume_mounts.extend(cvmfs_volume_mounts)

def add_environment_variable(self, name, value):
"""Add an environment variable.
Expand All @@ -222,24 +217,25 @@ def add_environment_variable(self, name, value):
:param value: Environment variable value.
"""
env_var = client.V1EnvVar(name, str(value))
if isinstance(
self.kubernetes_objects["deployment"].spec.template.spec.containers[0].env,
list,
):
self.kubernetes_objects["deployment"].spec.template.spec.containers[
0
].env.append(env_var)
else:
self.kubernetes_objects["deployment"].spec.template.spec.containers[
0
].env = [env_var]
self._session_container.env.append(env_var)

def add_run_with_root_permissions(self):
"""Run interactive session with root."""
security_context = client.V1SecurityContext(run_as_user=0)
self.kubernetes_objects["deployment"].spec.template.spec.containers[
0
].security_context = security_context
self._session_container.security_context = security_context

def add_user_secrets(self):
"""Mount the "file" secrets and set the "env" secrets in the container."""
secrets_store = REANAUserSecretsStore(self.owner_id)

# mount file secrets
secrets_volume = secrets_store.get_file_secrets_volume_as_k8s_specs()
secrets_volume_mount = secrets_store.get_secrets_volume_mount_as_k8s_spec()
self._pod_spec.volumes.append(secrets_volume)
self._session_container.volume_mounts.append(secrets_volume_mount)

# set environment secrets
self._session_container.env += secrets_store.get_env_secrets_as_k8s_spec()

def get_deployment_objects(self):
"""Return the alrady built Kubernetes objects."""
Expand All @@ -255,6 +251,7 @@ def build_interactive_jupyter_deployment_k8s_objects(
owner_id=None,
workflow_id=None,
image=None,
expose_secrets=True,
):
"""Build the Kubernetes specification for a Jupyter NB interactive session.
Expand All @@ -276,6 +273,8 @@ def build_interactive_jupyter_deployment_k8s_objects(
session belongs to.
:param image: Jupyter Notebook image to use, i.e.
``jupyter/tensorflow-notebook`` to enable ``tensorflow``.
:param expose_secrets: If true, mount the "file" secrets and set the
"env" secrets in jupyter's pod.
"""
image = image or JUPYTER_INTERACTIVE_SESSION_DEFAULT_IMAGE
cvmfs_repos = cvmfs_repos or []
Expand All @@ -297,6 +296,8 @@ def build_interactive_jupyter_deployment_k8s_objects(
deployment_builder.add_reana_shared_storage()
if cvmfs_repos:
deployment_builder.add_cvmfs_repo_mounts(cvmfs_repos)
if expose_secrets:
deployment_builder.add_user_secrets()
deployment_builder.add_environment_variable("NB_GID", 0)
# Changes umask so all files generated by the Jupyter Notebook can be
# modified by the root group users.
Expand Down
51 changes: 20 additions & 31 deletions reana_workflow_controller/rest/workflows_session.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,9 @@


from flask import Blueprint, jsonify, request
from webargs import fields
from webargs.flaskparser import use_kwargs

from reana_db.utils import _get_workflow_with_uuid_or_name
from reana_db.models import WorkflowSession, InteractiveSessionType, RunStatus

Expand All @@ -20,10 +23,14 @@


@blueprint.route(
"/workflows/<workflow_id_or_name>/open/" "<interactive_session_type>",
"/workflows/<workflow_id_or_name>/open/<interactive_session_type>",
methods=["POST"],
)
def open_interactive_session(workflow_id_or_name, interactive_session_type): # noqa
@use_kwargs({"user": fields.Str(required=True)}, location="query")
@use_kwargs({"image": fields.Str()}, location="json")
def open_interactive_session(
workflow_id_or_name, interactive_session_type, user, **kwargs
): # noqa
r"""Start an interactive session inside the workflow workspace.
---
Expand Down Expand Up @@ -109,45 +116,27 @@ def open_interactive_session(workflow_id_or_name, interactive_session_type): #
"""
try:
if interactive_session_type not in InteractiveSessionType.__members__:
return (
jsonify(
{
"message": "Interactive session type {0} not found, try "
"with one of: {1}".format(
interactive_session_type,
[e.name for e in InteractiveSessionType],
)
}
),
404,
error_msg = (
f"Interactive session type {interactive_session_type} not found, "
f"try with one of: {[e.name for e in InteractiveSessionType]}"
)
interactive_session_configuration = request.json if request.is_json else {}
user_uuid = request.args["user"]
workflow = None
workflow = _get_workflow_with_uuid_or_name(workflow_id_or_name, user_uuid)
return jsonify({"message": error_msg}), 404

workflow = _get_workflow_with_uuid_or_name(workflow_id_or_name, user_uuid=user)

if workflow.sessions.first() is not None:
return (
jsonify({"message": "Interactive session is already open"}),
404,
)
return jsonify({"message": "Interactive session is already open"}), 404

if workflow.status == RunStatus.deleted:
return (
jsonify(
{
"message": "Interactive session can't be opened from a deleted workflow"
}
),
404,
)
error_msg = "Interactive session can't be opened from a deleted workflow"
return jsonify({"message": error_msg}), 404

kwrm = KubernetesWorkflowRunManager(workflow)
access_path = kwrm.start_interactive_session(
interactive_session_type,
image=interactive_session_configuration.get("image", None),
image=kwargs.get("image"),
)
return jsonify({"path": "{}".format(access_path)}), 200
return jsonify({"path": str(access_path)}), 200

except (KeyError, ValueError) as e:
status_code = 400 if workflow else 404
Expand Down
1 change: 1 addition & 0 deletions reana_workflow_controller/workflow_run_manager.py
Original file line number Diff line number Diff line change
Expand Up @@ -343,6 +343,7 @@ def start_interactive_session(self, interactive_session_type, **kwargs):
cvmfs_repos=self.retrieve_required_cvmfs_repos(),
owner_id=self.workflow.owner_id,
workflow_id=self.workflow.id_,
expose_secrets=True,
**kwargs,
)

Expand Down

0 comments on commit 9a08565

Please sign in to comment.