Updating screen session environment variables to reflect new graphical login?

I use linux, and I like to do all my command-line work within a single screen session, so that I can restart my graphical login and such without losing my terminals. However, when I log out and back into my graphical session, this changes all my session environment variables, such as DBus sessions. This means that after logging in again, my screen session now has the old (and wrong) environment variables. So now when I try to start graphical programs from my screen session, at best they emit a warning about not being able to connect to the session bus. At worst, they fail to start completely.

So, what I'm looking for is a way to modify environment variables in a running instance of screen, so that all subsequently-created screen windows will inherit the new environment variables. Is there a way to do this?


You cannot start a shell script from the screen session since it would inherit the old environment. You can however us a fifo to get the new environment variables into the old screen session. You can fill that fifo when you start your graphical session.

#!/bin/bash
FIFO=/tmp/your_variables
[ -e $FIFO ] && cat $FIFO > /dev/null || mkfifo $FIFO

# save number of variables that follow
NVARS=2
echo $NVARS > $FIFO
echo ENV1=sth1 > $FIFO
echo ENV2=sth2 > $FIFO

Start that script in the background on login (it will only terminate when all variables are read from it).

Now you can read from the fifo, e.g. add this function to your .bashrc

update_session() {
  FIFO=/tmp/your_variables

  NVAR=$(cat $FIFO)
  for i in $(seq $NVAR); do
    export $(cat $FIFO)
  done
  #delete the pipe, or it will not work next time 
  rm $FIFO
}

so that you can in your old screen session

update_session

You can invoke the setenv command to change environment variables in the screen process interactively, by using: Ctrl-A+:setenv (Note the : character to enter a screen command.) You will be prompted for the environment variable name and value.

Note that (as per other answers/comments) this affects the (parent) screen process and therefore newly-created screen sessions, but not your current screen session nor any existing screen sessions.

You can specify the environment variable name and value at the same time if you want: Ctrl-A+:setenv DISPLAY :100. Will set the DISPLAY to ":100" for new screen sessions.

To remove an environment variable you can use 'unsetenv' - e.g. Ctrl-A+:unsetenv DISPLAY


I have implemented a script to do this. You can get it here: https://github.com/DarwinAwardWinner/screen-sendenv

After putting screen-sendenv.py into your $PATH, you can use the following snippet in your .bashrc:

VARS_TO_UPDATE="DISPLAY DBUS_SESSION_BUS_ADDRESS SESSION_MANAGER GPG_AGENT_INFO"
screen_pushenv () {
  screen-sendenv.py -t screen $VARS_TO_UPDATE
}
tmux_pushenv () {
  screen-sendenv.py -t tmux $VARS_TO_UPDATE
}
screen_pullenv () {
  tempfile=$(mktemp -q) && {
    for var in $VARS_TO_UPDATE; do
      screen sh -c "echo export $var=\$$var >> \"$tempfile\""
    done
    . "$tempfile"
    rm -f "$tempfile"
  }
}
tmux_pullenv () {
  for var in $VARS_TO_UPDATE; do
    expr="$(tmux showenv | grep "^$var=")"
    if [ -n "$expr" ]; then
      export "$expr"
    fi
  done
}

To use it, just run screen_pushenv before you run screen -r to reattach to your screen session. Then, after attaching with screen -r, you can update the environment in your existing shells with screen_pullenv. The tmux functions accomplish the same thing for tmux, another terminal multiplexer similar to screen.


This is probably a simpler solution (you decide). The important part being the alias that calls the savedisplay function every time screen command is run. The commands are not run automatically, hence can be put in ~/.bashrc instead of something very specialized like ~/.ssh/rc.

savedisplay() {
    # Write latest bash display to a file, This is used to 
    # update running bash sessions for a "screen -r"
    echo "export DISPLAY=$DISPLAY" > ~/.XDISPLAY
    echo "export XAUTHORITY=$XAUTHORITY" >> ~/.XDISPLAY
    # This will only update the environment for new windows
    screen -X setenv DISPLAY $DISPLAY
    screen -X setenv XAUTHORITY $XAUTHORITY
}

# run this to update env variable in a running session
updatedisplay() {
    source ~/.XDISPLAY 
}

alias screen='savedisplay && screen'

This thread is very old but the problem persists. I thought I'd post a slightly more robust version of the answer posted by @wecac. I put this into my .bashrc:

# Save X display-related environment variables for use in `screen` sessions.
savescreenenv() {
    # Write a bash script to update the environment.  The script is named for
    # the host so that `screen` sessions on different hosts can have different
    # environments.
    mkdir -p ~/.screenenv
    envfile=~/.screenenv/$(hostname)

    # Any output within this loop is captured to the $envfile script.
    for var in DISPLAY XAUTHORITY DBUS_SESSION_BUS_ADDRESS \
                       SESSION_MANAGER GPG_AGENT_INFO
    do
        # For each non-empty environment variable, write commands to:
        #  1. Restore the environment variable
        #  2. Set the environment variable in the current `screen` session.
        if [ ! -z "${!var}" ]; then
            echo "export $var='${!var}'"
            echo "screen -X setenv $var '${!var}'"
        fi
    done > $envfile
}

# Restore environment set by `savescreenenv`.  Run this command manually in
# each screen sub-terminal after disconnecting and re-connecting to `screen`.
updateenv() {
    envfile=~/.screenenv/$(hostname)
    if [ -f $envfile ]; then
        . $envfile
    fi
}

# Alias `screen` to save select environment variables first.
# Do not alias `screen` from within a screen session else old variables could
# be written to $envfile by accident.
unalias screen 2> /dev/null
if ! ${IN_SCREEN:-false}; then
    alias screen='savescreenenv && IN_SCREEN=true screen'
fi

The trick is to remember to run updateenv manually in each existing screen sub-terminal upon reconnecting. New sub-terminals will automatically get the correct environment if updateenv was run once in at least one existing sub-terminal.