#!/usr/bin/env python3 # SPDX-FileCopyrightText: Copyright (c) 2025 NVIDIA CORPORATION & AFFILIATES. All rights reserved. # SPDX-License-Identifier: Apache-2.0 # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. """ Manifest Injection Script Copies any Kubernetes manifest file into the PVC for later use by jobs. Both the source manifest path and destination path in the PVC are required. IMPORTANT: The PVC is mounted at /data in the access pod for security reasons. All destination paths must start with '/data/'. Usage: python3 inject_manifest.py --namespace --src --dest Examples: python3 inject_manifest.py --namespace --src ./disagg.yaml --dest /data/configs/disagg.yaml python3 inject_manifest.py --namespace --src ./my-data.yaml --dest /data/custom/path/data.yaml """ import argparse import sys from pathlib import Path from deploy.utils.kubernetes import ( PVC_ACCESS_POD_NAME, check_kubectl_access, cleanup_access_pod, ensure_clean_access_pod, run_command, ) def copy_manifest(namespace: str, manifest_path: Path, target_path: str) -> None: """Copy a manifest file into the PVC via the access pod.""" pod_name = PVC_ACCESS_POD_NAME if not manifest_path.exists(): print(f"ERROR: Manifest file not found: {manifest_path}") sys.exit(1) print(f"Copying {manifest_path} to {target_path} in PVC...") # Ensure destination directory exists target_dir = str(Path(target_path).parent) run_command( ["kubectl", "exec", pod_name, "-n", namespace, "--", "mkdir", "-p", target_dir], capture_output=False, ) # Copy file to pod run_command( [ "kubectl", "cp", str(manifest_path), f"{namespace}/{pod_name}:{target_path}", ], capture_output=False, ) # Verify the file was copied result = run_command( ["kubectl", "exec", pod_name, "-n", namespace, "--", "ls", "-la", target_path], capture_output=True, ) print("āœ“ Manifest successfully copied to PVC") print(f"File details: {result.stdout.strip()}") def main(): parser = argparse.ArgumentParser( description="Inject a Kubernetes manifest into the PVC", formatter_class=argparse.RawDescriptionHelpFormatter, epilog=__doc__, ) parser.add_argument( "--namespace", "-n", required=True, help="Kubernetes namespace containing the profiling PVC", ) parser.add_argument( "--src", required=True, type=Path, help="Path to manifest file to copy" ) parser.add_argument( "--dest", required=True, help="Absolute target path in PVC (must start with /data/, e.g., /data/configs/agg.yaml)", ) args = parser.parse_args() # Validate target_path to prevent directory traversal and ensure it's within PVC if not args.dest.startswith("/data/"): print("=" * 60) print("āŒ ERROR: Invalid target path") print("=" * 60) print("The PVC is mounted at /data in the access pod.") print("All paths must start with '/data/' for security reasons.") print("") print("šŸ’” QUICK FIX:") if args.dest.startswith("/"): # Suggest the fix suggested_path = f"/data{args.dest}" print(f" Change: {args.dest}") print(f" To: {suggested_path}") print("") print("šŸ“ Example commands:") print(" python3 -m deploy.utils.inject_manifest \\") print(f" --namespace {args.namespace} \\") print(f" --src {args.src} \\") print(f" --dest {suggested_path}") else: print(f" Use: /data/{args.dest.lstrip('/')}") print("") print("šŸ” Common patterns:") print(" /configs/file.yaml → /data/configs/file.yaml") print(" /results/data.yaml → /data/results/data.yaml") print("=" * 60) sys.exit(1) if ".." in args.dest: print("ERROR: Target path cannot contain '..'") sys.exit(1) print("šŸš€ Manifest Injection") print("=" * 40) # Validate inputs check_kubectl_access(args.namespace) # Deploy access pod ensure_clean_access_pod(args.namespace) try: # Copy manifest copy_manifest(args.namespace, args.src, args.dest) print("\nāœ… Manifest injection completed!") print(f"šŸ“ File available at: {args.dest}") finally: # Cleanup even on failure cleanup_access_pod(args.namespace) if __name__ == "__main__": main()