{
  "id": "django__django-10973",
  "question": "Use subprocess.run and PGPASSWORD for client in postgres backend\nDescription\n\t\n​subprocess.run was added in python 3.5 (which is the minimum version since Django 2.1). This function allows you to pass a custom environment for the subprocess.\nUsing this in django.db.backends.postgres.client to set PGPASSWORD simplifies the code and makes it more reliable.\n",
  "answer": "{\"task_id\": \"3af9c206-f947-44a1-800c-fbff867abee8\", \"workflow_run_id\": \"a0afe630-f134-4362-a6f4-92cfb8dfbbd1\", \"data\": {\"id\": \"a0afe630-f134-4362-a6f4-92cfb8dfbbd1\", \"workflow_id\": \"9a0a6efa-4d15-4876-b13f-4fb6a999caa2\", \"status\": \"succeeded\", \"outputs\": {\"final_patch\": [\"--- a/django/db/backends/postgresql/client.py\\n+++ b/django/db/backends/postgresql/client.py\\n@@ -1,4 +1,5 @@\\n import os\\n+import subprocess\\n from django.db.backends.base.client import BaseDatabaseClient\\n \\n \\n@@ -37,7 +38,13 @@ class DatabaseClient(BaseDatabaseClient):\\n         args = [self.executable_name, *self.settings_to_cmd_args_env(parameters)[0]]\\n         sigint_handler = signal.signal(signal.SIGINT, signal.SIG_IGN)\\n         try:\\n-            subprocess.call(args, env=env)\\n+            # Create custom environment with PGPASSWORD\\n+            custom_env = os.environ.copy()\\n+            if parameters.get('password'):\\n+                custom_env['PGPASSWORD'] = parameters['password']\\n+            # Update with any additional environment variables\\n+            custom_env.update(env)\\n+            subprocess.run(args, env=custom_env, check=False)\\n         finally:\\n             signal.signal(signal.SIGINT, sigint_handler)\", \"--- a/django/db/backends/postgresql/client.py\\n+++ b/django/db/backends/postgresql/client.py\\n@@ -1,4 +1,5 @@\\n import os\\n+import subprocess\\n from django.db.backends.base.client import BaseDatabaseClient\\n \\n \\n@@ -37,7 +38,16 @@ class DatabaseClient(BaseDatabaseClient):\\n         args = [self.executable_name, *self.settings_to_cmd_args_env(parameters)[0]]\\n         sigint_handler = signal.signal(signal.SIGINT, signal.SIG_IGN)\\n         try:\\n-            subprocess.call(args, env=env)\\n+            # Prepare environment with PGPASSWORD\\n+            run_env = os.environ.copy()\\n+            run_env.update(env)  # Add any environment from settings_to_cmd_args_env\\n+            \\n+            # Set PGPASSWORD if password is available\\n+            password = parameters.get('password')\\n+            if password:\\n+                run_env['PGPASSWORD'] = password\\n+            \\n+            subprocess.run(args, env=run_env)\\n         finally:\\n             signal.signal(signal.SIGINT, sigint_handler)\", \"--- a/django/db/backends/postgresql/client.py\\n+++ b/django/db/backends/postgresql/client.py\\n@@ -1,4 +1,5 @@\\n import os\\n+import subprocess\\n from django.db.backends.base.client import BaseDatabaseClient\\n \\n \\n@@ -37,7 +38,11 @@ class DatabaseClient(BaseDatabaseClient):\\n         args = [self.executable_name, *self.settings_to_cmd_args_env(parameters)[0]]\\n         sigint_handler = signal.signal(signal.SIGINT, signal.SIG_IGN)\\n         try:\\n-            subprocess.call(args, env=env)\\n+            # Use subprocess.run with PGPASSWORD in environment\\n+            run_env = {**os.environ, **env}\\n+            if parameters.get('password'):\\n+                run_env['PGPASSWORD'] = parameters['password']\\n+            subprocess.run(args, env=run_env)\\n         finally:\\n             signal.signal(signal.SIGINT, sigint_handler)\", \"--- a/django/db/backends/postgresql/client.py\\n+++ b/django/db/backends/postgresql/client.py\\n@@ -1,4 +1,5 @@\\n import os\\n+import subprocess\\n from django.db.backends.base.client import BaseDatabaseClient\\n \\n \\n@@ -34,10 +35,18 @@ class DatabaseClient(BaseDatabaseClient):\\n     def runshell(self, parameters):\\n         args, env = self.settings_to_cmd_args_env(parameters)\\n         args = [self.executable_name, *args]\\n+        \\n+        # Prepare environment for subprocess\\n+        subprocess_env = os.environ.copy()\\n+        subprocess_env.update(env)\\n+        \\n+        # Set PGPASSWORD to avoid password prompts\\n+        if parameters.get('password'):\\n+            subprocess_env['PGPASSWORD'] = parameters['password']\\n+        \\n         sigint_handler = signal.signal(signal.SIGINT, signal.SIG_IGN)\\n         try:\\n-            subprocess.call(args, env=env)\\n+            subprocess.run(args, env=subprocess_env)\\n         finally:\\n             signal.signal(signal.SIGINT, sigint_handler)\", \"--- a/django/db/backends/postgresql/client.py\\n+++ b/django/db/backends/postgresql/client.py\\n@@ -1,4 +1,5 @@\\n import os\\n+import subprocess\\n from django.db.backends.base.client import BaseDatabaseClient\\n \\n \\n@@ -37,7 +38,14 @@ class DatabaseClient(BaseDatabaseClient):\\n         args = [self.executable_name, *self.settings_to_cmd_args_env(parameters)[0]]\\n         sigint_handler = signal.signal(signal.SIGINT, signal.SIG_IGN)\\n         try:\\n-            subprocess.call(args, env=env)\\n+            # Build environment with PGPASSWORD for secure authentication\\n+            shell_env = dict(os.environ)\\n+            shell_env.update(env)\\n+            \\n+            if 'password' in parameters and parameters['password']:\\n+                shell_env['PGPASSWORD'] = parameters['password']\\n+            \\n+            subprocess.run(args, env=shell_env, check=False)\\n         finally:\\n             signal.signal(signal.SIGINT, sigint_handler)\", \"diff\\n--- a/django/db/backends/postgresql/client.py\\n+++ b/django/db/backends/postgresql/client.py\\n@@ -1,4 +1,5 @@\\n import os\\n+import subprocess\\n from django.db.backends.base.client import BaseDatabaseClient\\n \\n \\n@@ -37,7 +38,16 @@ class DatabaseClient(BaseDatabaseClient):\\n         args = [self.executable_name, *self.settings_to_cmd_args_env(parameters)[0]]\\n         sigint_handler = signal.signal(signal.SIGINT, signal.SIG_IGN)\\n         try:\\n-            subprocess.call(args, env=env)\\n+            # Prepare environment with PGPASSWORD\\n+            run_env = os.environ.copy()\\n+            run_env.update(env)  # Add any environment from settings_to_cmd_args_env\\n+            \\n+            # Set PGPASSWORD if password is available\\n+            password = parameters.get('password')\\n+            if password:\\n+                run_env['PGPASSWORD'] = password\\n+            \\n+            subprocess.run(args, env=run_env)\\n         finally:\\n             signal.signal(signal.SIGINT, sigint_handler)\\n\", \"diff\\n--- a/django/db/backends/postgresql/client.py\\n+++ b/django/db/backends/postgresql/client.py\\n@@ -1,4 +1,5 @@\\n import os\\n+import subprocess\\n from django.db.backends.base.client import BaseDatabaseClient\\n \\n \\n@@ -34,10 +35,18 @@ class DatabaseClient(BaseDatabaseClient):\\n     def runshell(self, parameters):\\n         args, env = self.settings_to_cmd_args_env(parameters)\\n         args = [self.executable_name, *args]\\n+        \\n+        # Prepare environment for subprocess\\n+        subprocess_env = os.environ.copy()\\n+        subprocess_env.update(env)\\n+        \\n+        # Set PGPASSWORD to avoid password prompts\\n+        if parameters.get('password'):\\n+            subprocess_env['PGPASSWORD'] = parameters['password']\\n+        \\n         sigint_handler = signal.signal(signal.SIGINT, signal.SIG_IGN)\\n         try:\\n-            subprocess.call(args, env=env)\\n+            subprocess.run(args, env=subprocess_env)\\n         finally:\\n             signal.signal(signal.SIGINT, sigint_handler)\\n\", \"diff\\n--- a/django/db/backends/postgresql/client.py\\n+++ b/django/db/backends/postgresql/client.py\\n@@ -1,4 +1,5 @@\\n import os\\n+import subprocess\\n from django.db.backends.base.client import BaseDatabaseClient\\n \\n \\n@@ -34,10 +35,18 @@ class DatabaseClient(BaseDatabaseClient):\\n     def runshell(self, parameters):\\n         args, env = self.settings_to_cmd_args_env(parameters)\\n         args = [self.executable_name, *args]\\n+        \\n+        # Prepare environment for subprocess\\n+        subprocess_env = os.environ.copy()\\n+        subprocess_env.update(env)\\n+        \\n+        # Set PGPASSWORD to avoid password prompts\\n+        if parameters.get('password'):\\n+            subprocess_env['PGPASSWORD'] = parameters['password']\\n+        \\n         sigint_handler = signal.signal(signal.SIGINT, signal.SIG_IGN)\\n         try:\\n-            subprocess.call(args, env=env)\\n+            subprocess.run(args, env=subprocess_env)\\n         finally:\\n             signal.signal(signal.SIGINT, sigint_handler)\\n\"]}, \"error\": \"\", \"elapsed_time\": 94.939403, \"total_tokens\": 36079, \"total_steps\": 30, \"created_at\": 1754646914, \"finished_at\": 1754647009}}"
}