Hello,

I'd like to ask about an odd behaviour relating to the SHELLOPTS environment 
variable. This is present in version 5.2.37(1)-release and in the development 
version.

In this case, the `nounset` option is only applied to the current shell and not 
to nested bash invocations:
> set -o nounset; echo $SHELLOPTS; bash -c "echo \$SHELLOPTS"

On the other hand, if there is a pre-existing SHELLOPTS environment variable 
(set outside the current shell), then the `nounset` option applies to all 
invocations of bash within the current shell. Even with the `+u` flag, it is 
not possible to avoid the `nounset` option being applied to the inner bash 
invocation.

The specific case where I found this behaviour to be problematic is in 
setup-ocaml, which is a github action: 
https://github.com/ocaml/setup-ocaml/pull/915. It applies "SHELLOPTS=igncr" 
globally so that it affects all cygwin bash invocations, however, a side effect 
is that other shell options set via bash option flags cascade down into nested 
bash invocations. Specifically, the `nounset` option applies to places where it 
didn't previously apply (which breaks a configure script of libuv). I know 
`igncr` only applies to cygwin bash, but it can be substituted for any option 
in regular bash and the same behaviour occurs.

To be clear, the intention here was to apply a specific shell option globally 
(namely igncr), and the unintentional side effect was that the behaviour of how 
other shell options are applied has changed.

I'm not sure if this behaviour of SHELLOPTS is intentional, but it is 
definitely confusing particularly from the lens of the given use case. This 
behaviour is not explicitly documented either: 
https://www.gnu.org/software/bash/manual/html_node/Bash-Variables.html#index-SHELLOPTS.

It seems the solution would be to either:
1) Document this cascading behaviour of SHELLOPTS, and decide on the correct 
behaviour of +u versus SHELLOPTS=nounset.
or
2) Re-export only the initial value of SHELLOPTS to the new environment, prior 
to modifications due to `set` or bash option flags. I've added a patch below 
that achieves this, if it seems reasonable then I will write a test and format 
it properly as a patch request.

Regards,
Tobi
---
 variables.c | 21 +++++++++++++++++++--
 1 file changed, 19 insertions(+), 2 deletions(-)

diff --git a/variables.c b/variables.c
index 1a0c2c45..ba65b97f 100644
--- a/variables.c
+++ b/variables.c
@@ -177,6 +177,11 @@ char **export_env = (char **)NULL;
 static int export_env_index;
 static int export_env_size;
 
+/* For the SHELLOPTS environment variable, we must save and re-export
+   the original value as at initialization, ignoring changes resulting from
+   other methods of modifying shell options. */
+static char *shell_opts_value = (char *)NULL;
+
 #if defined (READLINE)
 static int winsize_assignment;         /* currently assigning to LINES or 
COLUMNS */
 #endif
@@ -515,7 +520,12 @@ initialize_shell_variables (env, privmode)
              temp_var = bind_variable (name, string, 0);
              if (temp_var)
                {
-                 VSETATTR (temp_var, (att_exported | att_imported));
+                 VSETATTR (temp_var, att_imported);
+                 if (STREQ (name, "SHELLOPTS"))
+                   shell_opts_value = string;
+                 else
+                   VSETATTR (temp_var, att_exported);
+
                  if (ro)
                    VSETATTR (temp_var, att_readonly);
                }
@@ -5129,7 +5139,14 @@ maybe_make_export_env ()
          export_env = strvec_resize (export_env, export_env_size);
          environ = export_env;
        }
-      export_env[export_env_index = 0] = (char *)NULL;
+
+      if (shell_opts_value)
+       {
+         export_env[export_env_index = 0] = mk_env_string ("SHELLOPTS", 
shell_opts_value, 0);
+         export_env[export_env_index = 1] = (char *)NULL;
+        }
+      else
+        export_env[export_env_index = 0] = (char *)NULL;
 
       /* Make a dummy variable context from the temporary_env, stick it on
         the front of shell_variables, call make_var_export_array on the
---

Reply via email to