Create a pipe that writes to multiple files (tee)

后端 未结 5 1765
情话喂你
情话喂你 2021-02-09 14:52

I would like to create a pipe in a ksh script (using exec) that pipe\'s to a tee, and sends the output to a pipe.

Current:

#Redirect EVE         


        
5条回答
  •  自闭症患者
    2021-02-09 15:34

    Here's a solution I use. It works under ksh on my Mac. It's nicely encapsulated into start_logging() and stop_logging() functions to make life easy.

    The code looks like this in practice:

    # Optional:
    #   Set the name and location of the log file.
    #   OUTPUT_LOG=output.log    # default
    #   Set the name and location of the named pipe used.
    #   OUTPUT_PIPE=output.pipe  # default
    
    start_logging
    # Default is to append to an existing log file.
    # start_logging delete_existing_logfile
    echo "This is on standard out"
    echo "This is on standard err" >&2
    stop_logging
    

    Here is the whole file. The start and stop functions along with the example above are all at the bottom of the file. To make it easier to use, just put the start and stop functions in their own file and source them in the scripts where you need the logging.

    #!/bin/sh
    
    # Author: Harvey Chapman 
    # Description: POSIX shell functions that can be used with tee to simultaneously put
    #              stderr and stdout to both a file and stdout
    #
    # Based on:
    #    Re: How to redirect stderr and stdout to a file plus display at the same time
    #    http://www.travishartwell.net/blog/2006/08/19_2220
    
    #
    # Original example function from Travis Hartwell's blog.
    # Note: I've made minor changes to it.
    example()
    {
      OUTPUT_LOG=output.log
      OUTPUT_PIPE=output.pipe
    
      # This should really be -p to test that it's a pipe.
      if [ ! -e $OUTPUT_PIPE ]; then
          mkfifo $OUTPUT_PIPE
      fi
    
      # This should really be -f to test that it's a regular file.
      if [ -e $OUTPUT_LOG ]; then
          rm $OUTPUT_LOG
      fi
    
      exec 3>&1 4>&2
      tee $OUTPUT_LOG < $OUTPUT_PIPE >&3 &
      tpid=$!
      exec > $OUTPUT_PIPE 2>&1
    
      echo "This is on standard out"
      echo "This is on standard err" >&2
    
      exec 1>&3 3>&- 2>&4 4>&-
      wait $tpid
    
      rm $OUTPUT_PIPE
    }
    
    # A slightly reduced version of example()
    example2()
    {
      OUTPUT_LOG=output.log
      OUTPUT_PIPE=output.pipe
    
      rm -f $OUTPUT_PIPE
      mkfifo $OUTPUT_PIPE
      rm -f $OUTPUT_LOG
    
      tee $OUTPUT_LOG < $OUTPUT_PIPE &
      tpid=$!
    
      exec 3>&1 4>&2 >$OUTPUT_PIPE 2>&1
    
      echo "This is on standard out"
      echo "This is on standard err" >&2
    
      exec 1>&3 3>&- 2>&4 4>&-
      wait $tpid
      rm -f $OUTPUT_PIPE
    }
    
    #
    # Logging methods based on above. See the example below for how to use them.
    #
    
    # Usage: start_logging [delete_existing_logfile]
    start_logging()
    {
      # Check to see if OUTPUT_LOG and OUTPUT_PIPE need to be defined.
      if [ -z "$OUTPUT_LOG" ]; then
        OUTPUT_LOG=output.log
      fi
      if [ -z "$OUTPUT_PIPE" ]; then
        OUTPUT_PIPE=output.pipe
      fi
      # Make sure that we're not already logging.
      if [ -n "$OUTPUT_PID" ]; then
        echo "Logging already started!"
        return 1
      fi
    
      # Always remove the log and pipe first.
      rm -f $OUTPUT_PIPE
      # Delete the logfile first if told to.
      if [ "$1" = delete_existing_logfile ]; then
        rm -f $OUTPUT_LOG
      fi
    
      mkfifo $OUTPUT_PIPE
      tee -a $OUTPUT_LOG < $OUTPUT_PIPE &
      OUTPUT_PID=$!
    
      exec 3>&1 4>&2 >$OUTPUT_PIPE 2>&1
    }
    
    stop_logging()
    {
      # Make sure that we're currently logging.
      if [ -z "$OUTPUT_PID" ]; then
        echo "Logging not yet started!"
        return 1
      fi
      exec 1>&3 3>&- 2>&4 4>&-
      wait $OUTPUT_PID
      rm -f $OUTPUT_PIPE
      unset OUTPUT_PID
    }
    
    example3()
    {
      start_logging
      #start_logging delete_existing_logfile
      echo "This is on standard out"
      echo "This is on standard err" >&2
      stop_logging
    }
    
    #example
    #example2
    example3
    

提交回复
热议问题