Watching something be written to a file live with tail Announcing the arrival of Valued Associate #679: Cesar Manara Planned maintenance scheduled April 23, 2019 at 23:30 UTC (7:30pm US/Eastern)Redirect output of remote python app started through ssh to fileUbuntu RFID Screensaver lock-unlockHow can I offer an XP-compatible migration to Linux Mint?Is there a command in Linux which waits till it will be terminated?Ctrl+c in a sub process is killing a nohup'ed process earlier in the scripthow to get cronjob running every minuteCtrl-C'd an in-place recursive gzip - is this likely to have broken anything?Getting output of another script while preserving line-breakssocat and rich terminal again“Ctrl + c” combination works different on different SSH clients

Why does BitLocker not use RSA?

How to infer difference of population proportion between two groups when proportion is small?

Does the main washing effect of soap come from foam?

Did any compiler fully use 80-bit floating point?

Do i imagine the linear (straight line) homotopy in a correct way?

How can I prevent/balance waiting and turtling as a response to cooldown mechanics

Where did Ptolemy compare the Earth to the distance of fixed stars?

Why do C and C++ allow the expression (int) + 4*5?

Was the pager message from Nick Fury to Captain Marvel unnecessary?

Is there a verb for listening stealthily?

Should man-made satellites feature an intelligent inverted "cow catcher"?

Is there a spell that can create a permanent fire?

Pointing to problems without suggesting solutions

How to achieve cat-like agility?

French equivalents of おしゃれは足元から (Every good outfit starts with the shoes)

What is the proper term for etching or digging of wall to hide conduit of cables

Improvising over quartal voicings

Table formatting with tabularx?

How to name indistinguishable henchmen in a screenplay?

Calculation of line of sight system gain

How do I say "this must not happen"?

How to ask rejected full-time candidates to apply to teach individual courses?

Did John Wesley plagiarize Matthew Henry...?

What did Turing mean when saying that "machines cannot give rise to surprises" is due to a fallacy?



Watching something be written to a file live with tail



Announcing the arrival of Valued Associate #679: Cesar Manara
Planned maintenance scheduled April 23, 2019 at 23:30 UTC (7:30pm US/Eastern)Redirect output of remote python app started through ssh to fileUbuntu RFID Screensaver lock-unlockHow can I offer an XP-compatible migration to Linux Mint?Is there a command in Linux which waits till it will be terminated?Ctrl+c in a sub process is killing a nohup'ed process earlier in the scripthow to get cronjob running every minuteCtrl-C'd an in-place recursive gzip - is this likely to have broken anything?Getting output of another script while preserving line-breakssocat and rich terminal again“Ctrl + c” combination works different on different SSH clients



.everyoneloves__top-leaderboard:empty,.everyoneloves__mid-leaderboard:empty,.everyoneloves__bot-mid-leaderboard:empty height:90px;width:728px;box-sizing:border-box;








25















I have a python program which is, slowly, generating some output.



I want to capture that in a file, but I also thought I could watch it live with tail.



So in one terminal I'm doing :



python myprog.py > output.txt


and in another terminal :



tail -f output.txt


But it seems like the tail isn't showing me anything while the python program is running.



If I hit ctrl-c to kill the python script, suddenly the tail of output.txt starts filling up. But not while python is running.



What am I doing wrong?










share|improve this question



















  • 10





    How about python myprog.py | tee output.txt instead?

    – n8te
    Apr 3 at 23:35






  • 8





    @n8te tee might show the same problem if the program isn't flushing the output buffer regularly. This needs flush() and tee.

    – JPhi1618
    Apr 4 at 2:42






  • 1





    stdbuf can be used to alter the buffering status of file descriptors.

    – studog
    Apr 4 at 18:41






  • 1





    Terminology: There is no pipe anywhere in this scenario. There's a redirect to a regular file. (Which causes C stdio and Python to decide to make stdout full-buffered instead of line-buffered because it's not a TTY). Pipes are a different type of file (a buffer inside the kernel). I edited your question to correct that.

    – Peter Cordes
    Apr 4 at 19:02







  • 2





    Probably not needed in your situation but if you don't want to terminate the program you can use gdb and call fflush: see stackoverflow.com/questions/8251269/…

    – Mark Wagner
    Apr 4 at 19:04

















25















I have a python program which is, slowly, generating some output.



I want to capture that in a file, but I also thought I could watch it live with tail.



So in one terminal I'm doing :



python myprog.py > output.txt


and in another terminal :



tail -f output.txt


But it seems like the tail isn't showing me anything while the python program is running.



If I hit ctrl-c to kill the python script, suddenly the tail of output.txt starts filling up. But not while python is running.



What am I doing wrong?










share|improve this question



















  • 10





    How about python myprog.py | tee output.txt instead?

    – n8te
    Apr 3 at 23:35






  • 8





    @n8te tee might show the same problem if the program isn't flushing the output buffer regularly. This needs flush() and tee.

    – JPhi1618
    Apr 4 at 2:42






  • 1





    stdbuf can be used to alter the buffering status of file descriptors.

    – studog
    Apr 4 at 18:41






  • 1





    Terminology: There is no pipe anywhere in this scenario. There's a redirect to a regular file. (Which causes C stdio and Python to decide to make stdout full-buffered instead of line-buffered because it's not a TTY). Pipes are a different type of file (a buffer inside the kernel). I edited your question to correct that.

    – Peter Cordes
    Apr 4 at 19:02







  • 2





    Probably not needed in your situation but if you don't want to terminate the program you can use gdb and call fflush: see stackoverflow.com/questions/8251269/…

    – Mark Wagner
    Apr 4 at 19:04













25












25








25


4






I have a python program which is, slowly, generating some output.



I want to capture that in a file, but I also thought I could watch it live with tail.



So in one terminal I'm doing :



python myprog.py > output.txt


and in another terminal :



tail -f output.txt


But it seems like the tail isn't showing me anything while the python program is running.



If I hit ctrl-c to kill the python script, suddenly the tail of output.txt starts filling up. But not while python is running.



What am I doing wrong?










share|improve this question
















I have a python program which is, slowly, generating some output.



I want to capture that in a file, but I also thought I could watch it live with tail.



So in one terminal I'm doing :



python myprog.py > output.txt


and in another terminal :



tail -f output.txt


But it seems like the tail isn't showing me anything while the python program is running.



If I hit ctrl-c to kill the python script, suddenly the tail of output.txt starts filling up. But not while python is running.



What am I doing wrong?







linux command-line redirection stdout






share|improve this question















share|improve this question













share|improve this question




share|improve this question








edited Apr 4 at 19:01









Peter Cordes

2,5451622




2,5451622










asked Apr 3 at 23:31









interstarinterstar

438613




438613







  • 10





    How about python myprog.py | tee output.txt instead?

    – n8te
    Apr 3 at 23:35






  • 8





    @n8te tee might show the same problem if the program isn't flushing the output buffer regularly. This needs flush() and tee.

    – JPhi1618
    Apr 4 at 2:42






  • 1





    stdbuf can be used to alter the buffering status of file descriptors.

    – studog
    Apr 4 at 18:41






  • 1





    Terminology: There is no pipe anywhere in this scenario. There's a redirect to a regular file. (Which causes C stdio and Python to decide to make stdout full-buffered instead of line-buffered because it's not a TTY). Pipes are a different type of file (a buffer inside the kernel). I edited your question to correct that.

    – Peter Cordes
    Apr 4 at 19:02







  • 2





    Probably not needed in your situation but if you don't want to terminate the program you can use gdb and call fflush: see stackoverflow.com/questions/8251269/…

    – Mark Wagner
    Apr 4 at 19:04












  • 10





    How about python myprog.py | tee output.txt instead?

    – n8te
    Apr 3 at 23:35






  • 8





    @n8te tee might show the same problem if the program isn't flushing the output buffer regularly. This needs flush() and tee.

    – JPhi1618
    Apr 4 at 2:42






  • 1





    stdbuf can be used to alter the buffering status of file descriptors.

    – studog
    Apr 4 at 18:41






  • 1





    Terminology: There is no pipe anywhere in this scenario. There's a redirect to a regular file. (Which causes C stdio and Python to decide to make stdout full-buffered instead of line-buffered because it's not a TTY). Pipes are a different type of file (a buffer inside the kernel). I edited your question to correct that.

    – Peter Cordes
    Apr 4 at 19:02







  • 2





    Probably not needed in your situation but if you don't want to terminate the program you can use gdb and call fflush: see stackoverflow.com/questions/8251269/…

    – Mark Wagner
    Apr 4 at 19:04







10




10





How about python myprog.py | tee output.txt instead?

– n8te
Apr 3 at 23:35





How about python myprog.py | tee output.txt instead?

– n8te
Apr 3 at 23:35




8




8





@n8te tee might show the same problem if the program isn't flushing the output buffer regularly. This needs flush() and tee.

– JPhi1618
Apr 4 at 2:42





@n8te tee might show the same problem if the program isn't flushing the output buffer regularly. This needs flush() and tee.

– JPhi1618
Apr 4 at 2:42




1




1





stdbuf can be used to alter the buffering status of file descriptors.

– studog
Apr 4 at 18:41





stdbuf can be used to alter the buffering status of file descriptors.

– studog
Apr 4 at 18:41




1




1





Terminology: There is no pipe anywhere in this scenario. There's a redirect to a regular file. (Which causes C stdio and Python to decide to make stdout full-buffered instead of line-buffered because it's not a TTY). Pipes are a different type of file (a buffer inside the kernel). I edited your question to correct that.

– Peter Cordes
Apr 4 at 19:02






Terminology: There is no pipe anywhere in this scenario. There's a redirect to a regular file. (Which causes C stdio and Python to decide to make stdout full-buffered instead of line-buffered because it's not a TTY). Pipes are a different type of file (a buffer inside the kernel). I edited your question to correct that.

– Peter Cordes
Apr 4 at 19:02





2




2





Probably not needed in your situation but if you don't want to terminate the program you can use gdb and call fflush: see stackoverflow.com/questions/8251269/…

– Mark Wagner
Apr 4 at 19:04





Probably not needed in your situation but if you don't want to terminate the program you can use gdb and call fflush: see stackoverflow.com/questions/8251269/…

– Mark Wagner
Apr 4 at 19:04










5 Answers
5






active

oldest

votes


















42














You may also need to explicitly flush the buffer for it to get piped upon generation. This is because output is typically only printed when the pipe's buffer fills up (which is in kilobytes I belive), and when the stdin message ends. This is probably to save on read/writes. You could do this after every print, or if you are looping, after the last print within the loop.



import sys
...
print('Some message')
sys.stdout.flush()





share|improve this answer




















  • 7





    If you have read this far, please don't be thinking of closing and re-opening the file to do this, the seeks will be a problem, especially for very large files. (I've seen this done!).

    – mckenzm
    Apr 4 at 3:46






  • 11





    You can also use print's flush parameter to do just as well. For example, print('some message', flush=True).

    – Dan
    Apr 4 at 11:48







  • 12





    It has nothing to do with the pipe's buffer, but with the stdout mechanism which doesn't flush after newline if it doesn't write to a tty.

    – glglgl
    Apr 4 at 13:38






  • 1





    Instead of add flush() calls to the program, you can also use Python's -u command line switch; see BHC's answer.

    – Roel Schroeven
    Apr 5 at 9:30






  • 2





    @wizzwizz4, it's only different in that Python follows the standard C library's convention of, by default, configuring stderr to be unbuffered and stdout to be line-buffered when they point to a TTY, but stdout to be unbuffered if it's opened to a non-TTY device.

    – Charles Duffy
    Apr 6 at 1:00


















33














Run python with the unbuffered flag:



python -u myprog.py > output.txt


Output will then print in real time.






share|improve this answer


















  • 4





    This is the correct answer. Python by default writes unbuffered (or actually line-buffered for text I/O) when writing to the console, but buffered when stdout is redirected to a file. -u forces Python to be unbuffered (or line-buffered for text) for writes.

    – Roel Schroeven
    Apr 5 at 9:29











  • Know nothing of python but when I read what was happening I thought "gee, sounds like it is buffering output...."

    – ivanivan
    Apr 5 at 22:23


















19














Instead of trying to tail a live file, use tee instead. It was made to do exactly what you're trying to do.



From man tee:




tee(1) - Linux man page



Name tee - read from standard input and write to standard output and files



Synopsis



tee [OPTION]... [FILE]...


Description



Copy standard input to each FILE, and also to standard output.



-a, --append 
append to the given FILEs, do not overwrite
-i, --ignore-interrupts
ignore interrupt signals
--help
display this help and exit
--version
output version information and exit


If a FILE is -, copy again to standard output.




So in your case you'd run:



python myprog.py | tee output.txt


EDIT: As others have pointed out, this answer will run into the same issue OP was originally having unless sys.stdout.flush() is used in the python program as described in Davey's accepted answer. The testing I did before posting this answer did not accurately reflect OP's use-case.



tee can still be used as an alternative--albeit less than optimal--method of displaying the output while also writing to the file, but Davey's answer is clearly the correct and best answer.






share|improve this answer




















  • 1





    tail in another thread is a good solution for when you've started the application before you decide you want to see the output though.

    – Baldrickk
    Apr 4 at 7:45






  • 11





    That requires a permanent console session, this is why it’s often much easier to use tail -F or even better the follow function of less. But in all cases the flush should be used.

    – eckes
    Apr 4 at 9:22






  • 8





    This won't solve the problem that the OP is having. Python's output to the pipe will be buffered just like output to the file.

    – Barmar
    Apr 4 at 17:46


















9














Terminology: There is no pipe anywhere in this scenario. (I edited the question to fix that). Pipes are a different type of file (a buffer inside the kernel).



This is a redirect to a regular file.



C stdio, and Python, default to making stdout line-buffered when it's connected to a TTY, otherwise it's full-buffered. Line-buffered means the buffer is flushed after a newline. Full-buffered means it's only flushed to become visible to the OS (i.e. with a write() system call) when it's full.



You will see output eventually, in chunks of maybe 4kiB at a time. (I don't know the default buffer size.) This is generally more efficient, and means fewer writes to your actual disk. But not great for interactive monitoring, because output is hidden inside the memory of the writing process until it's flushed.



On Stack Overflow, there's a Disable output buffering Python Q&A which lists many ways to get unbuffered (or line-buffered?) output to stdout in Python. The question itself summarizes the answers.



Options include running python -u (Or I guess putting #!/usr/bin/python -u at the top of your script), or using the PYTHONUNBUFFERED environment variable for that program. Or explicit flushing after some/all print functions, like @Davey's answer suggests.




Some other programs have similar options, e.g. GNU grep has --line-buffered, and GNU sed has -u / --unbuffered, for use-cases like this, or for example piping the output of your python program. e.g. ./slowly-output-stuff | grep --line-buffered 'foo.*bar'.






share|improve this answer






























    0














    When I use tail, it is almost-always tracking a log file, eg (email) messages.



    This might be slightly off-the-wall, but instead of using print/print()/write() in your Python code, why not use the logging module? (from the PSL) NB a logging formatter can be configured NOT to output all those time and ID-codes associated with a traditional log.



    The output can be configured to go to a (data) file, and because there is no buffering delay or redirection, tail works happily and with immediacy.



    Regards






    share|improve this answer

























      Your Answer








      StackExchange.ready(function()
      var channelOptions =
      tags: "".split(" "),
      id: "3"
      ;
      initTagRenderer("".split(" "), "".split(" "), channelOptions);

      StackExchange.using("externalEditor", function()
      // Have to fire editor after snippets, if snippets enabled
      if (StackExchange.settings.snippets.snippetsEnabled)
      StackExchange.using("snippets", function()
      createEditor();
      );

      else
      createEditor();

      );

      function createEditor()
      StackExchange.prepareEditor(
      heartbeatType: 'answer',
      autoActivateHeartbeat: false,
      convertImagesToLinks: true,
      noModals: true,
      showLowRepImageUploadWarning: true,
      reputationToPostImages: 10,
      bindNavPrevention: true,
      postfix: "",
      imageUploader:
      brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
      contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
      allowUrls: true
      ,
      onDemand: true,
      discardSelector: ".discard-answer"
      ,immediatelyShowMarkdownHelp:true
      );



      );













      draft saved

      draft discarded


















      StackExchange.ready(
      function ()
      StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fsuperuser.com%2fquestions%2f1421123%2fwatching-something-be-written-to-a-file-live-with-tail%23new-answer', 'question_page');

      );

      Post as a guest















      Required, but never shown

























      5 Answers
      5






      active

      oldest

      votes








      5 Answers
      5






      active

      oldest

      votes









      active

      oldest

      votes






      active

      oldest

      votes









      42














      You may also need to explicitly flush the buffer for it to get piped upon generation. This is because output is typically only printed when the pipe's buffer fills up (which is in kilobytes I belive), and when the stdin message ends. This is probably to save on read/writes. You could do this after every print, or if you are looping, after the last print within the loop.



      import sys
      ...
      print('Some message')
      sys.stdout.flush()





      share|improve this answer




















      • 7





        If you have read this far, please don't be thinking of closing and re-opening the file to do this, the seeks will be a problem, especially for very large files. (I've seen this done!).

        – mckenzm
        Apr 4 at 3:46






      • 11





        You can also use print's flush parameter to do just as well. For example, print('some message', flush=True).

        – Dan
        Apr 4 at 11:48







      • 12





        It has nothing to do with the pipe's buffer, but with the stdout mechanism which doesn't flush after newline if it doesn't write to a tty.

        – glglgl
        Apr 4 at 13:38






      • 1





        Instead of add flush() calls to the program, you can also use Python's -u command line switch; see BHC's answer.

        – Roel Schroeven
        Apr 5 at 9:30






      • 2





        @wizzwizz4, it's only different in that Python follows the standard C library's convention of, by default, configuring stderr to be unbuffered and stdout to be line-buffered when they point to a TTY, but stdout to be unbuffered if it's opened to a non-TTY device.

        – Charles Duffy
        Apr 6 at 1:00















      42














      You may also need to explicitly flush the buffer for it to get piped upon generation. This is because output is typically only printed when the pipe's buffer fills up (which is in kilobytes I belive), and when the stdin message ends. This is probably to save on read/writes. You could do this after every print, or if you are looping, after the last print within the loop.



      import sys
      ...
      print('Some message')
      sys.stdout.flush()





      share|improve this answer




















      • 7





        If you have read this far, please don't be thinking of closing and re-opening the file to do this, the seeks will be a problem, especially for very large files. (I've seen this done!).

        – mckenzm
        Apr 4 at 3:46






      • 11





        You can also use print's flush parameter to do just as well. For example, print('some message', flush=True).

        – Dan
        Apr 4 at 11:48







      • 12





        It has nothing to do with the pipe's buffer, but with the stdout mechanism which doesn't flush after newline if it doesn't write to a tty.

        – glglgl
        Apr 4 at 13:38






      • 1





        Instead of add flush() calls to the program, you can also use Python's -u command line switch; see BHC's answer.

        – Roel Schroeven
        Apr 5 at 9:30






      • 2





        @wizzwizz4, it's only different in that Python follows the standard C library's convention of, by default, configuring stderr to be unbuffered and stdout to be line-buffered when they point to a TTY, but stdout to be unbuffered if it's opened to a non-TTY device.

        – Charles Duffy
        Apr 6 at 1:00













      42












      42








      42







      You may also need to explicitly flush the buffer for it to get piped upon generation. This is because output is typically only printed when the pipe's buffer fills up (which is in kilobytes I belive), and when the stdin message ends. This is probably to save on read/writes. You could do this after every print, or if you are looping, after the last print within the loop.



      import sys
      ...
      print('Some message')
      sys.stdout.flush()





      share|improve this answer















      You may also need to explicitly flush the buffer for it to get piped upon generation. This is because output is typically only printed when the pipe's buffer fills up (which is in kilobytes I belive), and when the stdin message ends. This is probably to save on read/writes. You could do this after every print, or if you are looping, after the last print within the loop.



      import sys
      ...
      print('Some message')
      sys.stdout.flush()






      share|improve this answer














      share|improve this answer



      share|improve this answer








      edited Apr 4 at 4:35









      user2313067

      2,1001911




      2,1001911










      answered Apr 4 at 0:05









      DaveyDavey

      53938




      53938







      • 7





        If you have read this far, please don't be thinking of closing and re-opening the file to do this, the seeks will be a problem, especially for very large files. (I've seen this done!).

        – mckenzm
        Apr 4 at 3:46






      • 11





        You can also use print's flush parameter to do just as well. For example, print('some message', flush=True).

        – Dan
        Apr 4 at 11:48







      • 12





        It has nothing to do with the pipe's buffer, but with the stdout mechanism which doesn't flush after newline if it doesn't write to a tty.

        – glglgl
        Apr 4 at 13:38






      • 1





        Instead of add flush() calls to the program, you can also use Python's -u command line switch; see BHC's answer.

        – Roel Schroeven
        Apr 5 at 9:30






      • 2





        @wizzwizz4, it's only different in that Python follows the standard C library's convention of, by default, configuring stderr to be unbuffered and stdout to be line-buffered when they point to a TTY, but stdout to be unbuffered if it's opened to a non-TTY device.

        – Charles Duffy
        Apr 6 at 1:00












      • 7





        If you have read this far, please don't be thinking of closing and re-opening the file to do this, the seeks will be a problem, especially for very large files. (I've seen this done!).

        – mckenzm
        Apr 4 at 3:46






      • 11





        You can also use print's flush parameter to do just as well. For example, print('some message', flush=True).

        – Dan
        Apr 4 at 11:48







      • 12





        It has nothing to do with the pipe's buffer, but with the stdout mechanism which doesn't flush after newline if it doesn't write to a tty.

        – glglgl
        Apr 4 at 13:38






      • 1





        Instead of add flush() calls to the program, you can also use Python's -u command line switch; see BHC's answer.

        – Roel Schroeven
        Apr 5 at 9:30






      • 2





        @wizzwizz4, it's only different in that Python follows the standard C library's convention of, by default, configuring stderr to be unbuffered and stdout to be line-buffered when they point to a TTY, but stdout to be unbuffered if it's opened to a non-TTY device.

        – Charles Duffy
        Apr 6 at 1:00







      7




      7





      If you have read this far, please don't be thinking of closing and re-opening the file to do this, the seeks will be a problem, especially for very large files. (I've seen this done!).

      – mckenzm
      Apr 4 at 3:46





      If you have read this far, please don't be thinking of closing and re-opening the file to do this, the seeks will be a problem, especially for very large files. (I've seen this done!).

      – mckenzm
      Apr 4 at 3:46




      11




      11





      You can also use print's flush parameter to do just as well. For example, print('some message', flush=True).

      – Dan
      Apr 4 at 11:48






      You can also use print's flush parameter to do just as well. For example, print('some message', flush=True).

      – Dan
      Apr 4 at 11:48





      12




      12





      It has nothing to do with the pipe's buffer, but with the stdout mechanism which doesn't flush after newline if it doesn't write to a tty.

      – glglgl
      Apr 4 at 13:38





      It has nothing to do with the pipe's buffer, but with the stdout mechanism which doesn't flush after newline if it doesn't write to a tty.

      – glglgl
      Apr 4 at 13:38




      1




      1





      Instead of add flush() calls to the program, you can also use Python's -u command line switch; see BHC's answer.

      – Roel Schroeven
      Apr 5 at 9:30





      Instead of add flush() calls to the program, you can also use Python's -u command line switch; see BHC's answer.

      – Roel Schroeven
      Apr 5 at 9:30




      2




      2





      @wizzwizz4, it's only different in that Python follows the standard C library's convention of, by default, configuring stderr to be unbuffered and stdout to be line-buffered when they point to a TTY, but stdout to be unbuffered if it's opened to a non-TTY device.

      – Charles Duffy
      Apr 6 at 1:00





      @wizzwizz4, it's only different in that Python follows the standard C library's convention of, by default, configuring stderr to be unbuffered and stdout to be line-buffered when they point to a TTY, but stdout to be unbuffered if it's opened to a non-TTY device.

      – Charles Duffy
      Apr 6 at 1:00













      33














      Run python with the unbuffered flag:



      python -u myprog.py > output.txt


      Output will then print in real time.






      share|improve this answer


















      • 4





        This is the correct answer. Python by default writes unbuffered (or actually line-buffered for text I/O) when writing to the console, but buffered when stdout is redirected to a file. -u forces Python to be unbuffered (or line-buffered for text) for writes.

        – Roel Schroeven
        Apr 5 at 9:29











      • Know nothing of python but when I read what was happening I thought "gee, sounds like it is buffering output...."

        – ivanivan
        Apr 5 at 22:23















      33














      Run python with the unbuffered flag:



      python -u myprog.py > output.txt


      Output will then print in real time.






      share|improve this answer


















      • 4





        This is the correct answer. Python by default writes unbuffered (or actually line-buffered for text I/O) when writing to the console, but buffered when stdout is redirected to a file. -u forces Python to be unbuffered (or line-buffered for text) for writes.

        – Roel Schroeven
        Apr 5 at 9:29











      • Know nothing of python but when I read what was happening I thought "gee, sounds like it is buffering output...."

        – ivanivan
        Apr 5 at 22:23













      33












      33








      33







      Run python with the unbuffered flag:



      python -u myprog.py > output.txt


      Output will then print in real time.






      share|improve this answer













      Run python with the unbuffered flag:



      python -u myprog.py > output.txt


      Output will then print in real time.







      share|improve this answer












      share|improve this answer



      share|improve this answer










      answered Apr 4 at 16:14









      BHCBHC

      43913




      43913







      • 4





        This is the correct answer. Python by default writes unbuffered (or actually line-buffered for text I/O) when writing to the console, but buffered when stdout is redirected to a file. -u forces Python to be unbuffered (or line-buffered for text) for writes.

        – Roel Schroeven
        Apr 5 at 9:29











      • Know nothing of python but when I read what was happening I thought "gee, sounds like it is buffering output...."

        – ivanivan
        Apr 5 at 22:23












      • 4





        This is the correct answer. Python by default writes unbuffered (or actually line-buffered for text I/O) when writing to the console, but buffered when stdout is redirected to a file. -u forces Python to be unbuffered (or line-buffered for text) for writes.

        – Roel Schroeven
        Apr 5 at 9:29











      • Know nothing of python but when I read what was happening I thought "gee, sounds like it is buffering output...."

        – ivanivan
        Apr 5 at 22:23







      4




      4





      This is the correct answer. Python by default writes unbuffered (or actually line-buffered for text I/O) when writing to the console, but buffered when stdout is redirected to a file. -u forces Python to be unbuffered (or line-buffered for text) for writes.

      – Roel Schroeven
      Apr 5 at 9:29





      This is the correct answer. Python by default writes unbuffered (or actually line-buffered for text I/O) when writing to the console, but buffered when stdout is redirected to a file. -u forces Python to be unbuffered (or line-buffered for text) for writes.

      – Roel Schroeven
      Apr 5 at 9:29













      Know nothing of python but when I read what was happening I thought "gee, sounds like it is buffering output...."

      – ivanivan
      Apr 5 at 22:23





      Know nothing of python but when I read what was happening I thought "gee, sounds like it is buffering output...."

      – ivanivan
      Apr 5 at 22:23











      19














      Instead of trying to tail a live file, use tee instead. It was made to do exactly what you're trying to do.



      From man tee:




      tee(1) - Linux man page



      Name tee - read from standard input and write to standard output and files



      Synopsis



      tee [OPTION]... [FILE]...


      Description



      Copy standard input to each FILE, and also to standard output.



      -a, --append 
      append to the given FILEs, do not overwrite
      -i, --ignore-interrupts
      ignore interrupt signals
      --help
      display this help and exit
      --version
      output version information and exit


      If a FILE is -, copy again to standard output.




      So in your case you'd run:



      python myprog.py | tee output.txt


      EDIT: As others have pointed out, this answer will run into the same issue OP was originally having unless sys.stdout.flush() is used in the python program as described in Davey's accepted answer. The testing I did before posting this answer did not accurately reflect OP's use-case.



      tee can still be used as an alternative--albeit less than optimal--method of displaying the output while also writing to the file, but Davey's answer is clearly the correct and best answer.






      share|improve this answer




















      • 1





        tail in another thread is a good solution for when you've started the application before you decide you want to see the output though.

        – Baldrickk
        Apr 4 at 7:45






      • 11





        That requires a permanent console session, this is why it’s often much easier to use tail -F or even better the follow function of less. But in all cases the flush should be used.

        – eckes
        Apr 4 at 9:22






      • 8





        This won't solve the problem that the OP is having. Python's output to the pipe will be buffered just like output to the file.

        – Barmar
        Apr 4 at 17:46















      19














      Instead of trying to tail a live file, use tee instead. It was made to do exactly what you're trying to do.



      From man tee:




      tee(1) - Linux man page



      Name tee - read from standard input and write to standard output and files



      Synopsis



      tee [OPTION]... [FILE]...


      Description



      Copy standard input to each FILE, and also to standard output.



      -a, --append 
      append to the given FILEs, do not overwrite
      -i, --ignore-interrupts
      ignore interrupt signals
      --help
      display this help and exit
      --version
      output version information and exit


      If a FILE is -, copy again to standard output.




      So in your case you'd run:



      python myprog.py | tee output.txt


      EDIT: As others have pointed out, this answer will run into the same issue OP was originally having unless sys.stdout.flush() is used in the python program as described in Davey's accepted answer. The testing I did before posting this answer did not accurately reflect OP's use-case.



      tee can still be used as an alternative--albeit less than optimal--method of displaying the output while also writing to the file, but Davey's answer is clearly the correct and best answer.






      share|improve this answer




















      • 1





        tail in another thread is a good solution for when you've started the application before you decide you want to see the output though.

        – Baldrickk
        Apr 4 at 7:45






      • 11





        That requires a permanent console session, this is why it’s often much easier to use tail -F or even better the follow function of less. But in all cases the flush should be used.

        – eckes
        Apr 4 at 9:22






      • 8





        This won't solve the problem that the OP is having. Python's output to the pipe will be buffered just like output to the file.

        – Barmar
        Apr 4 at 17:46













      19












      19








      19







      Instead of trying to tail a live file, use tee instead. It was made to do exactly what you're trying to do.



      From man tee:




      tee(1) - Linux man page



      Name tee - read from standard input and write to standard output and files



      Synopsis



      tee [OPTION]... [FILE]...


      Description



      Copy standard input to each FILE, and also to standard output.



      -a, --append 
      append to the given FILEs, do not overwrite
      -i, --ignore-interrupts
      ignore interrupt signals
      --help
      display this help and exit
      --version
      output version information and exit


      If a FILE is -, copy again to standard output.




      So in your case you'd run:



      python myprog.py | tee output.txt


      EDIT: As others have pointed out, this answer will run into the same issue OP was originally having unless sys.stdout.flush() is used in the python program as described in Davey's accepted answer. The testing I did before posting this answer did not accurately reflect OP's use-case.



      tee can still be used as an alternative--albeit less than optimal--method of displaying the output while also writing to the file, but Davey's answer is clearly the correct and best answer.






      share|improve this answer















      Instead of trying to tail a live file, use tee instead. It was made to do exactly what you're trying to do.



      From man tee:




      tee(1) - Linux man page



      Name tee - read from standard input and write to standard output and files



      Synopsis



      tee [OPTION]... [FILE]...


      Description



      Copy standard input to each FILE, and also to standard output.



      -a, --append 
      append to the given FILEs, do not overwrite
      -i, --ignore-interrupts
      ignore interrupt signals
      --help
      display this help and exit
      --version
      output version information and exit


      If a FILE is -, copy again to standard output.




      So in your case you'd run:



      python myprog.py | tee output.txt


      EDIT: As others have pointed out, this answer will run into the same issue OP was originally having unless sys.stdout.flush() is used in the python program as described in Davey's accepted answer. The testing I did before posting this answer did not accurately reflect OP's use-case.



      tee can still be used as an alternative--albeit less than optimal--method of displaying the output while also writing to the file, but Davey's answer is clearly the correct and best answer.







      share|improve this answer














      share|improve this answer



      share|improve this answer








      edited Apr 4 at 21:29

























      answered Apr 4 at 0:11









      n8ten8te

      5,42172235




      5,42172235







      • 1





        tail in another thread is a good solution for when you've started the application before you decide you want to see the output though.

        – Baldrickk
        Apr 4 at 7:45






      • 11





        That requires a permanent console session, this is why it’s often much easier to use tail -F or even better the follow function of less. But in all cases the flush should be used.

        – eckes
        Apr 4 at 9:22






      • 8





        This won't solve the problem that the OP is having. Python's output to the pipe will be buffered just like output to the file.

        – Barmar
        Apr 4 at 17:46












      • 1





        tail in another thread is a good solution for when you've started the application before you decide you want to see the output though.

        – Baldrickk
        Apr 4 at 7:45






      • 11





        That requires a permanent console session, this is why it’s often much easier to use tail -F or even better the follow function of less. But in all cases the flush should be used.

        – eckes
        Apr 4 at 9:22






      • 8





        This won't solve the problem that the OP is having. Python's output to the pipe will be buffered just like output to the file.

        – Barmar
        Apr 4 at 17:46







      1




      1





      tail in another thread is a good solution for when you've started the application before you decide you want to see the output though.

      – Baldrickk
      Apr 4 at 7:45





      tail in another thread is a good solution for when you've started the application before you decide you want to see the output though.

      – Baldrickk
      Apr 4 at 7:45




      11




      11





      That requires a permanent console session, this is why it’s often much easier to use tail -F or even better the follow function of less. But in all cases the flush should be used.

      – eckes
      Apr 4 at 9:22





      That requires a permanent console session, this is why it’s often much easier to use tail -F or even better the follow function of less. But in all cases the flush should be used.

      – eckes
      Apr 4 at 9:22




      8




      8





      This won't solve the problem that the OP is having. Python's output to the pipe will be buffered just like output to the file.

      – Barmar
      Apr 4 at 17:46





      This won't solve the problem that the OP is having. Python's output to the pipe will be buffered just like output to the file.

      – Barmar
      Apr 4 at 17:46











      9














      Terminology: There is no pipe anywhere in this scenario. (I edited the question to fix that). Pipes are a different type of file (a buffer inside the kernel).



      This is a redirect to a regular file.



      C stdio, and Python, default to making stdout line-buffered when it's connected to a TTY, otherwise it's full-buffered. Line-buffered means the buffer is flushed after a newline. Full-buffered means it's only flushed to become visible to the OS (i.e. with a write() system call) when it's full.



      You will see output eventually, in chunks of maybe 4kiB at a time. (I don't know the default buffer size.) This is generally more efficient, and means fewer writes to your actual disk. But not great for interactive monitoring, because output is hidden inside the memory of the writing process until it's flushed.



      On Stack Overflow, there's a Disable output buffering Python Q&A which lists many ways to get unbuffered (or line-buffered?) output to stdout in Python. The question itself summarizes the answers.



      Options include running python -u (Or I guess putting #!/usr/bin/python -u at the top of your script), or using the PYTHONUNBUFFERED environment variable for that program. Or explicit flushing after some/all print functions, like @Davey's answer suggests.




      Some other programs have similar options, e.g. GNU grep has --line-buffered, and GNU sed has -u / --unbuffered, for use-cases like this, or for example piping the output of your python program. e.g. ./slowly-output-stuff | grep --line-buffered 'foo.*bar'.






      share|improve this answer



























        9














        Terminology: There is no pipe anywhere in this scenario. (I edited the question to fix that). Pipes are a different type of file (a buffer inside the kernel).



        This is a redirect to a regular file.



        C stdio, and Python, default to making stdout line-buffered when it's connected to a TTY, otherwise it's full-buffered. Line-buffered means the buffer is flushed after a newline. Full-buffered means it's only flushed to become visible to the OS (i.e. with a write() system call) when it's full.



        You will see output eventually, in chunks of maybe 4kiB at a time. (I don't know the default buffer size.) This is generally more efficient, and means fewer writes to your actual disk. But not great for interactive monitoring, because output is hidden inside the memory of the writing process until it's flushed.



        On Stack Overflow, there's a Disable output buffering Python Q&A which lists many ways to get unbuffered (or line-buffered?) output to stdout in Python. The question itself summarizes the answers.



        Options include running python -u (Or I guess putting #!/usr/bin/python -u at the top of your script), or using the PYTHONUNBUFFERED environment variable for that program. Or explicit flushing after some/all print functions, like @Davey's answer suggests.




        Some other programs have similar options, e.g. GNU grep has --line-buffered, and GNU sed has -u / --unbuffered, for use-cases like this, or for example piping the output of your python program. e.g. ./slowly-output-stuff | grep --line-buffered 'foo.*bar'.






        share|improve this answer

























          9












          9








          9







          Terminology: There is no pipe anywhere in this scenario. (I edited the question to fix that). Pipes are a different type of file (a buffer inside the kernel).



          This is a redirect to a regular file.



          C stdio, and Python, default to making stdout line-buffered when it's connected to a TTY, otherwise it's full-buffered. Line-buffered means the buffer is flushed after a newline. Full-buffered means it's only flushed to become visible to the OS (i.e. with a write() system call) when it's full.



          You will see output eventually, in chunks of maybe 4kiB at a time. (I don't know the default buffer size.) This is generally more efficient, and means fewer writes to your actual disk. But not great for interactive monitoring, because output is hidden inside the memory of the writing process until it's flushed.



          On Stack Overflow, there's a Disable output buffering Python Q&A which lists many ways to get unbuffered (or line-buffered?) output to stdout in Python. The question itself summarizes the answers.



          Options include running python -u (Or I guess putting #!/usr/bin/python -u at the top of your script), or using the PYTHONUNBUFFERED environment variable for that program. Or explicit flushing after some/all print functions, like @Davey's answer suggests.




          Some other programs have similar options, e.g. GNU grep has --line-buffered, and GNU sed has -u / --unbuffered, for use-cases like this, or for example piping the output of your python program. e.g. ./slowly-output-stuff | grep --line-buffered 'foo.*bar'.






          share|improve this answer













          Terminology: There is no pipe anywhere in this scenario. (I edited the question to fix that). Pipes are a different type of file (a buffer inside the kernel).



          This is a redirect to a regular file.



          C stdio, and Python, default to making stdout line-buffered when it's connected to a TTY, otherwise it's full-buffered. Line-buffered means the buffer is flushed after a newline. Full-buffered means it's only flushed to become visible to the OS (i.e. with a write() system call) when it's full.



          You will see output eventually, in chunks of maybe 4kiB at a time. (I don't know the default buffer size.) This is generally more efficient, and means fewer writes to your actual disk. But not great for interactive monitoring, because output is hidden inside the memory of the writing process until it's flushed.



          On Stack Overflow, there's a Disable output buffering Python Q&A which lists many ways to get unbuffered (or line-buffered?) output to stdout in Python. The question itself summarizes the answers.



          Options include running python -u (Or I guess putting #!/usr/bin/python -u at the top of your script), or using the PYTHONUNBUFFERED environment variable for that program. Or explicit flushing after some/all print functions, like @Davey's answer suggests.




          Some other programs have similar options, e.g. GNU grep has --line-buffered, and GNU sed has -u / --unbuffered, for use-cases like this, or for example piping the output of your python program. e.g. ./slowly-output-stuff | grep --line-buffered 'foo.*bar'.







          share|improve this answer












          share|improve this answer



          share|improve this answer










          answered Apr 4 at 19:18









          Peter CordesPeter Cordes

          2,5451622




          2,5451622





















              0














              When I use tail, it is almost-always tracking a log file, eg (email) messages.



              This might be slightly off-the-wall, but instead of using print/print()/write() in your Python code, why not use the logging module? (from the PSL) NB a logging formatter can be configured NOT to output all those time and ID-codes associated with a traditional log.



              The output can be configured to go to a (data) file, and because there is no buffering delay or redirection, tail works happily and with immediacy.



              Regards






              share|improve this answer





























                0














                When I use tail, it is almost-always tracking a log file, eg (email) messages.



                This might be slightly off-the-wall, but instead of using print/print()/write() in your Python code, why not use the logging module? (from the PSL) NB a logging formatter can be configured NOT to output all those time and ID-codes associated with a traditional log.



                The output can be configured to go to a (data) file, and because there is no buffering delay or redirection, tail works happily and with immediacy.



                Regards






                share|improve this answer



























                  0












                  0








                  0







                  When I use tail, it is almost-always tracking a log file, eg (email) messages.



                  This might be slightly off-the-wall, but instead of using print/print()/write() in your Python code, why not use the logging module? (from the PSL) NB a logging formatter can be configured NOT to output all those time and ID-codes associated with a traditional log.



                  The output can be configured to go to a (data) file, and because there is no buffering delay or redirection, tail works happily and with immediacy.



                  Regards






                  share|improve this answer















                  When I use tail, it is almost-always tracking a log file, eg (email) messages.



                  This might be slightly off-the-wall, but instead of using print/print()/write() in your Python code, why not use the logging module? (from the PSL) NB a logging formatter can be configured NOT to output all those time and ID-codes associated with a traditional log.



                  The output can be configured to go to a (data) file, and because there is no buffering delay or redirection, tail works happily and with immediacy.



                  Regards







                  share|improve this answer














                  share|improve this answer



                  share|improve this answer








                  edited Apr 8 at 22:57









                  Alfabravo

                  528413




                  528413










                  answered Apr 8 at 20:29









                  d-nd-n

                  11




                  11



























                      draft saved

                      draft discarded
















































                      Thanks for contributing an answer to Super User!


                      • Please be sure to answer the question. Provide details and share your research!

                      But avoid


                      • Asking for help, clarification, or responding to other answers.

                      • Making statements based on opinion; back them up with references or personal experience.

                      To learn more, see our tips on writing great answers.




                      draft saved


                      draft discarded














                      StackExchange.ready(
                      function ()
                      StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fsuperuser.com%2fquestions%2f1421123%2fwatching-something-be-written-to-a-file-live-with-tail%23new-answer', 'question_page');

                      );

                      Post as a guest















                      Required, but never shown





















































                      Required, but never shown














                      Required, but never shown












                      Required, but never shown







                      Required, but never shown

































                      Required, but never shown














                      Required, but never shown












                      Required, but never shown







                      Required, but never shown







                      Popular posts from this blog

                      Adding axes to figuresAdding axes labels to LaTeX figuresLaTeX equivalent of ConTeXt buffersRotate a node but not its content: the case of the ellipse decorationHow to define the default vertical distance between nodes?TikZ scaling graphic and adjust node position and keep font sizeNumerical conditional within tikz keys?adding axes to shapesAlign axes across subfiguresAdding figures with a certain orderLine up nested tikz enviroments or how to get rid of themAdding axes labels to LaTeX figures

                      Tähtien Talli Jäsenet | Lähteet | NavigointivalikkoSuomen Hippos – Tähtien Talli

                      Do these cracks on my tires look bad? The Next CEO of Stack OverflowDry rot tire should I replace?Having to replace tiresFishtailed so easily? Bad tires? ABS?Filling the tires with something other than air, to avoid puncture hassles?Used Michelin tires safe to install?Do these tyre cracks necessitate replacement?Rumbling noise: tires or mechanicalIs it possible to fix noisy feathered tires?Are bad winter tires still better than summer tires in winter?Torque converter failure - Related to replacing only 2 tires?Why use snow tires on all 4 wheels on 2-wheel-drive cars?